diff --git a/backend/services/reporting-service/.dockerignore b/backend/services/reporting-service/.dockerignore new file mode 100644 index 00000000..3fc814ab --- /dev/null +++ b/backend/services/reporting-service/.dockerignore @@ -0,0 +1,8 @@ +node_modules +dist +coverage +.git +.env* +!.env.example +*.log +.claude/ diff --git a/backend/services/reporting-service/.env.example b/backend/services/reporting-service/.env.example new file mode 100644 index 00000000..63266a66 --- /dev/null +++ b/backend/services/reporting-service/.env.example @@ -0,0 +1,39 @@ +# 应用配置 +NODE_ENV=development +PORT=3008 +APP_NAME=reporting-service + +# 数据库 +DATABASE_URL="postgresql://postgres:postgres@localhost:5432/rwadurian_reporting?schema=public" + +# JWT (与 identity-service 共享密钥) +JWT_SECRET=your-super-secret-jwt-key-change-in-production +JWT_ACCESS_EXPIRES_IN=2h + +# Redis +REDIS_HOST=localhost +REDIS_PORT=6379 +REDIS_PASSWORD= + +# Kafka +KAFKA_BROKERS=localhost:9092 +KAFKA_GROUP_ID=reporting-service-group +KAFKA_CLIENT_ID=reporting-service + +# 外部服务 +IDENTITY_SERVICE_URL=http://localhost:3001 +PLANTING_SERVICE_URL=http://localhost:3003 +REFERRAL_SERVICE_URL=http://localhost:3004 +REWARD_SERVICE_URL=http://localhost:3005 +LEADERBOARD_SERVICE_URL=http://localhost:3007 +WALLET_SERVICE_URL=http://localhost:3002 + +# 文件存储 +FILE_STORAGE_PATH=./storage/reports +FILE_STORAGE_URL_PREFIX=http://localhost:3008/files + +# 报表缓存过期时间(秒) +REPORT_CACHE_TTL=3600 + +# 报表快照保留天数 +SNAPSHOT_RETENTION_DAYS=90 diff --git a/backend/services/reporting-service/.gitignore b/backend/services/reporting-service/.gitignore new file mode 100644 index 00000000..9269eb31 --- /dev/null +++ b/backend/services/reporting-service/.gitignore @@ -0,0 +1,38 @@ +# Dependencies +node_modules/ + +# Build output +dist/ + +# Test coverage +coverage/ + +# Environment files +.env +.env.local +.env.*.local +!.env.example + +# IDE +.idea/ +.vscode/ +*.swp +*.swo + +# OS +.DS_Store +Thumbs.db + +# Logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* + +# Claude Code +.claude/ + +# Temporary files +*.tmp +*.temp +.cache/ diff --git a/backend/services/reporting-service/Dockerfile.test b/backend/services/reporting-service/Dockerfile.test new file mode 100644 index 00000000..84d60ee0 --- /dev/null +++ b/backend/services/reporting-service/Dockerfile.test @@ -0,0 +1,26 @@ +# Test Dockerfile +FROM node:20-alpine + +WORKDIR /app + +# Install OpenSSL for Prisma compatibility +RUN apk add --no-cache openssl openssl-dev + +# Install dependencies first for caching +COPY package*.json ./ +RUN npm ci + +# Copy Prisma schema +COPY prisma ./prisma/ + +# Generate Prisma Client +RUN npx prisma generate + +# Copy source code +COPY . . + +# Build the application +RUN npm run build + +# Default command runs tests +CMD ["npm", "test"] diff --git a/backend/services/reporting-service/Makefile b/backend/services/reporting-service/Makefile new file mode 100644 index 00000000..e910be07 --- /dev/null +++ b/backend/services/reporting-service/Makefile @@ -0,0 +1,133 @@ +.PHONY: install build test test-unit test-integration test-e2e test-cov test-docker test-docker-all clean lint format db-migrate db-seed help + +# Variables +DOCKER_COMPOSE_TEST = docker compose -f docker-compose.test.yml + +# Help +help: + @echo "Available commands:" + @echo " make install - Install dependencies" + @echo " make build - Build the project" + @echo " make test - Run all tests" + @echo " make test-unit - Run unit tests only" + @echo " make test-integration - Run integration tests" + @echo " make test-e2e - Run end-to-end tests" + @echo " make test-cov - Run tests with coverage" + @echo " make test-docker - Run tests in Docker" + @echo " make test-docker-all - Run all tests in Docker" + @echo " make lint - Run linter" + @echo " make format - Format code" + @echo " make db-start - Start test database" + @echo " make db-stop - Stop test database" + @echo " make db-migrate - Run database migrations" + @echo " make db-seed - Seed database" + @echo " make clean - Clean build artifacts" + +# Install dependencies +install: + npm ci + npx prisma generate + +# Build +build: + npm run build + +# Lint +lint: + npm run lint + +# Format +format: + npm run format + +# Clean +clean: + rm -rf dist coverage node_modules/.cache + $(DOCKER_COMPOSE_TEST) down -v --remove-orphans 2>/dev/null || true + +############################### +# Testing +############################### + +# Run all tests (unit tests) +test: + npm test + +# Run unit tests only (domain and application layer) +test-unit: + npm test -- --testPathPattern="(spec|test)\\.ts$$" --testPathIgnorePatterns="e2e" + +# Run integration tests (requires database) +test-integration: db-start db-migrate + DATABASE_URL="postgresql://postgres:postgres@localhost:5433/rwadurian_reporting_test?schema=public" \ + REDIS_HOST=localhost \ + REDIS_PORT=6380 \ + npm test -- --testPathPattern="integration" --runInBand + @$(MAKE) db-stop + +# Run e2e tests (requires full stack) +test-e2e: db-start db-migrate + DATABASE_URL="postgresql://postgres:postgres@localhost:5433/rwadurian_reporting_test?schema=public" \ + REDIS_HOST=localhost \ + REDIS_PORT=6380 \ + npm run test:e2e -- --runInBand + @$(MAKE) db-stop + +# Run tests with coverage +test-cov: + npm run test:cov + +############################### +# Docker Testing +############################### + +# Start test infrastructure (postgres + redis) +db-start: + $(DOCKER_COMPOSE_TEST) up -d postgres-test redis-test + @echo "Waiting for services to be healthy..." + @sleep 5 + +# Stop test infrastructure +db-stop: + $(DOCKER_COMPOSE_TEST) down -v + +# Run database migrations +db-migrate: + DATABASE_URL="postgresql://postgres:postgres@localhost:5433/rwadurian_reporting_test?schema=public" \ + npx prisma migrate deploy 2>/dev/null || \ + DATABASE_URL="postgresql://postgres:postgres@localhost:5433/rwadurian_reporting_test?schema=public" \ + npx prisma db push --skip-generate + +# Seed database +db-seed: + DATABASE_URL="postgresql://postgres:postgres@localhost:5433/rwadurian_reporting_test?schema=public" \ + npx ts-node prisma/seed.ts + +# Build test Docker image +docker-build-test: + docker build -f Dockerfile.test -t reporting-service-test . + +# Run unit tests in Docker +test-docker: docker-build-test + $(DOCKER_COMPOSE_TEST) run --rm reporting-service-test npm run test + +# Run all tests in Docker (unit + integration + e2e) +test-docker-all: + $(DOCKER_COMPOSE_TEST) up --build --abort-on-container-exit --exit-code-from reporting-service-test + $(DOCKER_COMPOSE_TEST) down -v + +# Run tests with coverage in Docker +test-docker-cov: + $(DOCKER_COMPOSE_TEST) run --rm reporting-service-test npm run test:cov + $(DOCKER_COMPOSE_TEST) down -v + +############################### +# CI/CD Helpers +############################### + +# CI test command +ci-test: install test-cov lint + +# Full test suite +full-test: clean install lint test-unit test-docker-all + @echo "All tests passed!" diff --git a/backend/services/reporting-service/docker-compose.test.yml b/backend/services/reporting-service/docker-compose.test.yml new file mode 100644 index 00000000..05cfc537 --- /dev/null +++ b/backend/services/reporting-service/docker-compose.test.yml @@ -0,0 +1,50 @@ +version: '3.8' + +services: + postgres-test: + image: postgres:15-alpine + container_name: reporting-postgres-test + environment: + POSTGRES_USER: postgres + POSTGRES_PASSWORD: postgres + POSTGRES_DB: rwadurian_reporting_test + ports: + - "5433:5432" + healthcheck: + test: ["CMD-SHELL", "pg_isready -U postgres"] + interval: 5s + timeout: 5s + retries: 5 + tmpfs: + - /var/lib/postgresql/data + + redis-test: + image: redis:7-alpine + container_name: reporting-redis-test + ports: + - "6380:6379" + healthcheck: + test: ["CMD", "redis-cli", "ping"] + interval: 5s + timeout: 5s + retries: 5 + + reporting-service-test: + build: + context: . + dockerfile: Dockerfile.test + container_name: reporting-service-test + depends_on: + postgres-test: + condition: service_healthy + redis-test: + condition: service_healthy + environment: + NODE_ENV: test + DATABASE_URL: postgresql://postgres:postgres@postgres-test:5432/rwadurian_reporting_test?schema=public + REDIS_HOST: redis-test + REDIS_PORT: 6379 + JWT_SECRET: test-secret-key + volumes: + - ./coverage:/app/coverage + command: sh -c "npx prisma db push --skip-generate && npm run test:cov" diff --git a/backend/services/reporting-service/docs/API.md b/backend/services/reporting-service/docs/API.md new file mode 100644 index 00000000..b7859b08 --- /dev/null +++ b/backend/services/reporting-service/docs/API.md @@ -0,0 +1,506 @@ +# API 参考文档 + +## 1. 概述 + +Reporting Service 提供 RESTful API,用于报表的生成、查询和导出。 + +**Base URL**: `/api/v1` + +**响应格式**: 所有响应均使用统一的 JSON 格式 + +```json +{ + "success": true, + "data": { ... }, + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +**错误响应格式**: + +```json +{ + "statusCode": 400, + "code": "BAD_REQUEST", + "message": "Validation failed", + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +## 2. 健康检查 API + +### 2.1 健康检查 + +**Endpoint**: `GET /health` + +检查服务是否运行正常。 + +**响应示例**: + +```json +{ + "success": true, + "data": { + "status": "ok", + "service": "reporting-service", + "timestamp": "2024-01-15T10:30:00.000Z" + }, + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +### 2.2 就绪检查 + +**Endpoint**: `GET /health/ready` + +检查服务是否准备好处理请求(包括数据库连接等)。 + +**响应示例**: + +```json +{ + "success": true, + "data": { + "status": "ready", + "service": "reporting-service", + "timestamp": "2024-01-15T10:30:00.000Z" + }, + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +## 3. 报表定义 API + +### 3.1 获取所有报表定义 + +**Endpoint**: `GET /reports/definitions` + +获取所有可用的报表定义列表。 + +**响应示例**: + +```json +{ + "success": true, + "data": [ + { + "id": "1", + "reportCode": "RPT_LEADERBOARD", + "reportName": "排行榜报表", + "reportType": "LEADERBOARD_REPORT", + "description": "用户排行榜数据报表", + "parameters": { + "required": ["startDate", "endDate"], + "optional": ["limit", "offset"] + }, + "isActive": true, + "scheduleCron": "0 0 1 * * *", + "createdAt": "2024-01-01T00:00:00.000Z" + } + ], + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +### 3.2 获取单个报表定义 + +**Endpoint**: `GET /reports/definitions/:code` + +根据报表代码获取报表定义详情。 + +**路径参数**: + +| 参数 | 类型 | 描述 | +|------|------|------| +| code | string | 报表代码 (如 `RPT_LEADERBOARD`) | + +**响应示例**: + +```json +{ + "success": true, + "data": { + "id": "1", + "reportCode": "RPT_LEADERBOARD", + "reportName": "排行榜报表", + "reportType": "LEADERBOARD_REPORT", + "description": "用户排行榜数据报表", + "parameters": { + "required": ["startDate", "endDate"], + "optional": ["limit", "offset"] + }, + "isActive": true, + "scheduleCron": "0 0 1 * * *", + "lastGeneratedAt": "2024-01-14T01:00:00.000Z", + "createdAt": "2024-01-01T00:00:00.000Z", + "updatedAt": "2024-01-14T01:00:00.000Z" + }, + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +**错误响应** (404): + +```json +{ + "statusCode": 404, + "code": "NOT_FOUND", + "message": "Report definition not found: RPT_INVALID", + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +## 4. 报表生成 API + +### 4.1 生成报表 + +**Endpoint**: `POST /reports/generate` + +生成新的报表快照。 + +**请求体**: + +```json +{ + "reportCode": "RPT_LEADERBOARD", + "reportPeriod": "DAILY", + "startDate": "2024-01-01", + "endDate": "2024-01-01", + "filterParams": { + "limit": 100, + "category": "all" + } +} +``` + +**请求参数说明**: + +| 参数 | 类型 | 必填 | 描述 | +|------|------|------|------| +| reportCode | string | 是 | 报表代码 | +| reportPeriod | enum | 是 | 报表周期: `DAILY`, `WEEKLY`, `MONTHLY`, `QUARTERLY`, `YEARLY`, `CUSTOM` | +| startDate | string | 是 | 开始日期 (ISO 8601 格式) | +| endDate | string | 是 | 结束日期 (ISO 8601 格式) | +| filterParams | object | 否 | 额外筛选参数 | + +**响应示例**: + +```json +{ + "success": true, + "data": { + "id": "12345", + "reportCode": "RPT_LEADERBOARD", + "reportType": "LEADERBOARD_REPORT", + "reportPeriod": "DAILY", + "periodKey": "2024-01-01", + "rowCount": 100, + "status": "COMPLETED", + "generatedAt": "2024-01-15T10:30:00.000Z" + }, + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +**验证错误** (400): + +```json +{ + "statusCode": 400, + "code": "BAD_REQUEST", + "message": "reportPeriod must be one of the following values: DAILY, WEEKLY, MONTHLY, QUARTERLY, YEARLY, CUSTOM", + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +## 5. 报表快照 API + +### 5.1 查询报表快照列表 + +**Endpoint**: `GET /reports/snapshots` + +查询报表快照列表,支持分页和筛选。 + +**查询参数**: + +| 参数 | 类型 | 必填 | 描述 | +|------|------|------|------| +| reportCode | string | 否 | 按报表代码筛选 | +| reportPeriod | enum | 否 | 按报表周期筛选 | +| startDate | string | 否 | 开始日期范围 | +| endDate | string | 否 | 结束日期范围 | +| page | number | 否 | 页码 (默认: 1) | +| limit | number | 否 | 每页数量 (默认: 20, 最大: 100) | + +**请求示例**: + +``` +GET /reports/snapshots?reportCode=RPT_LEADERBOARD&reportPeriod=DAILY&page=1&limit=10 +``` + +**响应示例**: + +```json +{ + "success": true, + "data": [ + { + "id": "12345", + "reportCode": "RPT_LEADERBOARD", + "reportType": "LEADERBOARD_REPORT", + "reportPeriod": "DAILY", + "periodKey": "2024-01-15", + "rowCount": 100, + "periodStartAt": "2024-01-15T00:00:00.000Z", + "periodEndAt": "2024-01-15T23:59:59.999Z", + "generatedAt": "2024-01-16T01:00:00.000Z" + } + ], + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +### 5.2 获取快照详情 + +**Endpoint**: `GET /reports/snapshots/:id` + +根据快照ID获取详细数据。 + +**路径参数**: + +| 参数 | 类型 | 描述 | +|------|------|------| +| id | string | 快照ID | + +**响应示例**: + +```json +{ + "success": true, + "data": { + "id": "12345", + "reportCode": "RPT_LEADERBOARD", + "reportType": "LEADERBOARD_REPORT", + "reportPeriod": "DAILY", + "periodKey": "2024-01-15", + "snapshotData": { + "rows": [ + { "rank": 1, "userId": 1001, "username": "user1", "score": 9850 }, + { "rank": 2, "userId": 1002, "username": "user2", "score": 9720 } + ], + "summary": { + "totalEntries": 100, + "averageScore": 5432, + "maxScore": 9850 + } + }, + "summaryData": { + "totalEntries": 100, + "generationTime": "2.5s" + }, + "rowCount": 100, + "dataSource": ["leaderboard-service"], + "periodStartAt": "2024-01-15T00:00:00.000Z", + "periodEndAt": "2024-01-15T23:59:59.999Z", + "generatedAt": "2024-01-16T01:00:00.000Z" + }, + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +### 5.3 获取最新快照 + +**Endpoint**: `GET /reports/snapshots/:code/latest` + +获取指定报表代码的最新快照。 + +**路径参数**: + +| 参数 | 类型 | 描述 | +|------|------|------| +| code | string | 报表代码 | + +**响应示例**: + +```json +{ + "success": true, + "data": { + "id": "12345", + "reportCode": "RPT_LEADERBOARD", + "reportPeriod": "DAILY", + "periodKey": "2024-01-15", + "rowCount": 100, + "generatedAt": "2024-01-16T01:00:00.000Z" + }, + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +**错误响应** (404): + +```json +{ + "statusCode": 404, + "code": "NOT_FOUND", + "message": "No snapshot found for report: RPT_INVALID", + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +## 6. 导出 API + +### 6.1 导出报表 + +**Endpoint**: `POST /export` + +将报表快照导出为指定格式。 + +**请求体**: + +```json +{ + "snapshotId": "12345", + "format": "EXCEL", + "options": { + "includeCharts": true, + "sheetName": "排行榜数据" + } +} +``` + +**请求参数说明**: + +| 参数 | 类型 | 必填 | 描述 | +|------|------|------|------| +| snapshotId | string | 是 | 快照ID | +| format | enum | 是 | 导出格式: `EXCEL`, `CSV`, `PDF`, `JSON` | +| options | object | 否 | 导出选项 | + +**导出选项 (按格式)**: + +**Excel 选项**: +| 选项 | 类型 | 描述 | +|------|------|------| +| includeCharts | boolean | 是否包含图表 | +| sheetName | string | 工作表名称 | + +**PDF 选项**: +| 选项 | 类型 | 描述 | +|------|------|------| +| title | string | PDF标题 | +| orientation | string | 方向: `portrait`, `landscape` | + +**响应示例**: + +```json +{ + "success": true, + "data": { + "fileId": "file_67890", + "fileName": "RPT_LEADERBOARD_2024-01-15.xlsx", + "format": "EXCEL", + "fileSize": 102400, + "downloadUrl": "/api/v1/files/file_67890/download", + "expiresAt": "2024-01-22T10:30:00.000Z", + "createdAt": "2024-01-15T10:30:00.000Z" + }, + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +### 6.2 获取导出文件列表 + +**Endpoint**: `GET /export/files` + +查询已导出的文件列表。 + +**查询参数**: + +| 参数 | 类型 | 必填 | 描述 | +|------|------|------|------| +| snapshotId | string | 否 | 按快照ID筛选 | +| format | enum | 否 | 按格式筛选 | +| page | number | 否 | 页码 | +| limit | number | 否 | 每页数量 | + +**响应示例**: + +```json +{ + "success": true, + "data": [ + { + "fileId": "file_67890", + "fileName": "RPT_LEADERBOARD_2024-01-15.xlsx", + "format": "EXCEL", + "fileSize": 102400, + "snapshotId": "12345", + "createdAt": "2024-01-15T10:30:00.000Z" + } + ], + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +### 6.3 下载文件 + +**Endpoint**: `GET /files/:fileId/download` + +下载导出的文件。 + +**路径参数**: + +| 参数 | 类型 | 描述 | +|------|------|------| +| fileId | string | 文件ID | + +**响应**: 文件二进制流 + +**响应头**: +``` +Content-Type: application/vnd.openxmlformats-officedocument.spreadsheetml.sheet +Content-Disposition: attachment; filename="RPT_LEADERBOARD_2024-01-15.xlsx" +``` + +## 7. 错误码参考 + +| HTTP状态码 | 错误码 | 描述 | +|------------|--------|------| +| 400 | BAD_REQUEST | 请求参数验证失败 | +| 401 | UNAUTHORIZED | 未认证 | +| 403 | FORBIDDEN | 无权限访问 | +| 404 | NOT_FOUND | 资源不存在 | +| 409 | CONFLICT | 资源冲突 (如重复生成) | +| 422 | UNPROCESSABLE_ENTITY | 业务逻辑错误 | +| 500 | INTERNAL_ERROR | 服务器内部错误 | +| 503 | SERVICE_UNAVAILABLE | 服务不可用 | + +## 8. 认证 + +API 支持 JWT Bearer Token 认证(可配置)。 + +**请求头**: +``` +Authorization: Bearer +``` + +**未认证响应** (401): +```json +{ + "statusCode": 401, + "code": "UNAUTHORIZED", + "message": "Unauthorized", + "timestamp": "2024-01-15T10:30:00.000Z" +} +``` + +## 9. 限流 + +| 端点类型 | 限制 | +|----------|------| +| 读取操作 | 100 次/分钟 | +| 生成报表 | 10 次/分钟 | +| 导出操作 | 20 次/分钟 | + +超出限制返回 HTTP 429 Too Many Requests。 diff --git a/backend/services/reporting-service/docs/ARCHITECTURE.md b/backend/services/reporting-service/docs/ARCHITECTURE.md new file mode 100644 index 00000000..b5453450 --- /dev/null +++ b/backend/services/reporting-service/docs/ARCHITECTURE.md @@ -0,0 +1,313 @@ +# 架构设计文档 + +## 1. 概述 + +Reporting Service 采用 **领域驱动设计 (DDD)** 结合 **六边形架构 (Hexagonal Architecture)** 模式构建,实现业务逻辑与技术实现的解耦。 + +## 2. 架构图 + +``` +┌─────────────────────────────────────────────────────────────────────────────┐ +│ API Layer (端口) │ +│ ┌─────────────────┐ ┌─────────────────┐ ┌─────────────────────────────┐ │ +│ │ ReportController│ │ ExportController│ │ HealthController │ │ +│ └────────┬────────┘ └────────┬────────┘ └─────────────────────────────┘ │ +│ │ │ │ +│ ▼ ▼ │ +│ ┌─────────────────────────────────────────────────────────────────────────┐│ +│ │ Application Layer ││ +│ │ ┌──────────────────┐ ┌──────────────────┐ ┌────────────────────────┐ ││ +│ │ │ Commands/Queries │ │ Schedulers │ │ Application Services │ ││ +│ │ │ - GenerateReport │ │ - CronJobs │ │ - ReportingAppService │ ││ +│ │ │ - ExportReport │ │ - Periodic Tasks │ │ │ ││ +│ │ └────────┬─────────┘ └──────────────────┘ └────────────────────────┘ ││ +│ └───────────┼──────────────────────────────────────────────────────────────┘│ +│ │ │ +│ ▼ │ +│ ┌─────────────────────────────────────────────────────────────────────────┐│ +│ │ Domain Layer (核心) ││ +│ │ ┌──────────────────┐ ┌──────────────────┐ ┌────────────────────────┐ ││ +│ │ │ Aggregates │ │ Entities │ │ Value Objects │ ││ +│ │ │ - ReportDefinit. │ │ - ReportFile │ │ - DateRange │ ││ +│ │ │ - ReportSnapshot │ │ - AnalyticsMetric│ │ - ReportPeriod │ ││ +│ │ └──────────────────┘ └──────────────────┘ │ - SnapshotData │ ││ +│ │ └────────────────────────┘ ││ +│ │ ┌──────────────────┐ ┌──────────────────┐ ┌────────────────────────┐ ││ +│ │ │ Domain Events │ │ Domain Services │ │ Repository Ports │ ││ +│ │ │ - SnapshotCreated│ │ - ReportGenSvc │ │ (Interfaces Only) │ ││ +│ │ │ - ReportExported │ │ │ │ │ ││ +│ │ └──────────────────┘ └──────────────────┘ └────────────────────────┘ ││ +│ └─────────────────────────────────────────────────────────────────────────┘│ +│ ▲ │ +│ │ │ +│ ┌─────────────────────────────────────────────────────────────────────────┐│ +│ │ Infrastructure Layer (适配器) ││ +│ │ ┌──────────────────┐ ┌──────────────────┐ ┌────────────────────────┐ ││ +│ │ │ Persistence │ │ External APIs │ │ Export Services │ ││ +│ │ │ - Prisma/PG │ │ - LeaderboardSvc │ │ - ExcelExport │ ││ +│ │ │ - Repositories │ │ - PlantingSvc │ │ - CSVExport │ ││ +│ │ │ - Mappers │ │ │ │ - PDFExport │ ││ +│ │ └──────────────────┘ └──────────────────┘ └────────────────────────┘ ││ +│ │ ┌──────────────────┐ ┌──────────────────┐ ││ +│ │ │ Redis Cache │ │ Kafka │ ││ +│ │ │ - ReportCache │ │ - EventPublisher │ ││ +│ │ └──────────────────┘ └──────────────────┘ ││ +│ └─────────────────────────────────────────────────────────────────────────┘│ +└─────────────────────────────────────────────────────────────────────────────┘ +``` + +## 3. 目录结构 + +``` +src/ +├── api/ # API层 - 入站端口 +│ ├── controllers/ # HTTP控制器 +│ │ ├── report.controller.ts # 报表API +│ │ ├── export.controller.ts # 导出API +│ │ └── health.controller.ts # 健康检查 +│ └── dto/ # 数据传输对象 +│ ├── request/ # 请求DTO +│ └── response/ # 响应DTO +│ +├── application/ # 应用层 - 用例编排 +│ ├── commands/ # 命令处理器 +│ │ ├── generate-report/ # 生成报表命令 +│ │ └── export-report/ # 导出报表命令 +│ ├── queries/ # 查询处理器 +│ │ └── get-report-snapshot/ # 获取快照查询 +│ ├── schedulers/ # 定时任务 +│ │ └── report-generation.scheduler.ts +│ └── services/ # 应用服务 +│ └── reporting-application.service.ts +│ +├── domain/ # 领域层 - 业务核心 +│ ├── aggregates/ # 聚合根 +│ │ ├── report-definition/ # 报表定义聚合 +│ │ └── report-snapshot/ # 报表快照聚合 +│ ├── entities/ # 实体 +│ │ ├── report-file.entity.ts +│ │ └── analytics-metric.entity.ts +│ ├── value-objects/ # 值对象 +│ │ ├── date-range.vo.ts +│ │ ├── report-period.enum.ts +│ │ ├── report-type.enum.ts +│ │ └── snapshot-data.vo.ts +│ ├── events/ # 领域事件 +│ │ ├── snapshot-created.event.ts +│ │ └── report-exported.event.ts +│ ├── repositories/ # 仓储接口 +│ │ ├── report-definition.repository.interface.ts +│ │ └── report-snapshot.repository.interface.ts +│ └── services/ # 领域服务 +│ └── report-generation.service.ts +│ +├── infrastructure/ # 基础设施层 - 出站适配器 +│ ├── persistence/ # 持久化 +│ │ ├── prisma/ # Prisma配置 +│ │ ├── repositories/ # 仓储实现 +│ │ └── mappers/ # 对象映射器 +│ ├── external/ # 外部服务客户端 +│ │ ├── leaderboard-service/ +│ │ └── planting-service/ +│ ├── export/ # 导出服务实现 +│ │ ├── excel-export.service.ts +│ │ ├── csv-export.service.ts +│ │ └── pdf-export.service.ts +│ └── redis/ # Redis缓存 +│ └── report-cache.service.ts +│ +├── shared/ # 共享模块 +│ ├── decorators/ # 自定义装饰器 +│ ├── filters/ # 异常过滤器 +│ ├── guards/ # 守卫 +│ ├── interceptors/ # 拦截器 +│ └── strategies/ # Passport策略 +│ +└── config/ # 配置 + ├── app.config.ts + ├── database.config.ts + └── redis.config.ts +``` + +## 4. 核心设计原则 + +### 4.1 依赖倒置原则 (DIP) + +```typescript +// 领域层定义接口 (端口) +export interface IReportSnapshotRepository { + save(snapshot: ReportSnapshot): Promise; + findById(id: bigint): Promise; + findByCodeAndPeriodKey(code: string, periodKey: string): Promise; +} + +// 基础设施层实现接口 (适配器) +@Injectable() +export class ReportSnapshotRepository implements IReportSnapshotRepository { + constructor(private readonly prisma: PrismaService) {} + + async save(snapshot: ReportSnapshot): Promise { + // Prisma实现细节 + } +} +``` + +### 4.2 聚合根设计 + +```typescript +// ReportSnapshot 聚合根 +export class ReportSnapshot extends AggregateRoot { + private readonly _id: bigint; + private _reportType: ReportType; + private _reportCode: string; + private _reportPeriod: ReportPeriod; + private _snapshotData: SnapshotData; + private _events: DomainEvent[] = []; + + // 工厂方法 - 创建新快照 + public static create(props: CreateSnapshotProps): ReportSnapshot { + const snapshot = new ReportSnapshot(props); + snapshot.addDomainEvent(new SnapshotCreatedEvent(snapshot)); + return snapshot; + } + + // 从持久化重建 - 不触发事件 + public static reconstitute(props: ReconstitutionProps): ReportSnapshot { + return new ReportSnapshot(props); + } + + // 业务行为 + public updateData(newData: SnapshotData): void { + this.validateDataUpdate(newData); + this._snapshotData = newData; + this._updatedAt = new Date(); + } +} +``` + +### 4.3 值对象不可变性 + +```typescript +// DateRange 值对象 +export class DateRange { + private readonly _startDate: Date; + private readonly _endDate: Date; + + private constructor(startDate: Date, endDate: Date) { + this._startDate = startDate; + this._endDate = endDate; + Object.freeze(this); + } + + public static create(startDate: Date, endDate: Date): DateRange { + if (startDate > endDate) { + throw new DomainException('Start date must be before end date'); + } + return new DateRange(startDate, endDate); + } + + public equals(other: DateRange): boolean { + return this._startDate.getTime() === other._startDate.getTime() && + this._endDate.getTime() === other._endDate.getTime(); + } +} +``` + +## 5. 数据流 + +### 5.1 报表生成流程 + +``` +┌─────────┐ ┌──────────────┐ ┌─────────────────┐ ┌──────────────┐ +│ Client │───▶│ ReportCtrl │───▶│ GenerateHandler │───▶│ DomainSvc │ +└─────────┘ └──────────────┘ └─────────────────┘ └──────┬───────┘ + │ + ┌─────────────────────────────────────────────┘ + ▼ +┌──────────────────────┐ ┌──────────────────┐ ┌──────────────────┐ +│ External Services │───▶│ ReportSnapshot │───▶│ Repository │ +│ (Leaderboard/Plant.) │ │ (Aggregate) │ │ (Save to DB) │ +└──────────────────────┘ └──────────────────┘ └──────────────────┘ +``` + +### 5.2 报表导出流程 + +``` +┌─────────┐ ┌──────────────┐ ┌─────────────────┐ +│ Client │───▶│ ExportCtrl │───▶│ ExportHandler │ +└─────────┘ └──────────────┘ └────────┬────────┘ + │ + ┌───────────────────────────────────┴───────────────────────────┐ + ▼ ▼ ▼ ▼ +┌──────────────────┐ ┌──────────────────┐ ┌──────────────────┐ +│ ExcelExportSvc │ │ CSVExportSvc │ │ PDFExportSvc │ +│ (ExcelJS) │ │ (csv-stringify) │ │ (PDFKit) │ +└──────────────────┘ └──────────────────┘ └──────────────────┘ + │ │ │ + └───────────────────────┴────────────────────────┘ + ▼ + ┌──────────────────────┐ + │ ReportFile Entity │ + │ (Save to Storage) │ + └──────────────────────┘ +``` + +## 6. 关键组件 + +### 6.1 报表类型 (ReportType) + +| 类型 | 描述 | 数据来源 | +|------|------|----------| +| LEADERBOARD_REPORT | 排行榜报表 | Leaderboard Service | +| PLANTING_REPORT | 种植报表 | Planting Service | +| COMMUNITY_REPORT | 社区报表 | Community Service | +| SYSTEM_ACCOUNT_REPORT | 系统账户报表 | Account Service | +| ANALYTICS_DASHBOARD | 分析仪表板 | 多数据源聚合 | + +### 6.2 报表周期 (ReportPeriod) + +| 周期 | 描述 | 用途 | +|------|------|------| +| DAILY | 日报 | 每日运营数据 | +| WEEKLY | 周报 | 周度趋势分析 | +| MONTHLY | 月报 | 月度业绩汇总 | +| QUARTERLY | 季报 | 季度财务报表 | +| YEARLY | 年报 | 年度总结 | +| CUSTOM | 自定义 | 灵活时间范围 | + +### 6.3 导出格式 (OutputFormat) + +| 格式 | 实现库 | 特点 | +|------|--------|------| +| EXCEL | ExcelJS | 支持样式、图表、多Sheet | +| CSV | csv-stringify | 轻量、通用 | +| PDF | PDFKit | 适合打印、分发 | +| JSON | 内置 | API集成、数据交换 | + +## 7. 扩展点 + +### 7.1 添加新报表类型 + +1. 在 `ReportType` 枚举中添加新类型 +2. 创建对应的外部服务客户端 (如需要) +3. 在 `GenerateReportHandler` 中添加数据获取逻辑 +4. 更新 `ReportDefinition` 种子数据 + +### 7.2 添加新导出格式 + +1. 在 `OutputFormat` 枚举中添加新格式 +2. 创建新的导出服务 (实现 `IExportService` 接口) +3. 在 `ExportReportHandler` 中注册新服务 + +### 7.3 集成新数据源 + +1. 在 `infrastructure/external/` 下创建服务客户端 +2. 定义数据传输接口 +3. 在应用层注入并使用 + +## 8. 安全考虑 + +- **认证**: JWT Bearer Token (可配置) +- **授权**: 基于角色的访问控制 (RBAC) +- **数据脱敏**: 敏感数据在导出时进行脱敏处理 +- **审计日志**: 所有报表操作记录到审计表 diff --git a/backend/services/reporting-service/docs/DATA-MODEL.md b/backend/services/reporting-service/docs/DATA-MODEL.md new file mode 100644 index 00000000..861126b2 --- /dev/null +++ b/backend/services/reporting-service/docs/DATA-MODEL.md @@ -0,0 +1,680 @@ +# 数据模型文档 + +## 1. 数据模型概述 + +### 1.1 实体关系图 (ER Diagram) + +``` +┌─────────────────────┐ +│ ReportDefinition │ +│ (报表定义) │ +│ ───────────────── │ +│ definition_id PK │ +│ report_code UK │ +│ report_type │ +│ report_name │ +│ parameters JSON │ +│ schedule_cron │ +│ output_formats[] │ +└─────────────────────┘ + +┌─────────────────────┐ ┌─────────────────────┐ +│ ReportSnapshot │ 1───n │ ReportFile │ +│ (报表快照) │ │ (报表文件) │ +│ ───────────────── │ │ ───────────────── │ +│ snapshot_id PK │ │ file_id PK │ +│ report_code │ │ snapshot_id FK │ +│ report_period │ │ file_name │ +│ period_key UK(c) │ │ file_path │ +│ snapshot_data JSON │ │ file_format │ +│ summary_data JSON │ │ file_size │ +└─────────────────────┘ └─────────────────────┘ + +┌─────────────────────┐ ┌─────────────────────┐ +│ AnalyticsMetric │ │ PlantingDailyStat │ +│ (分析指标) │ │ (认种日统计) │ +│ ───────────────── │ │ ───────────────── │ +│ metric_id PK │ │ stat_id PK │ +│ metric_type │ │ stat_date │ +│ metric_code │ │ province_code │ +│ dimension_time │ │ city_code │ +│ metric_value DEC │ │ order_count │ +│ metric_data JSON │ │ total_amount DEC │ +└─────────────────────┘ └─────────────────────┘ + +┌─────────────────────┐ ┌──────────────────────────┐ +│ CommunityStat │ │ SystemAccountMonthlyStat │ +│ (社区统计) │ │ (系统账户月度统计) │ +│ ───────────────── │ │ ───────────────────── │ +│ stat_id PK │ │ stat_id PK │ +│ community_id │ │ account_id │ +│ community_name │ │ account_type │ +│ stat_date │ │ stat_month │ +│ total_planting │ │ monthly_hashpower DEC │ +│ member_count │ │ cumulative_mining DEC │ +└─────────────────────┘ └──────────────────────────┘ + +┌──────────────────────────┐ ┌─────────────────────┐ +│ SystemAccountIncomeRecord│ │ ReportEvent │ +│ (系统账户收益流水) │ │ (报表事件) │ +│ ───────────────────── │ │ ───────────────── │ +│ record_id PK │ │ event_id PK │ +│ account_id │ │ event_type │ +│ income_type │ │ aggregate_id │ +│ income_amount DEC │ │ aggregate_type │ +│ source_type │ │ event_data JSON │ +│ occurred_at │ │ occurred_at │ +└──────────────────────────┘ └─────────────────────┘ +``` + +### 1.2 聚合根划分 + +| 聚合根 | 表 | 说明 | +|--------|-----|------| +| ReportDefinition | report_definitions | 报表定义,包含配置和调度规则 | +| ReportSnapshot | report_snapshots, report_files | 报表快照,包含生成的文件 | +| AnalyticsMetric | analytics_metrics | 独立的分析指标聚合 | + +## 2. 核心表结构 + +### 2.1 ReportDefinition (报表定义表) + +```sql +CREATE TABLE report_definitions ( + definition_id BIGSERIAL PRIMARY KEY, + + -- 报表基本信息 + report_type VARCHAR(50) NOT NULL, + report_name VARCHAR(200) NOT NULL, + report_code VARCHAR(50) NOT NULL UNIQUE, + description TEXT, + + -- 报表参数 (JSON) + parameters JSONB NOT NULL DEFAULT '{}', + + -- 调度配置 + schedule_cron VARCHAR(100), + schedule_timezone VARCHAR(50) DEFAULT 'Asia/Shanghai', + schedule_enabled BOOLEAN DEFAULT FALSE, + + -- 输出格式 + output_formats VARCHAR(20)[] NOT NULL DEFAULT '{}', + + -- 状态 + is_active BOOLEAN DEFAULT TRUE, + + -- 时间戳 + created_at TIMESTAMP DEFAULT NOW(), + updated_at TIMESTAMP DEFAULT NOW(), + last_generated_at TIMESTAMP +); + +-- 索引 +CREATE INDEX idx_def_type ON report_definitions(report_type); +CREATE INDEX idx_def_active ON report_definitions(is_active); +CREATE INDEX idx_def_scheduled ON report_definitions(schedule_enabled); +``` + +**字段说明:** + +| 字段 | 类型 | 说明 | +|------|------|------| +| definition_id | BIGSERIAL | 主键,自增ID | +| report_type | VARCHAR(50) | 报表类型: LEADERBOARD, PLANTING, ANALYTICS 等 | +| report_name | VARCHAR(200) | 报表显示名称 | +| report_code | VARCHAR(50) | 报表唯一编码,用于API调用 | +| description | TEXT | 报表描述说明 | +| parameters | JSONB | 报表参数配置 | +| schedule_cron | VARCHAR(100) | Cron 表达式,定时生成 | +| schedule_timezone | VARCHAR(50) | 调度时区 | +| schedule_enabled | BOOLEAN | 是否启用定时生成 | +| output_formats | VARCHAR(20)[] | 支持的输出格式数组 | +| is_active | BOOLEAN | 是否激活 | +| last_generated_at | TIMESTAMP | 最后生成时间 | + +**parameters JSON 结构示例:** + +```json +{ + "dataSource": "leaderboard_rankings", + "aggregation": "daily", + "filters": { + "regionLevel": "province", + "minScore": 100 + }, + "columns": [ + { "field": "rank", "label": "排名" }, + { "field": "userName", "label": "用户名" }, + { "field": "score", "label": "积分" } + ] +} +``` + +### 2.2 ReportSnapshot (报表快照表) + +```sql +CREATE TABLE report_snapshots ( + snapshot_id BIGSERIAL PRIMARY KEY, + + -- 报表信息 + report_type VARCHAR(50) NOT NULL, + report_code VARCHAR(50) NOT NULL, + report_period VARCHAR(20) NOT NULL, + period_key VARCHAR(30) NOT NULL, + + -- 快照数据 + snapshot_data JSONB NOT NULL, + summary_data JSONB, + + -- 数据来源 + data_sources VARCHAR(100)[] DEFAULT '{}', + data_freshness INTEGER DEFAULT 0, + + -- 过滤条件 + filter_params JSONB, + + -- 统计信息 + row_count INTEGER DEFAULT 0, + + -- 时间范围 + period_start_at TIMESTAMP NOT NULL, + period_end_at TIMESTAMP NOT NULL, + + -- 时间戳 + generated_at TIMESTAMP DEFAULT NOW(), + expires_at TIMESTAMP, + + UNIQUE(report_code, period_key) +); + +-- 索引 +CREATE INDEX idx_snapshot_type ON report_snapshots(report_type); +CREATE INDEX idx_snapshot_code ON report_snapshots(report_code); +CREATE INDEX idx_snapshot_period ON report_snapshots(period_key); +CREATE INDEX idx_snapshot_generated ON report_snapshots(generated_at DESC); +CREATE INDEX idx_snapshot_expires ON report_snapshots(expires_at); +``` + +**字段说明:** + +| 字段 | 类型 | 说明 | +|------|------|------| +| snapshot_id | BIGSERIAL | 主键 | +| report_type | VARCHAR(50) | 报表类型 | +| report_code | VARCHAR(50) | 报表编码 | +| report_period | VARCHAR(20) | 报表周期: DAILY, WEEKLY, MONTHLY, YEARLY | +| period_key | VARCHAR(30) | 周期标识: 2024-01-15, 2024-W03, 2024-01 | +| snapshot_data | JSONB | 报表数据内容 | +| summary_data | JSONB | 汇总数据 | +| data_sources | VARCHAR(100)[] | 数据来源服务列表 | +| data_freshness | INTEGER | 数据新鲜度(秒) | +| filter_params | JSONB | 应用的过滤参数 | +| row_count | INTEGER | 数据行数 | +| period_start_at | TIMESTAMP | 周期开始时间 | +| period_end_at | TIMESTAMP | 周期结束时间 | +| generated_at | TIMESTAMP | 生成时间 | +| expires_at | TIMESTAMP | 过期时间 | + +**snapshot_data JSON 结构示例:** + +```json +{ + "items": [ + { + "rank": 1, + "userId": "12345", + "userName": "张三", + "score": 9850, + "treeCount": 100, + "region": "广东省" + } + ], + "metadata": { + "totalRecords": 1000, + "generatedAt": "2024-01-15T08:00:00Z", + "version": "1.0" + } +} +``` + +### 2.3 ReportFile (报表文件表) + +```sql +CREATE TABLE report_files ( + file_id BIGSERIAL PRIMARY KEY, + snapshot_id BIGINT NOT NULL REFERENCES report_snapshots(snapshot_id) ON DELETE CASCADE, + + -- 文件信息 + file_name VARCHAR(500) NOT NULL, + file_path VARCHAR(1000) NOT NULL, + file_url VARCHAR(1000), + file_size BIGINT NOT NULL, + file_format VARCHAR(20) NOT NULL, + mime_type VARCHAR(100) NOT NULL, + + -- 访问信息 + download_count INTEGER DEFAULT 0, + last_download_at TIMESTAMP, + + -- 时间戳 + created_at TIMESTAMP DEFAULT NOW(), + expires_at TIMESTAMP +); + +-- 索引 +CREATE INDEX idx_file_snapshot ON report_files(snapshot_id); +CREATE INDEX idx_file_format ON report_files(file_format); +CREATE INDEX idx_file_created ON report_files(created_at DESC); +``` + +**字段说明:** + +| 字段 | 类型 | 说明 | +|------|------|------| +| file_id | BIGSERIAL | 主键 | +| snapshot_id | BIGINT | 关联的快照ID | +| file_name | VARCHAR(500) | 文件名 | +| file_path | VARCHAR(1000) | 文件存储路径 | +| file_url | VARCHAR(1000) | 文件下载URL | +| file_size | BIGINT | 文件大小(字节) | +| file_format | VARCHAR(20) | 文件格式: EXCEL, CSV, PDF | +| mime_type | VARCHAR(100) | MIME类型 | +| download_count | INTEGER | 下载次数 | +| last_download_at | TIMESTAMP | 最后下载时间 | + +### 2.4 AnalyticsMetric (分析指标表) + +```sql +CREATE TABLE analytics_metrics ( + metric_id BIGSERIAL PRIMARY KEY, + + -- 指标信息 + metric_type VARCHAR(50) NOT NULL, + metric_code VARCHAR(50) NOT NULL, + + -- 维度 + dimension_time DATE, + dimension_region VARCHAR(100), + dimension_user_type VARCHAR(50), + dimension_right_type VARCHAR(50), + + -- 指标值 + metric_value DECIMAL(20, 8) NOT NULL, + metric_data JSONB, + + -- 时间戳 + calculated_at TIMESTAMP DEFAULT NOW(), + + UNIQUE(metric_code, dimension_time, dimension_region, dimension_user_type, dimension_right_type) +); + +-- 索引 +CREATE INDEX idx_metric_type ON analytics_metrics(metric_type); +CREATE INDEX idx_metric_code ON analytics_metrics(metric_code); +CREATE INDEX idx_metric_time ON analytics_metrics(dimension_time); +CREATE INDEX idx_metric_region ON analytics_metrics(dimension_region); +``` + +**指标类型 (metric_type):** + +| 类型 | 说明 | +|------|------| +| PLANTING_COUNT | 认种数量 | +| PLANTING_AMOUNT | 认种金额 | +| USER_ACTIVE | 活跃用户数 | +| REVENUE_TOTAL | 总收益 | +| HASHPOWER_DAILY | 日算力 | + +## 3. 统计表结构 + +### 3.1 PlantingDailyStat (认种日统计表) + +```sql +CREATE TABLE planting_daily_stats ( + stat_id BIGSERIAL PRIMARY KEY, + + -- 统计日期 + stat_date DATE NOT NULL, + + -- 区域维度 + province_code VARCHAR(10), + city_code VARCHAR(10), + + -- 统计数据 + order_count INTEGER DEFAULT 0, + tree_count INTEGER DEFAULT 0, + total_amount DECIMAL(20, 8) DEFAULT 0, + new_user_count INTEGER DEFAULT 0, + active_user_count INTEGER DEFAULT 0, + + -- 时间戳 + created_at TIMESTAMP DEFAULT NOW(), + updated_at TIMESTAMP DEFAULT NOW(), + + UNIQUE(stat_date, province_code, city_code) +); + +-- 索引 +CREATE INDEX idx_pds_date ON planting_daily_stats(stat_date); +CREATE INDEX idx_pds_province ON planting_daily_stats(province_code); +CREATE INDEX idx_pds_city ON planting_daily_stats(city_code); +``` + +### 3.2 CommunityStat (社区统计表) + +```sql +CREATE TABLE community_stats ( + stat_id BIGSERIAL PRIMARY KEY, + + -- 社区信息 + community_id BIGINT NOT NULL, + community_name VARCHAR(200) NOT NULL, + parent_community_id BIGINT, + + -- 统计日期 + stat_date DATE NOT NULL, + + -- 统计数据 + total_planting INTEGER DEFAULT 0, + daily_planting INTEGER DEFAULT 0, + weekly_planting INTEGER DEFAULT 0, + monthly_planting INTEGER DEFAULT 0, + member_count INTEGER DEFAULT 0, + + -- 时间戳 + created_at TIMESTAMP DEFAULT NOW(), + updated_at TIMESTAMP DEFAULT NOW(), + + UNIQUE(community_id, stat_date) +); + +-- 索引 +CREATE INDEX idx_cs_community ON community_stats(community_id); +CREATE INDEX idx_cs_name ON community_stats(community_name); +CREATE INDEX idx_cs_date ON community_stats(stat_date); +CREATE INDEX idx_cs_parent ON community_stats(parent_community_id); +``` + +### 3.3 SystemAccountMonthlyStat (系统账户月度统计表) + +```sql +CREATE TABLE system_account_monthly_stats ( + stat_id BIGSERIAL PRIMARY KEY, + + -- 账户信息 + account_id BIGINT NOT NULL, + account_type VARCHAR(30) NOT NULL, + account_name VARCHAR(200) NOT NULL, + region_code VARCHAR(10) NOT NULL, + + -- 统计月份 + stat_month VARCHAR(7) NOT NULL, -- 格式: YYYY-MM + + -- 月度数据 + monthly_hashpower DECIMAL(20, 8) DEFAULT 0, + cumulative_hashpower DECIMAL(20, 8) DEFAULT 0, + monthly_mining DECIMAL(20, 8) DEFAULT 0, + cumulative_mining DECIMAL(20, 8) DEFAULT 0, + monthly_commission DECIMAL(20, 8) DEFAULT 0, + cumulative_commission DECIMAL(20, 8) DEFAULT 0, + monthly_planting_bonus DECIMAL(20, 8) DEFAULT 0, + cumulative_planting_bonus DECIMAL(20, 8) DEFAULT 0, + + -- 时间戳 + created_at TIMESTAMP DEFAULT NOW(), + updated_at TIMESTAMP DEFAULT NOW(), + + UNIQUE(account_id, stat_month) +); + +-- 索引 +CREATE INDEX idx_sams_type ON system_account_monthly_stats(account_type); +CREATE INDEX idx_sams_month ON system_account_monthly_stats(stat_month); +CREATE INDEX idx_sams_region ON system_account_monthly_stats(region_code); +``` + +**account_type 枚举值:** + +| 值 | 说明 | +|-----|------| +| PROVINCE_COMPANY | 省公司账户 | +| CITY_COMPANY | 市公司账户 | +| PLATFORM | 平台账户 | + +### 3.4 SystemAccountIncomeRecord (系统账户收益流水表) + +```sql +CREATE TABLE system_account_income_records ( + record_id BIGSERIAL PRIMARY KEY, + + -- 账户信息 + account_id BIGINT NOT NULL, + account_type VARCHAR(30) NOT NULL, + + -- 收益信息 + income_type VARCHAR(50) NOT NULL, + income_amount DECIMAL(20, 8) NOT NULL, + currency VARCHAR(10) NOT NULL, + + -- 来源信息 + source_type VARCHAR(50) NOT NULL, + source_id VARCHAR(100), + source_user_id BIGINT, + source_address VARCHAR(200), + transaction_no VARCHAR(100), + + -- 备注 + memo TEXT, + + -- 时间戳 + occurred_at TIMESTAMP NOT NULL, + created_at TIMESTAMP DEFAULT NOW() +); + +-- 索引 +CREATE INDEX idx_sair_account ON system_account_income_records(account_id); +CREATE INDEX idx_sair_type ON system_account_income_records(account_type); +CREATE INDEX idx_sair_income_type ON system_account_income_records(income_type); +CREATE INDEX idx_sair_source_type ON system_account_income_records(source_type); +CREATE INDEX idx_sair_address ON system_account_income_records(source_address); +CREATE INDEX idx_sair_txno ON system_account_income_records(transaction_no); +CREATE INDEX idx_sair_occurred ON system_account_income_records(occurred_at DESC); +``` + +**income_type 枚举值:** + +| 值 | 说明 | +|-----|------| +| MINING_REWARD | 挖矿奖励 | +| COMMISSION | 佣金收入 | +| PLANTING_BONUS | 认种奖励 | +| REFERRAL_BONUS | 推荐奖励 | + +**source_type 枚举值:** + +| 值 | 说明 | +|-----|------| +| PLANTING_ORDER | 认种订单 | +| MINING_SETTLEMENT | 挖矿结算 | +| COMMISSION_SETTLEMENT | 佣金结算 | +| MANUAL_ADJUSTMENT | 手工调整 | + +## 4. 事件表 + +### 4.1 ReportEvent (报表事件表) + +```sql +CREATE TABLE report_events ( + event_id BIGSERIAL PRIMARY KEY, + event_type VARCHAR(50) NOT NULL, + + -- 聚合根信息 + aggregate_id VARCHAR(100) NOT NULL, + aggregate_type VARCHAR(50) NOT NULL, + + -- 事件数据 + event_data JSONB NOT NULL, + + -- 元数据 + user_id BIGINT, + occurred_at TIMESTAMP(6) DEFAULT NOW(), + version INTEGER DEFAULT 1 +); + +-- 索引 +CREATE INDEX idx_report_event_aggregate ON report_events(aggregate_type, aggregate_id); +CREATE INDEX idx_report_event_type ON report_events(event_type); +CREATE INDEX idx_report_event_occurred ON report_events(occurred_at); +``` + +**event_type 枚举值:** + +| 值 | 说明 | +|-----|------| +| REPORT_DEFINITION_CREATED | 报表定义创建 | +| REPORT_DEFINITION_UPDATED | 报表定义更新 | +| REPORT_SNAPSHOT_GENERATED | 报表快照生成 | +| REPORT_FILE_EXPORTED | 报表文件导出 | +| REPORT_FILE_DOWNLOADED | 报表文件下载 | + +**event_data 示例:** + +```json +{ + "snapshotId": "12345", + "reportCode": "RPT_LEADERBOARD", + "period": "DAILY", + "periodKey": "2024-01-15", + "rowCount": 1000, + "generatedAt": "2024-01-15T08:00:00Z" +} +``` + +## 5. 数据字典 + +### 5.1 报表周期 (ReportPeriod) + +| 值 | 说明 | period_key 格式 | +|-----|------|----------------| +| DAILY | 日报 | 2024-01-15 | +| WEEKLY | 周报 | 2024-W03 | +| MONTHLY | 月报 | 2024-01 | +| YEARLY | 年报 | 2024 | +| CUSTOM | 自定义 | 2024-01-01_2024-01-31 | + +### 5.2 导出格式 (ExportFormat) + +| 值 | MIME类型 | 说明 | +|-----|----------|------| +| EXCEL | application/vnd.openxmlformats-officedocument.spreadsheetml.sheet | Excel 2007+ | +| CSV | text/csv | 逗号分隔值 | +| PDF | application/pdf | PDF 文档 | +| JSON | application/json | JSON 格式 | + +### 5.3 报表类型 (ReportType) + +| 值 | 说明 | +|-----|------| +| LEADERBOARD | 排行榜报表 | +| PLANTING | 认种统计报表 | +| ANALYTICS | 分析报表 | +| COMMUNITY | 社区统计报表 | +| SYSTEM_ACCOUNT | 系统账户报表 | +| INCOME_DETAIL | 收益明细报表 | + +## 6. 索引策略 + +### 6.1 查询模式分析 + +| 查询场景 | 使用索引 | 频率 | +|---------|---------|------| +| 按报表编码查询快照 | idx_snapshot_code | 高 | +| 按时间范围查询快照 | idx_snapshot_generated | 高 | +| 按周期查询快照 | idx_snapshot_period | 中 | +| 按账户查询收益流水 | idx_sair_account | 高 | +| 按日期查询统计数据 | idx_pds_date, idx_cs_date | 高 | + +### 6.2 复合索引建议 + +```sql +-- 高频查询: 按报表编码和周期查询最新快照 +CREATE INDEX idx_snapshot_code_generated +ON report_snapshots(report_code, generated_at DESC); + +-- 高频查询: 按账户和月份查询统计 +CREATE INDEX idx_sams_account_month +ON system_account_monthly_stats(account_id, stat_month DESC); + +-- 高频查询: 按日期和区域查询认种统计 +CREATE INDEX idx_pds_date_region +ON planting_daily_stats(stat_date, province_code, city_code); +``` + +## 7. 数据保留策略 + +### 7.1 保留周期 + +| 表 | 保留周期 | 清理策略 | +|-----|---------|---------| +| report_snapshots | 90天 | 按 expires_at 清理 | +| report_files | 30天 | 按 expires_at 清理 | +| analytics_metrics | 365天 | 按 calculated_at 归档 | +| planting_daily_stats | 永久 | 按年归档 | +| system_account_income_records | 永久 | 按年归档 | +| report_events | 180天 | 按 occurred_at 清理 | + +### 7.2 清理脚本 + +```sql +-- 清理过期快照 +DELETE FROM report_snapshots +WHERE expires_at IS NOT NULL AND expires_at < NOW(); + +-- 清理过期文件 +DELETE FROM report_files +WHERE expires_at IS NOT NULL AND expires_at < NOW(); + +-- 清理旧事件 +DELETE FROM report_events +WHERE occurred_at < NOW() - INTERVAL '180 days'; +``` + +## 8. Prisma Schema 映射 + +### 8.1 模型与表映射 + +| Prisma Model | 数据库表 | 主键映射 | +|--------------|---------|---------| +| ReportDefinition | report_definitions | definition_id | +| ReportSnapshot | report_snapshots | snapshot_id | +| ReportFile | report_files | file_id | +| AnalyticsMetric | analytics_metrics | metric_id | +| PlantingDailyStat | planting_daily_stats | stat_id | +| CommunityStat | community_stats | stat_id | +| SystemAccountMonthlyStat | system_account_monthly_stats | stat_id | +| SystemAccountIncomeRecord | system_account_income_records | record_id | +| ReportEvent | report_events | event_id | + +### 8.2 关系映射 + +```prisma +// ReportSnapshot 1:N ReportFile +model ReportFile { + snapshot ReportSnapshot @relation(fields: [snapshotId], references: [id], onDelete: Cascade) + snapshotId BigInt @map("snapshot_id") +} +``` + +### 8.3 类型映射 + +| Prisma 类型 | PostgreSQL 类型 | 说明 | +|------------|----------------|------| +| BigInt | BIGSERIAL | 大整数主键 | +| String @db.VarChar(n) | VARCHAR(n) | 变长字符串 | +| String @db.Text | TEXT | 长文本 | +| Json | JSONB | JSON 二进制 | +| Decimal @db.Decimal(20,8) | DECIMAL(20,8) | 高精度数值 | +| DateTime | TIMESTAMP | 时间戳 | +| DateTime @db.Date | DATE | 日期 | +| Boolean | BOOLEAN | 布尔值 | +| String[] | VARCHAR(n)[] | 字符串数组 | diff --git a/backend/services/reporting-service/docs/DEPLOYMENT.md b/backend/services/reporting-service/docs/DEPLOYMENT.md new file mode 100644 index 00000000..8c9ab2a2 --- /dev/null +++ b/backend/services/reporting-service/docs/DEPLOYMENT.md @@ -0,0 +1,849 @@ +# 部署指南 + +## 1. 部署架构 + +### 1.1 生产环境架构 + +``` + ┌─────────────────┐ + │ Load Balancer │ + │ (Nginx/ALB) │ + └────────┬────────┘ + │ + ┌───────────────────┼───────────────────┐ + │ │ │ + ▼ ▼ ▼ + ┌─────────────────┐ ┌─────────────────┐ ┌─────────────────┐ + │ Reporting │ │ Reporting │ │ Reporting │ + │ Service #1 │ │ Service #2 │ │ Service #3 │ + └────────┬────────┘ └────────┬────────┘ └────────┬────────┘ + │ │ │ + └───────────────────┼───────────────────┘ + │ + ┌──────────────────┼──────────────────┐ + │ │ │ + ▼ ▼ ▼ + ┌─────────────────┐ ┌─────────────────┐ ┌─────────────────┐ + │ PostgreSQL │ │ Redis │ │ S3/MinIO │ + │ (Primary) │ │ Cluster │ │ (Exports) │ + └─────────────────┘ └─────────────────┘ └─────────────────┘ +``` + +### 1.2 Kubernetes 部署架构 + +``` +┌─────────────────────────────────────────────────────────────┐ +│ Kubernetes Cluster │ +│ ┌─────────────────────────────────────────────────────┐ │ +│ │ Namespace: rwadurian │ │ +│ │ ┌─────────────┐ ┌─────────────┐ ┌────────────┐ │ │ +│ │ │ Ingress │ │ Service │ │ ConfigMap │ │ │ +│ │ │ Controller │ │ (ClusterIP)│ │ & Secrets │ │ │ +│ │ └──────┬──────┘ └──────┬──────┘ └────────────┘ │ │ +│ │ │ │ │ │ +│ │ ▼ ▼ │ │ +│ │ ┌─────────────────────────────────────────────┐ │ │ +│ │ │ Deployment │ │ │ +│ │ │ ┌─────────┐ ┌─────────┐ ┌─────────┐ │ │ │ +│ │ │ │ Pod 1 │ │ Pod 2 │ │ Pod 3 │ │ │ │ +│ │ │ └─────────┘ └─────────┘ └─────────┘ │ │ │ +│ │ └─────────────────────────────────────────────┘ │ │ +│ └─────────────────────────────────────────────────────┘ │ +└─────────────────────────────────────────────────────────────┘ +``` + +## 2. Docker 部署 + +### 2.1 生产 Dockerfile + +```dockerfile +# Dockerfile +# Build stage +FROM node:20-alpine AS builder + +WORKDIR /app + +# 安装构建依赖 +RUN apk add --no-cache openssl openssl-dev + +# 复制依赖文件 +COPY package*.json ./ +COPY prisma ./prisma/ + +# 安装依赖 +RUN npm ci --only=production && npm cache clean --force + +# 生成 Prisma Client +RUN npx prisma generate + +# 复制源代码 +COPY . . + +# 构建应用 +RUN npm run build + +# Production stage +FROM node:20-alpine AS production + +WORKDIR /app + +# 安装运行时依赖 +RUN apk add --no-cache openssl dumb-init + +# 创建非 root 用户 +RUN addgroup -g 1001 -S nodejs && \ + adduser -S nestjs -u 1001 + +# 从构建阶段复制必要文件 +COPY --from=builder --chown=nestjs:nodejs /app/dist ./dist +COPY --from=builder --chown=nestjs:nodejs /app/node_modules ./node_modules +COPY --from=builder --chown=nestjs:nodejs /app/prisma ./prisma +COPY --from=builder --chown=nestjs:nodejs /app/package*.json ./ + +# 切换到非 root 用户 +USER nestjs + +# 暴露端口 +EXPOSE 3000 + +# 健康检查 +HEALTHCHECK --interval=30s --timeout=10s --start-period=5s --retries=3 \ + CMD wget --no-verbose --tries=1 --spider http://localhost:3000/api/v1/health || exit 1 + +# 使用 dumb-init 启动 +ENTRYPOINT ["/usr/bin/dumb-init", "--"] +CMD ["node", "dist/main.js"] +``` + +### 2.2 Docker Compose (生产) + +```yaml +# docker-compose.yml +version: '3.8' + +services: + reporting-service: + build: + context: . + dockerfile: Dockerfile + container_name: reporting-service + restart: unless-stopped + ports: + - "3000:3000" + environment: + NODE_ENV: production + PORT: 3000 + DATABASE_URL: postgresql://${DB_USER}:${DB_PASSWORD}@postgres:5432/${DB_NAME}?schema=public + REDIS_HOST: redis + REDIS_PORT: 6379 + JWT_SECRET: ${JWT_SECRET} + depends_on: + postgres: + condition: service_healthy + redis: + condition: service_healthy + networks: + - rwadurian-network + deploy: + resources: + limits: + cpus: '1' + memory: 512M + reservations: + cpus: '0.5' + memory: 256M + + postgres: + image: postgres:15-alpine + container_name: reporting-postgres + restart: unless-stopped + environment: + POSTGRES_USER: ${DB_USER} + POSTGRES_PASSWORD: ${DB_PASSWORD} + POSTGRES_DB: ${DB_NAME} + volumes: + - postgres-data:/var/lib/postgresql/data + healthcheck: + test: ["CMD-SHELL", "pg_isready -U ${DB_USER}"] + interval: 10s + timeout: 5s + retries: 5 + networks: + - rwadurian-network + + redis: + image: redis:7-alpine + container_name: reporting-redis + restart: unless-stopped + command: redis-server --appendonly yes + volumes: + - redis-data:/data + healthcheck: + test: ["CMD", "redis-cli", "ping"] + interval: 10s + timeout: 5s + retries: 5 + networks: + - rwadurian-network + +volumes: + postgres-data: + redis-data: + +networks: + rwadurian-network: + driver: bridge +``` + +### 2.3 构建和运行 + +```bash +# 构建镜像 +docker build -t reporting-service:latest . + +# 运行容器 +docker compose up -d + +# 查看日志 +docker compose logs -f reporting-service + +# 检查健康状态 +docker compose ps +``` + +## 3. Kubernetes 部署 + +### 3.1 Namespace + +```yaml +# k8s/namespace.yaml +apiVersion: v1 +kind: Namespace +metadata: + name: rwadurian + labels: + app.kubernetes.io/name: rwadurian +``` + +### 3.2 ConfigMap + +```yaml +# k8s/configmap.yaml +apiVersion: v1 +kind: ConfigMap +metadata: + name: reporting-service-config + namespace: rwadurian +data: + NODE_ENV: "production" + PORT: "3000" + REDIS_HOST: "redis-service" + REDIS_PORT: "6379" +``` + +### 3.3 Secret + +```yaml +# k8s/secret.yaml +apiVersion: v1 +kind: Secret +metadata: + name: reporting-service-secret + namespace: rwadurian +type: Opaque +stringData: + DATABASE_URL: "postgresql://user:password@postgres-service:5432/rwadurian_reporting?schema=public" + JWT_SECRET: "your-super-secret-jwt-key" +``` + +### 3.4 Deployment + +```yaml +# k8s/deployment.yaml +apiVersion: apps/v1 +kind: Deployment +metadata: + name: reporting-service + namespace: rwadurian + labels: + app: reporting-service +spec: + replicas: 3 + selector: + matchLabels: + app: reporting-service + template: + metadata: + labels: + app: reporting-service + spec: + containers: + - name: reporting-service + image: reporting-service:latest + imagePullPolicy: Always + ports: + - containerPort: 3000 + name: http + envFrom: + - configMapRef: + name: reporting-service-config + - secretRef: + name: reporting-service-secret + resources: + requests: + cpu: "250m" + memory: "256Mi" + limits: + cpu: "1000m" + memory: "512Mi" + livenessProbe: + httpGet: + path: /api/v1/health + port: http + initialDelaySeconds: 30 + periodSeconds: 10 + timeoutSeconds: 5 + failureThreshold: 3 + readinessProbe: + httpGet: + path: /api/v1/health/ready + port: http + initialDelaySeconds: 5 + periodSeconds: 5 + timeoutSeconds: 3 + failureThreshold: 3 + lifecycle: + preStop: + exec: + command: ["/bin/sh", "-c", "sleep 10"] + terminationGracePeriodSeconds: 30 +``` + +### 3.5 Service + +```yaml +# k8s/service.yaml +apiVersion: v1 +kind: Service +metadata: + name: reporting-service + namespace: rwadurian + labels: + app: reporting-service +spec: + type: ClusterIP + ports: + - port: 80 + targetPort: 3000 + protocol: TCP + name: http + selector: + app: reporting-service +``` + +### 3.6 Ingress + +```yaml +# k8s/ingress.yaml +apiVersion: networking.k8s.io/v1 +kind: Ingress +metadata: + name: reporting-service-ingress + namespace: rwadurian + annotations: + kubernetes.io/ingress.class: nginx + nginx.ingress.kubernetes.io/ssl-redirect: "true" + nginx.ingress.kubernetes.io/proxy-body-size: "50m" +spec: + tls: + - hosts: + - reporting.rwadurian.com + secretName: reporting-tls-secret + rules: + - host: reporting.rwadurian.com + http: + paths: + - path: / + pathType: Prefix + backend: + service: + name: reporting-service + port: + number: 80 +``` + +### 3.7 HPA (Horizontal Pod Autoscaler) + +```yaml +# k8s/hpa.yaml +apiVersion: autoscaling/v2 +kind: HorizontalPodAutoscaler +metadata: + name: reporting-service-hpa + namespace: rwadurian +spec: + scaleTargetRef: + apiVersion: apps/v1 + kind: Deployment + name: reporting-service + minReplicas: 2 + maxReplicas: 10 + metrics: + - type: Resource + resource: + name: cpu + target: + type: Utilization + averageUtilization: 70 + - type: Resource + resource: + name: memory + target: + type: Utilization + averageUtilization: 80 +``` + +### 3.8 部署命令 + +```bash +# 创建命名空间 +kubectl apply -f k8s/namespace.yaml + +# 应用配置 +kubectl apply -f k8s/configmap.yaml +kubectl apply -f k8s/secret.yaml + +# 部署应用 +kubectl apply -f k8s/deployment.yaml +kubectl apply -f k8s/service.yaml +kubectl apply -f k8s/ingress.yaml +kubectl apply -f k8s/hpa.yaml + +# 检查状态 +kubectl get pods -n rwadurian +kubectl get svc -n rwadurian +kubectl get ingress -n rwadurian + +# 查看日志 +kubectl logs -f deployment/reporting-service -n rwadurian + +# 扩缩容 +kubectl scale deployment reporting-service --replicas=5 -n rwadurian +``` + +## 4. 数据库迁移 + +### 4.1 使用 Prisma Migrate + +```bash +# 创建迁移 +npx prisma migrate dev --name init + +# 应用迁移 (生产环境) +npx prisma migrate deploy + +# 重置数据库 (开发环境) +npx prisma migrate reset +``` + +### 4.2 迁移脚本 + +```bash +#!/bin/bash +# scripts/migrate.sh + +set -e + +echo "Running database migrations..." + +# 等待数据库就绪 +until pg_isready -h $DB_HOST -p $DB_PORT -U $DB_USER; do + echo "Waiting for database..." + sleep 2 +done + +# 运行迁移 +npx prisma migrate deploy + +echo "Migrations completed successfully!" +``` + +### 4.3 Kubernetes Job + +```yaml +# k8s/migration-job.yaml +apiVersion: batch/v1 +kind: Job +metadata: + name: reporting-service-migration + namespace: rwadurian +spec: + ttlSecondsAfterFinished: 100 + template: + spec: + containers: + - name: migration + image: reporting-service:latest + command: ["npx", "prisma", "migrate", "deploy"] + envFrom: + - secretRef: + name: reporting-service-secret + restartPolicy: Never + backoffLimit: 3 +``` + +## 5. CI/CD 流程 + +### 5.1 GitHub Actions + +```yaml +# .github/workflows/deploy.yml +name: Deploy + +on: + push: + branches: [main] + tags: ['v*'] + +env: + REGISTRY: ghcr.io + IMAGE_NAME: ${{ github.repository }}/reporting-service + +jobs: + test: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 + with: + node-version: '20' + cache: 'npm' + - run: npm ci + - run: npx prisma generate + - run: npm test + + build: + needs: test + runs-on: ubuntu-latest + permissions: + contents: read + packages: write + outputs: + image-tag: ${{ steps.meta.outputs.tags }} + steps: + - uses: actions/checkout@v4 + + - name: Set up Docker Buildx + uses: docker/setup-buildx-action@v3 + + - name: Log in to Container Registry + uses: docker/login-action@v3 + with: + registry: ${{ env.REGISTRY }} + username: ${{ github.actor }} + password: ${{ secrets.GITHUB_TOKEN }} + + - name: Extract metadata + id: meta + uses: docker/metadata-action@v5 + with: + images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }} + tags: | + type=ref,event=branch + type=ref,event=tag + type=sha,prefix={{branch}}- + + - name: Build and push + uses: docker/build-push-action@v5 + with: + context: . + push: true + tags: ${{ steps.meta.outputs.tags }} + labels: ${{ steps.meta.outputs.labels }} + cache-from: type=gha + cache-to: type=gha,mode=max + + deploy: + needs: build + runs-on: ubuntu-latest + if: github.ref == 'refs/heads/main' + steps: + - uses: actions/checkout@v4 + + - name: Set up kubectl + uses: azure/setup-kubectl@v3 + + - name: Configure kubeconfig + run: | + mkdir -p ~/.kube + echo "${{ secrets.KUBE_CONFIG }}" | base64 -d > ~/.kube/config + + - name: Update deployment + run: | + kubectl set image deployment/reporting-service \ + reporting-service=${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:main-${{ github.sha }} \ + -n rwadurian + kubectl rollout status deployment/reporting-service -n rwadurian +``` + +### 5.2 GitLab CI/CD + +```yaml +# .gitlab-ci.yml +stages: + - test + - build + - deploy + +variables: + DOCKER_IMAGE: $CI_REGISTRY_IMAGE/reporting-service + +test: + stage: test + image: node:20-alpine + script: + - npm ci + - npx prisma generate + - npm test + +build: + stage: build + image: docker:24 + services: + - docker:24-dind + script: + - docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY + - docker build -t $DOCKER_IMAGE:$CI_COMMIT_SHA . + - docker push $DOCKER_IMAGE:$CI_COMMIT_SHA + only: + - main + - tags + +deploy: + stage: deploy + image: bitnami/kubectl:latest + script: + - kubectl set image deployment/reporting-service reporting-service=$DOCKER_IMAGE:$CI_COMMIT_SHA -n rwadurian + - kubectl rollout status deployment/reporting-service -n rwadurian + only: + - main + environment: + name: production +``` + +## 6. 监控和日志 + +### 6.1 Prometheus 指标 + +```yaml +# k8s/servicemonitor.yaml +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: reporting-service + namespace: rwadurian +spec: + selector: + matchLabels: + app: reporting-service + endpoints: + - port: http + path: /metrics + interval: 30s +``` + +### 6.2 日志配置 + +```typescript +// src/main.ts +import { WinstonModule } from 'nest-winston'; +import * as winston from 'winston'; + +const app = await NestFactory.create(AppModule, { + logger: WinstonModule.createLogger({ + transports: [ + new winston.transports.Console({ + format: winston.format.combine( + winston.format.timestamp(), + winston.format.json(), + ), + }), + ], + }), +}); +``` + +### 6.3 Grafana 仪表板 + +关键指标: +- 请求速率 (requests/sec) +- 响应时间 (p50, p95, p99) +- 错误率 +- CPU/内存使用率 +- 数据库连接池状态 +- Redis 缓存命中率 + +## 7. 安全配置 + +### 7.1 网络策略 + +```yaml +# k8s/network-policy.yaml +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: reporting-service-network-policy + namespace: rwadurian +spec: + podSelector: + matchLabels: + app: reporting-service + policyTypes: + - Ingress + - Egress + ingress: + - from: + - namespaceSelector: + matchLabels: + name: ingress-nginx + ports: + - protocol: TCP + port: 3000 + egress: + - to: + - podSelector: + matchLabels: + app: postgres + ports: + - protocol: TCP + port: 5432 + - to: + - podSelector: + matchLabels: + app: redis + ports: + - protocol: TCP + port: 6379 +``` + +### 7.2 Pod 安全策略 + +```yaml +# k8s/pod-security.yaml +apiVersion: v1 +kind: Pod +metadata: + name: reporting-service +spec: + securityContext: + runAsNonRoot: true + runAsUser: 1001 + fsGroup: 1001 + containers: + - name: reporting-service + securityContext: + allowPrivilegeEscalation: false + readOnlyRootFilesystem: true + capabilities: + drop: + - ALL +``` + +## 8. 备份和恢复 + +### 8.1 数据库备份 + +```bash +#!/bin/bash +# scripts/backup.sh + +BACKUP_DIR="/backups" +TIMESTAMP=$(date +%Y%m%d_%H%M%S) +BACKUP_FILE="$BACKUP_DIR/reporting_${TIMESTAMP}.sql.gz" + +pg_dump -h $DB_HOST -U $DB_USER -d $DB_NAME | gzip > $BACKUP_FILE + +# 上传到 S3 +aws s3 cp $BACKUP_FILE s3://rwadurian-backups/reporting/ + +# 清理旧备份 (保留 7 天) +find $BACKUP_DIR -name "*.sql.gz" -mtime +7 -delete +``` + +### 8.2 Kubernetes CronJob + +```yaml +# k8s/backup-cronjob.yaml +apiVersion: batch/v1 +kind: CronJob +metadata: + name: reporting-db-backup + namespace: rwadurian +spec: + schedule: "0 2 * * *" # 每天凌晨 2 点 + jobTemplate: + spec: + template: + spec: + containers: + - name: backup + image: postgres:15-alpine + command: ["/scripts/backup.sh"] + envFrom: + - secretRef: + name: reporting-service-secret + volumeMounts: + - name: backup-scripts + mountPath: /scripts + volumes: + - name: backup-scripts + configMap: + name: backup-scripts + defaultMode: 0755 + restartPolicy: OnFailure +``` + +## 9. 故障排除 + +### 9.1 常见问题 + +**Q: Pod 启动失败** +```bash +# 查看 Pod 状态 +kubectl describe pod -n rwadurian + +# 查看日志 +kubectl logs -n rwadurian --previous +``` + +**Q: 数据库连接失败** +```bash +# 检查网络策略 +kubectl get networkpolicy -n rwadurian + +# 测试连接 +kubectl exec -it -n rwadurian -- nc -zv postgres-service 5432 +``` + +**Q: 内存不足** +```bash +# 查看资源使用 +kubectl top pod -n rwadurian + +# 增加资源限制 +kubectl patch deployment reporting-service -n rwadurian \ + -p '{"spec":{"template":{"spec":{"containers":[{"name":"reporting-service","resources":{"limits":{"memory":"1Gi"}}}]}}}}' +``` + +### 9.2 回滚 + +```bash +# 查看部署历史 +kubectl rollout history deployment/reporting-service -n rwadurian + +# 回滚到上一版本 +kubectl rollout undo deployment/reporting-service -n rwadurian + +# 回滚到指定版本 +kubectl rollout undo deployment/reporting-service --to-revision=2 -n rwadurian +``` diff --git a/backend/services/reporting-service/docs/DEVELOPMENT.md b/backend/services/reporting-service/docs/DEVELOPMENT.md new file mode 100644 index 00000000..fe6c08bc --- /dev/null +++ b/backend/services/reporting-service/docs/DEVELOPMENT.md @@ -0,0 +1,493 @@ +# 开发指南 + +## 1. 环境要求 + +### 1.1 必需软件 + +| 软件 | 版本 | 用途 | +|------|------|------| +| Node.js | 20.x LTS | 运行时环境 | +| npm | 10.x | 包管理器 | +| Docker | 24.x+ | 容器化 | +| Docker Compose | 2.x | 容器编排 | +| PostgreSQL | 15.x | 数据库 | +| Redis | 7.x | 缓存 | + +### 1.2 推荐 IDE + +- **VS Code** (推荐) + - 插件: ESLint, Prettier, Prisma, TypeScript +- **WebStorm** +- **Cursor** + +## 2. 项目设置 + +### 2.1 克隆项目 + +```bash +cd backend/services +git clone reporting-service +cd reporting-service +``` + +### 2.2 安装依赖 + +```bash +npm install +``` + +### 2.3 环境配置 + +创建 `.env.development` 文件: + +```bash +cp .env.example .env.development +``` + +编辑 `.env.development`: + +```env +# Application +NODE_ENV=development +PORT=3000 + +# Database +DATABASE_URL=postgresql://postgres:postgres@localhost:5432/rwadurian_reporting?schema=public + +# Redis +REDIS_HOST=localhost +REDIS_PORT=6379 + +# JWT +JWT_SECRET=your-development-secret-key + +# External Services (可选) +LEADERBOARD_SERVICE_URL=http://localhost:3001 +PLANTING_SERVICE_URL=http://localhost:3002 +``` + +### 2.4 数据库设置 + +启动 PostgreSQL (使用 Docker): + +```bash +docker run -d \ + --name reporting-postgres \ + -e POSTGRES_USER=postgres \ + -e POSTGRES_PASSWORD=postgres \ + -e POSTGRES_DB=rwadurian_reporting \ + -p 5432:5432 \ + postgres:15-alpine +``` + +生成 Prisma Client: + +```bash +npx prisma generate +``` + +同步数据库 Schema: + +```bash +npx prisma db push +``` + +(可选) 填充种子数据: + +```bash +npm run prisma:seed +``` + +### 2.5 启动 Redis + +```bash +docker run -d \ + --name reporting-redis \ + -p 6379:6379 \ + redis:7-alpine +``` + +## 3. 开发工作流 + +### 3.1 启动开发服务器 + +```bash +# 热重载模式 +npm run start:dev + +# 调试模式 +npm run start:debug +``` + +服务启动后访问: +- API: http://localhost:3000/api/v1 +- Swagger: http://localhost:3000/api (如已启用) + +### 3.2 常用命令 + +```bash +# 构建项目 +npm run build + +# 运行 Lint +npm run lint + +# 格式化代码 +npm run format + +# 运行测试 +npm test + +# 运行测试 (watch 模式) +npm run test:watch + +# 生成测试覆盖率 +npm run test:cov + +# Prisma Studio (数据库可视化) +npm run prisma:studio +``` + +### 3.3 使用 Makefile + +```bash +# 查看所有可用命令 +make help + +# 安装依赖 +make install + +# 构建项目 +make build + +# 运行所有测试 +make test + +# 运行单元测试 +make test-unit + +# 运行集成测试 +make test-integration + +# 运行 E2E 测试 +make test-e2e + +# Docker 测试 +make test-docker-all +``` + +## 4. 代码规范 + +### 4.1 目录结构规范 + +``` +src/ +├── api/ # API层 +│ ├── controllers/ # 每个资源一个控制器文件 +│ └── dto/ +│ ├── request/ # 请求 DTO +│ └── response/ # 响应 DTO +├── application/ # 应用层 +│ ├── commands/ # 每个命令独立目录 +│ │ └── xxx/ +│ │ ├── xxx.command.ts +│ │ └── xxx.handler.ts +│ └── queries/ # 每个查询独立目录 +├── domain/ # 领域层 +│ ├── aggregates/ # 每个聚合独立目录 +│ │ └── xxx/ +│ │ ├── index.ts +│ │ ├── xxx.aggregate.ts +│ │ └── xxx.spec.ts # 单元测试 +│ ├── value-objects/ # 值对象 +│ │ ├── xxx.vo.ts +│ │ └── xxx.spec.ts +│ └── repositories/ # 仅接口定义 +└── infrastructure/ # 基础设施层 +``` + +### 4.2 命名规范 + +**文件命名**: +- 使用 kebab-case: `report-definition.aggregate.ts` +- 后缀约定: + - `.aggregate.ts` - 聚合根 + - `.entity.ts` - 实体 + - `.vo.ts` - 值对象 + - `.service.ts` - 服务 + - `.controller.ts` - 控制器 + - `.dto.ts` - DTO + - `.spec.ts` - 单元测试 + - `.e2e-spec.ts` - E2E 测试 + - `.integration.spec.ts` - 集成测试 + +**类命名**: +- 使用 PascalCase +- 聚合根: `ReportSnapshot` +- 值对象: `DateRange` +- 服务: `ReportingApplicationService` +- 控制器: `ReportController` + +**接口命名**: +- 以 `I` 开头: `IReportSnapshotRepository` + +### 4.3 代码风格 + +**TypeScript 配置** (`tsconfig.json`): + +```json +{ + "compilerOptions": { + "strict": true, + "strictNullChecks": true, + "noImplicitAny": true, + "noUnusedLocals": true, + "noUnusedParameters": true + } +} +``` + +**ESLint 规则** (关键规则): + +```javascript +{ + rules: { + '@typescript-eslint/explicit-function-return-type': 'warn', + '@typescript-eslint/no-explicit-any': 'warn', + '@typescript-eslint/no-unused-vars': 'error', + 'no-console': 'warn' + } +} +``` + +### 4.4 领域驱动设计规范 + +**聚合根规则**: + +```typescript +// 1. 使用私有属性 + getter +export class ReportSnapshot { + private readonly _id: bigint; + private _snapshotData: SnapshotData; + + get id(): bigint { return this._id; } + get snapshotData(): SnapshotData { return this._snapshotData; } + + // 2. 使用工厂方法创建 + public static create(props: CreateProps): ReportSnapshot { + const snapshot = new ReportSnapshot(props); + snapshot.addDomainEvent(new SnapshotCreatedEvent(snapshot)); + return snapshot; + } + + // 3. 使用 reconstitute 重建 (不触发事件) + public static reconstitute(props: ReconstitutionProps): ReportSnapshot { + return new ReportSnapshot(props); + } + + // 4. 业务方法封装状态变更 + public updateData(newData: SnapshotData): void { + this.validateDataUpdate(newData); + this._snapshotData = newData; + this._updatedAt = new Date(); + } + + // 5. 私有构造函数 + private constructor(props: Props) { ... } +} +``` + +**值对象规则**: + +```typescript +// 1. 不可变 +export class DateRange { + private readonly _startDate: Date; + private readonly _endDate: Date; + + private constructor(startDate: Date, endDate: Date) { + this._startDate = startDate; + this._endDate = endDate; + Object.freeze(this); // 冻结对象 + } + + // 2. 工厂方法 + 验证 + public static create(startDate: Date, endDate: Date): DateRange { + if (startDate > endDate) { + throw new DomainException('Invalid date range'); + } + return new DateRange(startDate, endDate); + } + + // 3. 实现 equals 方法 + public equals(other: DateRange): boolean { + return this._startDate.getTime() === other._startDate.getTime() && + this._endDate.getTime() === other._endDate.getTime(); + } +} +``` + +## 5. Git 工作流 + +### 5.1 分支策略 + +``` +main # 生产分支 +├── develop # 开发分支 +│ ├── feature/xxx # 功能分支 +│ ├── bugfix/xxx # 修复分支 +│ └── refactor/xxx # 重构分支 +└── release/x.x.x # 发布分支 +``` + +### 5.2 提交规范 + +使用 Conventional Commits: + +```bash +# 格式 +(): + +# 示例 +feat(report): add daily report generation +fix(export): resolve Excel formatting issue +refactor(domain): extract value object for date range +test(e2e): add report generation tests +docs(api): update API documentation +``` + +**Type 类型**: +- `feat`: 新功能 +- `fix`: Bug 修复 +- `refactor`: 代码重构 +- `test`: 测试相关 +- `docs`: 文档更新 +- `chore`: 构建/配置变更 +- `perf`: 性能优化 + +### 5.3 PR 规范 + +PR 标题遵循提交规范,PR 描述需包含: + +```markdown +## Summary +简要说明本次变更内容 + +## Changes +- 变更点 1 +- 变更点 2 + +## Test Plan +- [ ] 单元测试通过 +- [ ] 集成测试通过 +- [ ] 手动测试场景 +``` + +## 6. 调试技巧 + +### 6.1 VS Code 调试配置 + +`.vscode/launch.json`: + +```json +{ + "version": "0.2.0", + "configurations": [ + { + "type": "node", + "request": "launch", + "name": "Debug NestJS", + "runtimeArgs": ["--nolazy", "-r", "ts-node/register"], + "args": ["${workspaceFolder}/src/main.ts"], + "sourceMaps": true, + "envFile": "${workspaceFolder}/.env.development" + }, + { + "type": "node", + "request": "launch", + "name": "Debug Jest Tests", + "program": "${workspaceFolder}/node_modules/.bin/jest", + "args": ["--runInBand", "--watchAll=false"], + "console": "integratedTerminal", + "internalConsoleOptions": "neverOpen" + } + ] +} +``` + +### 6.2 日志调试 + +```typescript +import { Logger } from '@nestjs/common'; + +@Injectable() +export class ReportingApplicationService { + private readonly logger = new Logger(ReportingApplicationService.name); + + async generateReport(dto: GenerateReportDto) { + this.logger.debug(`Generating report: ${dto.reportCode}`); + this.logger.log(`Report generated successfully`); + this.logger.warn(`Performance warning: slow query`); + this.logger.error(`Failed to generate report`, error.stack); + } +} +``` + +### 6.3 Prisma 调试 + +启用查询日志: + +```typescript +// prisma.service.ts +const prisma = new PrismaClient({ + log: [ + { level: 'query', emit: 'event' }, + { level: 'error', emit: 'stdout' }, + { level: 'warn', emit: 'stdout' }, + ], +}); + +prisma.$on('query', (e) => { + console.log('Query: ' + e.query); + console.log('Params: ' + e.params); + console.log('Duration: ' + e.duration + 'ms'); +}); +``` + +## 7. 常见问题 + +### Q: Prisma Client 未生成 + +```bash +npx prisma generate +``` + +### Q: 数据库连接失败 + +1. 检查 PostgreSQL 是否运行: `docker ps` +2. 检查 DATABASE_URL 环境变量 +3. 检查网络连接和端口 + +### Q: 测试失败 - 表不存在 + +```bash +# 推送 schema 到测试数据库 +DATABASE_URL="postgresql://..." npx prisma db push +``` + +### Q: TypeScript 编译错误 + +```bash +# 清理并重新构建 +rm -rf dist +npm run build +``` + +### Q: 端口被占用 + +```bash +# 查找占用端口的进程 +lsof -i :3000 +# 或在 Windows +netstat -ano | findstr :3000 +``` diff --git a/backend/services/reporting-service/docs/README.md b/backend/services/reporting-service/docs/README.md new file mode 100644 index 00000000..f66d4f63 --- /dev/null +++ b/backend/services/reporting-service/docs/README.md @@ -0,0 +1,45 @@ +# Reporting Service 文档索引 + +## 概述 + +Reporting Service 是 RWA Durian 平台的报表与分析服务,负责生成、存储和导出各类业务报表。 + +## 文档目录 + +| 文档 | 描述 | +|------|------| +| [架构设计](./ARCHITECTURE.md) | 服务整体架构、DDD分层、六边形架构 | +| [API参考](./API.md) | RESTful API 接口详细说明 | +| [开发指南](./DEVELOPMENT.md) | 本地开发环境搭建、代码规范 | +| [测试指南](./TESTING.md) | 单元测试、集成测试、E2E测试说明 | +| [部署指南](./DEPLOYMENT.md) | 容器化部署、环境配置、CI/CD | +| [数据模型](./DATA-MODEL.md) | 数据库表结构、实体关系 | + +## 快速开始 + +```bash +# 安装依赖 +npm install + +# 生成 Prisma Client +npx prisma generate + +# 启动开发服务器 +npm run start:dev + +# 运行测试 +npm test +``` + +## 技术栈 + +- **框架**: NestJS 10.x +- **语言**: TypeScript 5.x +- **数据库**: PostgreSQL 15 + Prisma ORM +- **缓存**: Redis 7 +- **消息队列**: Kafka (可选) +- **导出格式**: Excel, CSV, PDF, JSON + +## 联系方式 + +如有问题,请联系开发团队或提交 Issue。 diff --git a/backend/services/reporting-service/docs/TESTING.md b/backend/services/reporting-service/docs/TESTING.md new file mode 100644 index 00000000..651158a2 --- /dev/null +++ b/backend/services/reporting-service/docs/TESTING.md @@ -0,0 +1,888 @@ +# 测试指南 + +## 1. 测试策略概述 + +### 1.1 测试金字塔 + +``` + ┌───────────┐ + │ E2E │ ← 端到端测试 (少量) + │ Tests │ + ─┴───────────┴─ + ┌───────────────┐ + │ Integration │ ← 集成测试 (中等) + │ Tests │ + ─┴───────────────┴─ + ┌───────────────────┐ + │ Unit Tests │ ← 单元测试 (大量) + └───────────────────┘ +``` + +### 1.2 测试类型分布 + +| 测试类型 | 数量 | 覆盖范围 | 执行时间 | +|---------|------|---------|---------| +| 单元测试 | 20+ | 领域层、值对象、聚合根 | ~2s | +| 集成测试 | 11+ | 数据库仓储、应用服务 | ~10s | +| E2E 测试 | 12+ | HTTP API、完整流程 | ~15s | +| Docker 测试 | 31+ | 容器化环境全量测试 | ~60s | + +## 2. 单元测试 + +### 2.1 测试范围 + +单元测试主要覆盖: +- **值对象 (Value Objects)**: 不可变性、验证逻辑、相等性比较 +- **聚合根 (Aggregates)**: 业务规则、状态变更、领域事件 +- **领域服务**: 业务逻辑计算 + +### 2.2 目录结构 + +``` +src/ +├── domain/ +│ ├── value-objects/ +│ │ ├── date-range.vo.ts +│ │ ├── date-range.spec.ts ← 值对象单元测试 +│ │ ├── report-period.vo.ts +│ │ ├── report-period.spec.ts +│ │ ├── export-format.vo.ts +│ │ └── export-format.spec.ts +│ └── aggregates/ +│ ├── report-definition/ +│ │ ├── report-definition.aggregate.ts +│ │ └── report-definition.spec.ts ← 聚合根单元测试 +│ └── report-snapshot/ +│ ├── report-snapshot.aggregate.ts +│ └── report-snapshot.spec.ts +``` + +### 2.3 值对象测试示例 + +```typescript +// src/domain/value-objects/date-range.spec.ts +describe('DateRange Value Object', () => { + describe('create', () => { + it('should create valid date range', () => { + const startDate = new Date('2024-01-01'); + const endDate = new Date('2024-01-31'); + + const dateRange = DateRange.create(startDate, endDate); + + expect(dateRange.startDate).toEqual(startDate); + expect(dateRange.endDate).toEqual(endDate); + }); + + it('should throw error when start date is after end date', () => { + const startDate = new Date('2024-01-31'); + const endDate = new Date('2024-01-01'); + + expect(() => DateRange.create(startDate, endDate)) + .toThrow('Start date cannot be after end date'); + }); + }); + + describe('equals', () => { + it('should return true for equal date ranges', () => { + const range1 = DateRange.create( + new Date('2024-01-01'), + new Date('2024-01-31') + ); + const range2 = DateRange.create( + new Date('2024-01-01'), + new Date('2024-01-31') + ); + + expect(range1.equals(range2)).toBe(true); + }); + }); + + describe('getDays', () => { + it('should calculate correct number of days', () => { + const dateRange = DateRange.create( + new Date('2024-01-01'), + new Date('2024-01-31') + ); + + expect(dateRange.getDays()).toBe(31); + }); + }); +}); +``` + +### 2.4 聚合根测试示例 + +```typescript +// src/domain/aggregates/report-snapshot/report-snapshot.spec.ts +describe('ReportSnapshot Aggregate', () => { + const mockDefinition = ReportDefinition.reconstitute({ + id: 1n, + code: 'RPT_TEST', + name: 'Test Report', + description: 'Test description', + category: 'TEST', + dataSource: 'test_table', + isActive: true, + createdAt: new Date(), + updatedAt: new Date(), + }); + + describe('create', () => { + it('should create snapshot with pending status', () => { + const snapshot = ReportSnapshot.create({ + definition: mockDefinition, + period: ReportPeriod.DAILY, + dateRange: DateRange.create( + new Date('2024-01-01'), + new Date('2024-01-01') + ), + }); + + expect(snapshot.status).toBe(SnapshotStatus.PENDING); + expect(snapshot.reportCode).toBe('RPT_TEST'); + }); + }); + + describe('markAsProcessing', () => { + it('should transition from pending to processing', () => { + const snapshot = createPendingSnapshot(); + + snapshot.markAsProcessing(); + + expect(snapshot.status).toBe(SnapshotStatus.PROCESSING); + }); + + it('should throw error if not in pending status', () => { + const snapshot = createCompletedSnapshot(); + + expect(() => snapshot.markAsProcessing()) + .toThrow('Cannot start processing'); + }); + }); + + describe('complete', () => { + it('should set data and mark as completed', () => { + const snapshot = createProcessingSnapshot(); + const data = { items: [{ id: 1 }], total: 1 }; + + snapshot.complete(data); + + expect(snapshot.status).toBe(SnapshotStatus.COMPLETED); + expect(snapshot.snapshotData).toEqual(data); + }); + }); +}); +``` + +### 2.5 运行单元测试 + +```bash +# 运行所有单元测试 +npm test + +# 运行特定文件 +npm test -- date-range.spec.ts + +# Watch 模式 +npm run test:watch + +# 带覆盖率 +npm run test:cov +``` + +## 3. 集成测试 + +### 3.1 测试范围 + +集成测试覆盖: +- **仓储实现**: Prisma 数据库操作 +- **应用服务**: 命令/查询处理 +- **缓存服务**: Redis 缓存操作 + +### 3.2 目录结构 + +``` +src/ +├── infrastructure/ +│ └── repositories/ +│ ├── prisma-report-definition.repository.ts +│ └── prisma-report-definition.repository.integration.spec.ts +├── application/ +│ └── services/ +│ ├── reporting-application.service.ts +│ └── reporting-application.service.integration.spec.ts +``` + +### 3.3 数据库集成测试 + +```typescript +// prisma-report-definition.repository.integration.spec.ts +describe('PrismaReportDefinitionRepository (Integration)', () => { + let repository: PrismaReportDefinitionRepository; + let prismaService: PrismaService; + + beforeAll(async () => { + const module = await Test.createTestingModule({ + providers: [ + PrismaReportDefinitionRepository, + PrismaService, + ], + }).compile(); + + repository = module.get(PrismaReportDefinitionRepository); + prismaService = module.get(PrismaService); + }); + + beforeEach(async () => { + // 清理测试数据 + await prismaService.reportSnapshot.deleteMany(); + await prismaService.reportDefinition.deleteMany(); + }); + + afterAll(async () => { + await prismaService.$disconnect(); + }); + + describe('save', () => { + it('should persist report definition to database', async () => { + const definition = ReportDefinition.create({ + code: 'RPT_TEST', + name: 'Test Report', + description: 'Test', + category: 'TEST', + dataSource: 'test_table', + }); + + await repository.save(definition); + + const found = await prismaService.reportDefinition.findUnique({ + where: { code: 'RPT_TEST' }, + }); + expect(found).not.toBeNull(); + expect(found?.name).toBe('Test Report'); + }); + }); + + describe('findByCode', () => { + it('should return null for non-existent code', async () => { + const result = await repository.findByCode('NON_EXISTENT'); + expect(result).toBeNull(); + }); + + it('should return definition when exists', async () => { + // 创建测试数据 + await prismaService.reportDefinition.create({ + data: { + code: 'RPT_FIND', + name: 'Find Test', + description: 'Test', + category: 'TEST', + dataSource: 'test_table', + isActive: true, + }, + }); + + const result = await repository.findByCode('RPT_FIND'); + + expect(result).not.toBeNull(); + expect(result?.code).toBe('RPT_FIND'); + }); + }); +}); +``` + +### 3.4 应用服务集成测试 + +```typescript +// reporting-application.service.integration.spec.ts +describe('ReportingApplicationService (Integration)', () => { + let service: ReportingApplicationService; + let prismaService: PrismaService; + + beforeAll(async () => { + const module = await Test.createTestingModule({ + imports: [AppModule], + }).compile(); + + service = module.get(ReportingApplicationService); + prismaService = module.get(PrismaService); + }); + + beforeEach(async () => { + await cleanDatabase(prismaService); + await seedTestData(prismaService); + }); + + describe('generateReport', () => { + it('should create snapshot for valid report', async () => { + const result = await service.generateReport({ + reportCode: 'RPT_LEADERBOARD', + reportPeriod: ReportPeriod.DAILY, + startDate: new Date('2024-01-01'), + endDate: new Date('2024-01-01'), + }); + + expect(result).toBeDefined(); + expect(result.status).toBe(SnapshotStatus.COMPLETED); + }); + + it('should throw when report definition not found', async () => { + await expect( + service.generateReport({ + reportCode: 'NON_EXISTENT', + reportPeriod: ReportPeriod.DAILY, + startDate: new Date('2024-01-01'), + endDate: new Date('2024-01-01'), + }) + ).rejects.toThrow(ReportDefinitionNotFoundException); + }); + }); +}); +``` + +### 3.5 测试数据库配置 + +```env +# .env.test +DATABASE_URL=postgresql://postgres:postgres@localhost:5433/rwadurian_reporting_test?schema=public +REDIS_HOST=localhost +REDIS_PORT=6380 +``` + +### 3.6 运行集成测试 + +```bash +# 启动测试依赖 +docker compose -f docker-compose.test.yml up -d postgres-test redis-test + +# 推送 Schema +DATABASE_URL="postgresql://postgres:postgres@localhost:5433/rwadurian_reporting_test?schema=public" \ + npx prisma db push + +# 运行集成测试 +npm run test:integration + +# 或使用 Makefile +make test-integration +``` + +## 4. E2E 测试 + +### 4.1 测试范围 + +E2E 测试覆盖: +- HTTP API 端点 +- 请求验证 +- 响应格式 +- 错误处理 +- 完整业务流程 + +### 4.2 目录结构 + +``` +test/ +├── app.e2e-spec.ts # 主 E2E 测试文件 +├── jest-e2e.json # E2E Jest 配置 +└── setup-e2e.ts # E2E 测试设置 +``` + +### 4.3 E2E 测试配置 + +```json +// test/jest-e2e.json +{ + "moduleFileExtensions": ["js", "json", "ts"], + "rootDir": ".", + "testEnvironment": "node", + "testRegex": ".e2e-spec.ts$", + "testTimeout": 30000, + "transform": { + "^.+\\.(t|j)s$": "ts-jest" + }, + "setupFilesAfterEnv": ["/setup-e2e.ts"], + "moduleNameMapper": { + "^@/(.*)$": "/../src/$1", + "^@domain/(.*)$": "/../src/domain/$1", + "^@application/(.*)$": "/../src/application/$1", + "^@infrastructure/(.*)$": "/../src/infrastructure/$1", + "^@api/(.*)$": "/../src/api/$1" + } +} +``` + +```typescript +// test/setup-e2e.ts +import * as dotenv from 'dotenv'; +import * as path from 'path'; + +dotenv.config({ path: path.resolve(__dirname, '../.env.test') }); +``` + +### 4.4 E2E 测试示例 + +```typescript +// test/app.e2e-spec.ts +describe('Reporting Service (e2e)', () => { + let app: INestApplication; + + beforeAll(async () => { + const moduleFixture = await Test.createTestingModule({ + imports: [AppModule], + }).compile(); + + app = moduleFixture.createNestApplication(); + app.setGlobalPrefix('api/v1'); + app.useGlobalPipes( + new ValidationPipe({ + whitelist: true, + forbidNonWhitelisted: true, + transform: true, + }), + ); + // 注意: TransformInterceptor 已在 AppModule 中注册 + // 不要重复注册,避免响应双重包装 + await app.init(); + }); + + afterAll(async () => { + await app.close(); + }); + + describe('Health Check', () => { + it('/api/v1/health (GET) should return ok status', () => { + return request(app.getHttpServer()) + .get('/api/v1/health') + .expect(200) + .expect((res) => { + expect(res.body.success).toBe(true); + expect(res.body.data.status).toBe('ok'); + expect(res.body.data.service).toBe('reporting-service'); + }); + }); + }); + + describe('Reports API', () => { + describe('POST /api/v1/reports/generate', () => { + it('should validate request body', () => { + return request(app.getHttpServer()) + .post('/api/v1/reports/generate') + .send({}) + .expect(400); + }); + + it('should reject invalid report period', () => { + return request(app.getHttpServer()) + .post('/api/v1/reports/generate') + .send({ + reportCode: 'RPT_LEADERBOARD', + reportPeriod: 'INVALID_PERIOD', + startDate: '2024-01-01', + endDate: '2024-01-31', + }) + .expect(400); + }); + }); + }); + + describe('Export API', () => { + describe('POST /api/v1/export', () => { + it('should validate request body', () => { + return request(app.getHttpServer()) + .post('/api/v1/export') + .send({}) + .expect(400); + }); + + it('should reject invalid format', () => { + return request(app.getHttpServer()) + .post('/api/v1/export') + .send({ + snapshotId: '1', + format: 'INVALID_FORMAT', + }) + .expect(400); + }); + }); + }); +}); +``` + +### 4.5 运行 E2E 测试 + +```bash +# 确保测试数据库运行 +docker compose -f docker-compose.test.yml up -d postgres-test redis-test + +# 运行 E2E 测试 +npm run test:e2e + +# 或使用 Makefile +make test-e2e +``` + +## 5. Docker 测试 + +### 5.1 测试架构 + +``` +┌─────────────────────────────────────────────────────────┐ +│ Docker Network │ +│ ┌──────────────┐ ┌──────────────┐ ┌──────────────┐ │ +│ │ PostgreSQL │ │ Redis │ │ Service │ │ +│ │ :5432 │ │ :6379 │ │ Test │ │ +│ │ (tmpfs) │ │ │ │ Container │ │ +│ └──────────────┘ └──────────────┘ └──────────────┘ │ +└─────────────────────────────────────────────────────────┘ +``` + +### 5.2 Docker Compose 配置 + +```yaml +# docker-compose.test.yml +version: '3.8' + +services: + postgres-test: + image: postgres:15-alpine + container_name: reporting-postgres-test + environment: + POSTGRES_USER: postgres + POSTGRES_PASSWORD: postgres + POSTGRES_DB: rwadurian_reporting_test + ports: + - "5433:5432" + healthcheck: + test: ["CMD-SHELL", "pg_isready -U postgres"] + interval: 5s + timeout: 5s + retries: 5 + tmpfs: + - /var/lib/postgresql/data # 使用内存存储加速测试 + + redis-test: + image: redis:7-alpine + container_name: reporting-redis-test + ports: + - "6380:6379" + healthcheck: + test: ["CMD", "redis-cli", "ping"] + interval: 5s + timeout: 5s + retries: 5 + + reporting-service-test: + build: + context: . + dockerfile: Dockerfile.test + container_name: reporting-service-test + depends_on: + postgres-test: + condition: service_healthy + redis-test: + condition: service_healthy + environment: + NODE_ENV: test + DATABASE_URL: postgresql://postgres:postgres@postgres-test:5432/rwadurian_reporting_test?schema=public + REDIS_HOST: redis-test + REDIS_PORT: 6379 + JWT_SECRET: test-secret-key + volumes: + - ./coverage:/app/coverage + command: sh -c "npx prisma db push --skip-generate && npm run test:cov" +``` + +### 5.3 测试 Dockerfile + +```dockerfile +# Dockerfile.test +FROM node:20-alpine + +WORKDIR /app + +# 安装 OpenSSL (Prisma 依赖) +RUN apk add --no-cache openssl openssl-dev + +# 安装依赖 +COPY package*.json ./ +RUN npm ci + +# 复制 Prisma Schema +COPY prisma ./prisma/ + +# 生成 Prisma Client +RUN npx prisma generate + +# 复制源代码 +COPY . . + +# 构建应用 +RUN npm run build + +# 默认运行测试 +CMD ["npm", "test"] +``` + +### 5.4 .dockerignore + +``` +node_modules +dist +coverage +.git +.env* +!.env.example +*.log +.claude/ +``` + +### 5.5 运行 Docker 测试 + +```bash +# 构建并运行所有测试 +docker compose -f docker-compose.test.yml up --build --abort-on-container-exit + +# 查看测试结果 +docker compose -f docker-compose.test.yml logs reporting-service-test + +# 清理 +docker compose -f docker-compose.test.yml down -v + +# 使用 Makefile +make test-docker-all +``` + +## 6. Makefile 命令 + +```makefile +# Makefile +.PHONY: test test-unit test-integration test-e2e test-docker-all + +# 运行所有测试 +test: test-unit test-integration test-e2e + +# 单元测试 +test-unit: + npm test + +# 集成测试 +test-integration: + npm run test:integration + +# E2E 测试 +test-e2e: + npm run test:e2e + +# Docker 中运行所有测试 +test-docker-all: + docker compose -f docker-compose.test.yml up --build --abort-on-container-exit + docker compose -f docker-compose.test.yml down -v + +# 测试覆盖率 +test-cov: + npm run test:cov + +# 清理测试容器 +test-clean: + docker compose -f docker-compose.test.yml down -v +``` + +## 7. 测试最佳实践 + +### 7.1 测试命名规范 + +```typescript +describe('ClassName', () => { + describe('methodName', () => { + it('should [expected behavior] when [condition]', () => { + // test implementation + }); + }); +}); +``` + +### 7.2 AAA 模式 + +```typescript +it('should create valid date range', () => { + // Arrange - 准备测试数据 + const startDate = new Date('2024-01-01'); + const endDate = new Date('2024-01-31'); + + // Act - 执行被测试的操作 + const dateRange = DateRange.create(startDate, endDate); + + // Assert - 验证结果 + expect(dateRange.startDate).toEqual(startDate); + expect(dateRange.endDate).toEqual(endDate); +}); +``` + +### 7.3 测试隔离 + +```typescript +describe('Repository Integration Tests', () => { + beforeEach(async () => { + // 每个测试前清理数据 + await prismaService.reportSnapshot.deleteMany(); + await prismaService.reportDefinition.deleteMany(); + }); + + afterAll(async () => { + // 测试结束后断开连接 + await prismaService.$disconnect(); + }); +}); +``` + +### 7.4 Mock 使用 + +```typescript +// 使用 Jest mock +const mockRepository = { + findByCode: jest.fn(), + save: jest.fn(), + findAll: jest.fn(), +}; + +beforeEach(() => { + jest.clearAllMocks(); +}); + +it('should call repository with correct parameters', async () => { + mockRepository.findByCode.mockResolvedValue(mockDefinition); + + await service.getReportDefinition('RPT_TEST'); + + expect(mockRepository.findByCode).toHaveBeenCalledWith('RPT_TEST'); +}); +``` + +## 8. CI/CD 集成 + +### 8.1 GitHub Actions + +```yaml +# .github/workflows/test.yml +name: Test + +on: + push: + branches: [main, develop] + pull_request: + branches: [main] + +jobs: + test: + runs-on: ubuntu-latest + + services: + postgres: + image: postgres:15-alpine + env: + POSTGRES_USER: postgres + POSTGRES_PASSWORD: postgres + POSTGRES_DB: rwadurian_reporting_test + ports: + - 5432:5432 + options: >- + --health-cmd pg_isready + --health-interval 10s + --health-timeout 5s + --health-retries 5 + + redis: + image: redis:7-alpine + ports: + - 6379:6379 + options: >- + --health-cmd "redis-cli ping" + --health-interval 10s + --health-timeout 5s + --health-retries 5 + + steps: + - uses: actions/checkout@v4 + + - name: Setup Node.js + uses: actions/setup-node@v4 + with: + node-version: '20' + cache: 'npm' + + - name: Install dependencies + run: npm ci + + - name: Generate Prisma Client + run: npx prisma generate + + - name: Push database schema + run: npx prisma db push + env: + DATABASE_URL: postgresql://postgres:postgres@localhost:5432/rwadurian_reporting_test?schema=public + + - name: Run unit tests + run: npm test + + - name: Run integration tests + run: npm run test:integration + env: + DATABASE_URL: postgresql://postgres:postgres@localhost:5432/rwadurian_reporting_test?schema=public + REDIS_HOST: localhost + REDIS_PORT: 6379 + + - name: Run E2E tests + run: npm run test:e2e + env: + DATABASE_URL: postgresql://postgres:postgres@localhost:5432/rwadurian_reporting_test?schema=public + REDIS_HOST: localhost + REDIS_PORT: 6379 + + - name: Upload coverage + uses: codecov/codecov-action@v3 + with: + files: ./coverage/lcov.info +``` + +## 9. 故障排除 + +### 9.1 常见问题 + +**Q: 测试找不到模块** +```bash +# 重新生成 Prisma Client +npx prisma generate +``` + +**Q: 数据库连接失败** +```bash +# 检查容器状态 +docker ps + +# 检查数据库日志 +docker logs reporting-postgres-test +``` + +**Q: E2E 测试响应格式错误** +```typescript +// 确保不要重复注册 TransformInterceptor +// AppModule 中已注册,测试中不需要再次注册 +``` + +**Q: Docker 测试 Prisma 错误** +```dockerfile +# 确保 Dockerfile.test 包含 OpenSSL +RUN apk add --no-cache openssl openssl-dev +``` + +### 9.2 调试测试 + +```bash +# 运行单个测试文件 +npm test -- --testPathPattern=date-range.spec.ts + +# 详细输出 +npm test -- --verbose + +# 调试模式 +node --inspect-brk node_modules/.bin/jest --runInBand +``` diff --git a/backend/services/reporting-service/nest-cli.json b/backend/services/reporting-service/nest-cli.json new file mode 100644 index 00000000..f9aa683b --- /dev/null +++ b/backend/services/reporting-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/reporting-service/package-lock.json b/backend/services/reporting-service/package-lock.json new file mode 100644 index 00000000..df85f60e --- /dev/null +++ b/backend/services/reporting-service/package-lock.json @@ -0,0 +1,11805 @@ +{ + "name": "reporting-service", + "version": "1.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "reporting-service", + "version": "1.0.0", + "license": "UNLICENSED", + "dependencies": { + "@nestjs/common": "^10.0.0", + "@nestjs/config": "^3.1.1", + "@nestjs/core": "^10.0.0", + "@nestjs/jwt": "^10.2.0", + "@nestjs/microservices": "^10.3.0", + "@nestjs/passport": "^10.0.3", + "@nestjs/platform-express": "^10.0.0", + "@nestjs/schedule": "^4.0.0", + "@nestjs/swagger": "^7.1.17", + "@prisma/client": "^5.7.0", + "class-transformer": "^0.5.1", + "class-validator": "^0.14.0", + "csv-stringify": "^6.4.4", + "exceljs": "^4.4.0", + "ioredis": "^5.3.2", + "kafkajs": "^2.2.4", + "passport": "^0.7.0", + "passport-jwt": "^4.0.1", + "pdfkit": "^0.14.0", + "reflect-metadata": "^0.1.13", + "rxjs": "^7.8.1", + "uuid": "^9.0.1" + }, + "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/pdfkit": "^0.13.3", + "@types/supertest": "^2.0.12", + "@types/uuid": "^9.0.7", + "@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/@fast-csv/format": { + "version": "4.3.5", + "resolved": "https://registry.npmjs.org/@fast-csv/format/-/format-4.3.5.tgz", + "integrity": "sha512-8iRn6QF3I8Ak78lNAa+Gdl5MJJBM5vRHivFtMRUWINdevNo00K7OXxS2PshawLKTejVwieIlPmK5YlLu6w4u8A==", + "license": "MIT", + "dependencies": { + "@types/node": "^14.0.1", + "lodash.escaperegexp": "^4.1.2", + "lodash.isboolean": "^3.0.3", + "lodash.isequal": "^4.5.0", + "lodash.isfunction": "^3.0.9", + "lodash.isnil": "^4.0.0" + } + }, + "node_modules/@fast-csv/format/node_modules/@types/node": { + "version": "14.18.63", + "resolved": "https://registry.npmjs.org/@types/node/-/node-14.18.63.tgz", + "integrity": "sha512-fAtCfv4jJg+ExtXhvCkCqUKZ+4ok/JQk01qDKhL5BDDoS3AxKXhV5/MAVUZyQnSEd2GT92fkgZl0pz0Q0AzcIQ==", + "license": "MIT" + }, + "node_modules/@fast-csv/parse": { + "version": "4.3.6", + "resolved": "https://registry.npmjs.org/@fast-csv/parse/-/parse-4.3.6.tgz", + "integrity": "sha512-uRsLYksqpbDmWaSmzvJcuApSEe38+6NQZBUsuAyMZKqHxH0g1wcJgsKUvN3WC8tewaqFjBMMGrkHmC+T7k8LvA==", + "license": "MIT", + "dependencies": { + "@types/node": "^14.0.1", + "lodash.escaperegexp": "^4.1.2", + "lodash.groupby": "^4.6.0", + "lodash.isfunction": "^3.0.9", + "lodash.isnil": "^4.0.0", + "lodash.isundefined": "^3.0.1", + "lodash.uniq": "^4.5.0" + } + }, + "node_modules/@fast-csv/parse/node_modules/@types/node": { + "version": "14.18.63", + "resolved": "https://registry.npmjs.org/@types/node/-/node-14.18.63.tgz", + "integrity": "sha512-fAtCfv4jJg+ExtXhvCkCqUKZ+4ok/JQk01qDKhL5BDDoS3AxKXhV5/MAVUZyQnSEd2GT92fkgZl0pz0Q0AzcIQ==", + "license": "MIT" + }, + "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/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/schedule": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/@nestjs/schedule/-/schedule-4.1.2.tgz", + "integrity": "sha512-hCTQ1lNjIA5EHxeu8VvQu2Ed2DBLS1GSC6uKPYlBiQe6LL9a7zfE9iVSK+zuK8E2odsApteEBmfAQchc8Hx0Gg==", + "license": "MIT", + "dependencies": { + "cron": "3.2.1", + "uuid": "11.0.3" + }, + "peerDependencies": { + "@nestjs/common": "^8.0.0 || ^9.0.0 || ^10.0.0", + "@nestjs/core": "^8.0.0 || ^9.0.0 || ^10.0.0" + } + }, + "node_modules/@nestjs/schedule/node_modules/uuid": { + "version": "11.0.3", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-11.0.3.tgz", + "integrity": "sha512-d0z310fCWv5dJwnX1Y/MncBAqGMKEzlBb1AOf7z9K8ALnd0utBX/msg/fA0+sbyN1ihbMsLhrBlnl1ak7Wa0rg==", + "funding": [ + "https://github.com/sponsors/broofa", + "https://github.com/sponsors/ctavan" + ], + "license": "MIT", + "bin": { + "uuid": "dist/esm/bin/uuid" + } + }, + "node_modules/@nestjs/schematics": { + "version": "10.2.3", + "resolved": "https://registry.npmjs.org/@nestjs/schematics/-/schematics-10.2.3.tgz", + "integrity": "sha512-4e8gxaCk7DhBxVUly2PjYL4xC2ifDFexCqq1/u4TtivLGXotVk0wHdYuPYe1tHTHuR1lsOkRbfOCpkdTnigLVg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@angular-devkit/core": "17.3.11", + "@angular-devkit/schematics": "17.3.11", + "comment-json": "4.2.5", + "jsonc-parser": "3.3.1", + "pluralize": "8.0.0" + }, + "peerDependencies": { + "typescript": ">=4.8.2" + } + }, + "node_modules/@nestjs/schematics/node_modules/jsonc-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/jsonc-parser/-/jsonc-parser-3.3.1.tgz", + "integrity": "sha512-HUgH65KyejrUFPvHFPbqOY0rsFip3Bo5wb4ngvdi1EpCYWUQDC5V+Y7mZws+DLkr4M//zQJoanu1SP+87Dv1oQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@nestjs/swagger": { + "version": "7.4.2", + "resolved": "https://registry.npmjs.org/@nestjs/swagger/-/swagger-7.4.2.tgz", + "integrity": "sha512-Mu6TEn1M/owIvAx2B4DUQObQXqo2028R2s9rSZ/hJEgBK95+doTwS0DjmVA2wTeZTyVtXOoN7CsoM5pONBzvKQ==", + "license": "MIT", + "dependencies": { + "@microsoft/tsdoc": "^0.15.0", + "@nestjs/mapped-types": "2.0.5", + "js-yaml": "4.1.0", + "lodash": "4.17.21", + "path-to-regexp": "3.3.0", + "swagger-ui-dist": "5.17.14" + }, + "peerDependencies": { + "@fastify/static": "^6.0.0 || ^7.0.0", + "@nestjs/common": "^9.0.0 || ^10.0.0", + "@nestjs/core": "^9.0.0 || ^10.0.0", + "class-transformer": "*", + "class-validator": "*", + "reflect-metadata": "^0.1.12 || ^0.2.0" + }, + "peerDependenciesMeta": { + "@fastify/static": { + "optional": true + }, + "class-transformer": { + "optional": true + }, + "class-validator": { + "optional": true + } + } + }, + "node_modules/@nestjs/testing": { + "version": "10.4.20", + "resolved": "https://registry.npmjs.org/@nestjs/testing/-/testing-10.4.20.tgz", + "integrity": "sha512-nMkRDukDKskdPruM6EsgMq7yJua+CPZM6I6FrLP8yXw8BiVSPv9Nm0CtcGGwt3kgZF9hfxKjGqLjsvVBsv6Vfw==", + "dev": true, + "license": "MIT", + "dependencies": { + "tslib": "2.8.1" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/nest" + }, + "peerDependencies": { + "@nestjs/common": "^10.0.0", + "@nestjs/core": "^10.0.0", + "@nestjs/microservices": "^10.0.0", + "@nestjs/platform-express": "^10.0.0" + }, + "peerDependenciesMeta": { + "@nestjs/microservices": { + "optional": true + }, + "@nestjs/platform-express": { + "optional": true + } + } + }, + "node_modules/@noble/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/@swc/helpers": { + "version": "0.3.17", + "resolved": "https://registry.npmjs.org/@swc/helpers/-/helpers-0.3.17.tgz", + "integrity": "sha512-tb7Iu+oZ+zWJZ3HJqwx8oNwSDIU440hmVMDPhpACWQWnrZHK99Bxs70gT1L2dnr5Hg50ZRWEFkQCAnOVVV0z1Q==", + "license": "MIT", + "dependencies": { + "tslib": "^2.4.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/luxon": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/@types/luxon/-/luxon-3.4.2.tgz", + "integrity": "sha512-TifLZlFudklWlMBfhubvgqTXRzLDI5pCbGa4P8a3wPyUQSW+1xQ5eDsreP9DWHX3tjq1ke96uYG/nwundroWcA==", + "license": "MIT" + }, + "node_modules/@types/methods": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/@types/methods/-/methods-1.1.4.tgz", + "integrity": "sha512-ymXWVrDiCxTBE3+RIrrP533E70eA+9qu7zdWoHuOmGujkYtzf4HQF96b8nwHLqhuf4ykX61IGRIB38CC6/sImQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/mime": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.5.tgz", + "integrity": "sha512-/pyBZWSLD2n0dcHE3hq8s8ZvcETHtEuF+3E7XVt0Ig2nvsVQXdghHVcEkIWjy9A0wKfTn97a/PSDYohKIlnP/w==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/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/pdfkit": { + "version": "0.13.9", + "resolved": "https://registry.npmjs.org/@types/pdfkit/-/pdfkit-0.13.9.tgz", + "integrity": "sha512-RDG8Yb1zT7I01FfpwK7nMSA433XWpblMqSCtA5vJlSyavWZb303HUYPCel6JTiDDFqwGLvtAnYbH8N/e0Cb89g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "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": "2.0.16", + "resolved": "https://registry.npmjs.org/@types/supertest/-/supertest-2.0.16.tgz", + "integrity": "sha512-6c2ogktZ06tr2ENoZivgm7YnprnhYE4ZoXGMY+oA7IuAf17M8FWvujXZGmxLv8y0PTyts4x5A+erSwVUFA8XSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/superagent": "*" + } + }, + "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/archiver": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/archiver/-/archiver-5.3.2.tgz", + "integrity": "sha512-+25nxyyznAXF7Nef3y0EbBeqmGZgeN/BxHX29Rs39djAfaFalmQ89SE6CWyDCHzGL0yt/ycBtNOmGTW0FyGWNw==", + "license": "MIT", + "dependencies": { + "archiver-utils": "^2.1.0", + "async": "^3.2.4", + "buffer-crc32": "^0.2.1", + "readable-stream": "^3.6.0", + "readdir-glob": "^1.1.2", + "tar-stream": "^2.2.0", + "zip-stream": "^4.1.0" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/archiver-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/archiver-utils/-/archiver-utils-2.1.0.tgz", + "integrity": "sha512-bEL/yUb/fNNiNTuUz979Z0Yg5L+LzLxGJz8x79lYmR54fmTIb6ob/hNQgkQnIUDWIFjZVQwl9Xs356I6BAMHfw==", + "license": "MIT", + "dependencies": { + "glob": "^7.1.4", + "graceful-fs": "^4.2.0", + "lazystream": "^1.0.0", + "lodash.defaults": "^4.2.0", + "lodash.difference": "^4.5.0", + "lodash.flatten": "^4.4.0", + "lodash.isplainobject": "^4.0.6", + "lodash.union": "^4.6.0", + "normalize-path": "^3.0.0", + "readable-stream": "^2.0.0" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/archiver-utils/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==", + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/archiver-utils/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", + "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/archiver-utils/node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", + "license": "MIT" + }, + "node_modules/archiver-utils/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/archiver-utils/node_modules/readable-stream": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", + "license": "MIT", + "dependencies": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "node_modules/archiver-utils/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "license": "MIT" + }, + "node_modules/archiver-utils/node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "license": "MIT", + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "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-buffer-byte-length": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.2.tgz", + "integrity": "sha512-LHE+8BuR7RYGDKvnrmcuSq3tDcKv9OFEXQt/HpbZhY7V6h0zlUXutnAD82GiFx9rdieCMjkvtcsPqBwgUl1Iiw==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "is-array-buffer": "^3.0.5" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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/async": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.6.tgz", + "integrity": "sha512-htCUDlxyyCLMgaM3xXg0C0LW2xqfuQ6p05pCEIsXuyQ+a1koYKTuBMzRNwmybfLgvJDMd0r1LTn4+E0Ti6C2AA==", + "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==", + "dev": true, + "license": "MIT" + }, + "node_modules/available-typed-arrays": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", + "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", + "license": "MIT", + "dependencies": { + "possible-typed-array-names": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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==", + "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==", + "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/big-integer": { + "version": "1.6.52", + "resolved": "https://registry.npmjs.org/big-integer/-/big-integer-1.6.52.tgz", + "integrity": "sha512-QxD8cf2eVqJOOz63z6JIN9BzvVs/dlySa5HGSBH5xtR8dPteIRQnBxxKqkNTiT6jbDTF6jAfrd4oMcND9RGbQg==", + "license": "Unlicense", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/binary": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/binary/-/binary-0.3.0.tgz", + "integrity": "sha512-D4H1y5KYwpJgK8wk1Cue5LLPgmwHKYSChkbspQg5JtVuR5ulGckxfR62H3AE9UDkdMC8yyXlqYihuz3Aqg2XZg==", + "license": "MIT", + "dependencies": { + "buffers": "~0.1.1", + "chainsaw": "~0.1.0" + }, + "engines": { + "node": "*" + } + }, + "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==", + "license": "MIT", + "dependencies": { + "buffer": "^5.5.0", + "inherits": "^2.0.4", + "readable-stream": "^3.4.0" + } + }, + "node_modules/bluebird": { + "version": "3.4.7", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.4.7.tgz", + "integrity": "sha512-iD3898SR7sWVRHbiQv+sHUtHnMvC1o3nW5rAcqnq3uOn07DSAppZYUkIGslDz6gXC7HfunPe7YVBgoEJASPcHA==", + "license": "MIT" + }, + "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==", + "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/brotli": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/brotli/-/brotli-1.3.3.tgz", + "integrity": "sha512-oTKjJdShmDuGW94SyyaoQvAjf30dZaHnjJ8uAF+u2/vGJkJbJPJAT1gDiOJP5v1Zb6f9KEyW/1HpuaWIXtGHPg==", + "license": "MIT", + "dependencies": { + "base64-js": "^1.1.2" + } + }, + "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==", + "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-crc32": { + "version": "0.2.13", + "resolved": "https://registry.npmjs.org/buffer-crc32/-/buffer-crc32-0.2.13.tgz", + "integrity": "sha512-VO9Ht/+p3SN7SKWqcrgEzjGbRSJYTx+Q1pTQC0wrWqHx0vpJraQ6GtHx8tvcg1rlK1byhU5gccxgOgj7B0TDkQ==", + "license": "MIT", + "engines": { + "node": "*" + } + }, + "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/buffer-indexof-polyfill": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/buffer-indexof-polyfill/-/buffer-indexof-polyfill-1.0.2.tgz", + "integrity": "sha512-I7wzHwA3t1/lwXQh+A5PbNvJxgfo5r3xulgpYDB5zckTu/Z9oUK9biouBKQUjEqzaz3HnAT6TYoovmE+GqSf7A==", + "license": "MIT", + "engines": { + "node": ">=0.10" + } + }, + "node_modules/buffers": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/buffers/-/buffers-0.1.1.tgz", + "integrity": "sha512-9q/rDEGSb/Qsvv2qvzIzdluL5k7AaJOTrw23z9reQthrbF7is4CtlT0DXyO1oei2DCp4uojjzQ7igaSHp1kAEQ==", + "engines": { + "node": ">=0.2.0" + } + }, + "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==", + "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/chainsaw": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/chainsaw/-/chainsaw-0.1.0.tgz", + "integrity": "sha512-75kWfWt6MEKNC8xYXIdRpDehRYY/tNSgwKaJq+dbbDcxORuVrrQ+SEHoWsniVn9XPYfP4gmdWIeDk/4YNp1rNQ==", + "license": "MIT/X11", + "dependencies": { + "traverse": ">=0.3.0 <0.4" + }, + "engines": { + "node": "*" + } + }, + "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": "2.1.2", + "resolved": "https://registry.npmjs.org/clone/-/clone-2.1.2.tgz", + "integrity": "sha512-3Pe/CF1Nn94hyhIYpjtiLhdCoEoz0DqQ+988E9gmeEdQZlojxnOb74wctFyuwWQHzqyf9X7C7MG8juUpqBJT8w==", + "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==", + "dev": true, + "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/compress-commons": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/compress-commons/-/compress-commons-4.1.2.tgz", + "integrity": "sha512-D3uMHtGc/fcO1Gt1/L7i1e33VOvD4A9hfQLP+6ewd+BvG/gQ84Yh4oftEhAdjSMgBgwGL+jsppT7JYNpo6MHHg==", + "license": "MIT", + "dependencies": { + "buffer-crc32": "^0.2.13", + "crc32-stream": "^4.0.2", + "normalize-path": "^3.0.0", + "readable-stream": "^3.6.0" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "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==", + "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/crc-32": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/crc-32/-/crc-32-1.2.2.tgz", + "integrity": "sha512-ROmzCKrTnOwybPcJApAA6WBWij23HVfGVNKqqrZpuyZOHqK2CwHSvpGuyt/UNNvaIjEd8X5IFGp4Mh+Ie1IHJQ==", + "license": "Apache-2.0", + "bin": { + "crc32": "bin/crc32.njs" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/crc32-stream": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/crc32-stream/-/crc32-stream-4.0.3.tgz", + "integrity": "sha512-NT7w2JVU7DFroFdYkeq8cywxrgjPHWkdX1wjpRQXPX5Asews3tA+Ght6lddQO5Mkumffp3X7GEqku3epj2toIw==", + "license": "MIT", + "dependencies": { + "crc-32": "^1.2.0", + "readable-stream": "^3.4.0" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/create-jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/create-jest/-/create-jest-29.7.0.tgz", + "integrity": "sha512-Adz2bdH0Vq3F53KEMJOoftQFutWCukm6J24wbPWRO4k1kMY7gS7ds/uoJkNuV8wDCtWWnuwGcJwpWcih+zEW1Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "jest-config": "^29.7.0", + "jest-util": "^29.7.0", + "prompts": "^2.0.1" + }, + "bin": { + "create-jest": "bin/create-jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/create-require": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz", + "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/cron": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/cron/-/cron-3.2.1.tgz", + "integrity": "sha512-w2n5l49GMmmkBFEsH9FIDhjZ1n1QgTMOCMGuQtOXs5veNiosZmso6bQGuqOJSYAXXrG84WQFVneNk+Yt0Ua9iw==", + "license": "MIT", + "dependencies": { + "@types/luxon": "~3.4.0", + "luxon": "~3.5.0" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/crypto-js": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/crypto-js/-/crypto-js-4.2.0.tgz", + "integrity": "sha512-KALDyEYgpY+Rlob/iriUtjV6d5Eq+Y191A5g4UqLAi8CyGP9N1+FdVbkc1SxKc2r4YAYqG8JzO2KGL+AizD70Q==", + "license": "MIT" + }, + "node_modules/csv-stringify": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/csv-stringify/-/csv-stringify-6.6.0.tgz", + "integrity": "sha512-YW32lKOmIBgbxtu3g5SaiqWNwa/9ISQt2EcgOq0+RAIFufFp9is6tqNnKahqE5kuKvrnYAzs28r+s6pXJR8Vcw==", + "license": "MIT" + }, + "node_modules/dayjs": { + "version": "1.11.19", + "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.11.19.tgz", + "integrity": "sha512-t5EcLVS6QPBNqM2z8fakk/NKel+Xzshgt8FFKAn+qwlD1pzZWxh0nVCrvFK7ZDb6XucZeF9z8C7CBWTRIVApAw==", + "license": "MIT" + }, + "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-equal": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/deep-equal/-/deep-equal-2.2.3.tgz", + "integrity": "sha512-ZIwpnevOurS8bpT4192sqAowWM76JDKSHYzMLty3BZGSswgq6pBaH3DhCSW5xVAZICZyKdOBPjwww5wfgT/6PA==", + "license": "MIT", + "dependencies": { + "array-buffer-byte-length": "^1.0.0", + "call-bind": "^1.0.5", + "es-get-iterator": "^1.1.3", + "get-intrinsic": "^1.2.2", + "is-arguments": "^1.1.1", + "is-array-buffer": "^3.0.2", + "is-date-object": "^1.0.5", + "is-regex": "^1.1.4", + "is-shared-array-buffer": "^1.0.2", + "isarray": "^2.0.5", + "object-is": "^1.1.5", + "object-keys": "^1.1.1", + "object.assign": "^4.1.4", + "regexp.prototype.flags": "^1.5.1", + "side-channel": "^1.0.4", + "which-boxed-primitive": "^1.0.2", + "which-collection": "^1.0.1", + "which-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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/defaults/node_modules/clone": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz", + "integrity": "sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8" + } + }, + "node_modules/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==", + "license": "MIT", + "dependencies": { + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "gopd": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/define-properties": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", + "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", + "license": "MIT", + "dependencies": { + "define-data-property": "^1.0.1", + "has-property-descriptors": "^1.0.0", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "dev": true, + "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/dfa": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/dfa/-/dfa-1.2.0.tgz", + "integrity": "sha512-ED3jP8saaweFTjeGX8HQPjeC1YYyZs98jGNZx6IiBvxW7JG5v492kamAQB3m2wop07CvU/RQmzcKr6bgcC5D/Q==", + "license": "MIT" + }, + "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/duplexer2": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/duplexer2/-/duplexer2-0.1.4.tgz", + "integrity": "sha512-asLFVfWWtJ90ZyOUHMqk7/S2w2guQKxUI2itj3d92ADHhxUSbCMGi1f1cBcJ7xM1To+pE/Khbwo1yuNbMEPKeA==", + "license": "BSD-3-Clause", + "dependencies": { + "readable-stream": "^2.0.2" + } + }, + "node_modules/duplexer2/node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", + "license": "MIT" + }, + "node_modules/duplexer2/node_modules/readable-stream": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", + "license": "MIT", + "dependencies": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "node_modules/duplexer2/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "license": "MIT" + }, + "node_modules/duplexer2/node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "license": "MIT", + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "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/end-of-stream": { + "version": "1.4.5", + "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.5.tgz", + "integrity": "sha512-ooEGc6HP26xXq/N+GCGOT0JKCLDGrq2bQUZrQ7gyrJiZANJ/8YDTxTpQBXGMn+WbIQXNVpyWymm7KYVICQnyOg==", + "license": "MIT", + "dependencies": { + "once": "^1.4.0" + } + }, + "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-get-iterator": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/es-get-iterator/-/es-get-iterator-1.1.3.tgz", + "integrity": "sha512-sPZmqHBe6JIiTfN5q2pEi//TwxmAFHwj/XEuYjTuse78i8KxaqMTTzxPoFKuzRpDpTJ+0NAbpfenkmH2rePtuw==", + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.1.3", + "has-symbols": "^1.0.3", + "is-arguments": "^1.1.1", + "is-map": "^2.0.2", + "is-set": "^2.0.2", + "is-string": "^1.0.7", + "isarray": "^2.0.5", + "stop-iteration-iterator": "^1.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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==", + "dev": true, + "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/exceljs": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/exceljs/-/exceljs-4.4.0.tgz", + "integrity": "sha512-XctvKaEMaj1Ii9oDOqbW/6e1gXknSY4g/aLCDicOXqBE4M0nRWkUu0PTp++UPNzoFY12BNHMfs/VadKIS6llvg==", + "license": "MIT", + "dependencies": { + "archiver": "^5.0.0", + "dayjs": "^1.8.34", + "fast-csv": "^4.3.1", + "jszip": "^3.10.1", + "readable-stream": "^3.6.0", + "saxes": "^5.0.1", + "tmp": "^0.2.0", + "unzipper": "^0.10.11", + "uuid": "^8.3.0" + }, + "engines": { + "node": ">=8.3.0" + } + }, + "node_modules/exceljs/node_modules/uuid": { + "version": "8.3.2", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz", + "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==", + "license": "MIT", + "bin": { + "uuid": "dist/bin/uuid" + } + }, + "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/external-editor/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/fast-csv": { + "version": "4.3.6", + "resolved": "https://registry.npmjs.org/fast-csv/-/fast-csv-4.3.6.tgz", + "integrity": "sha512-2RNSpuwwsJGP0frGsOmTb9oUF+VkFSM4SyLTDgwf2ciHWTarN0lQTC+F2f/t5J9QjW+c65VFIAAu85GsvMIusw==", + "license": "MIT", + "dependencies": { + "@fast-csv/format": "4.3.5", + "@fast-csv/parse": "4.3.6" + }, + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true, + "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/fontkit": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/fontkit/-/fontkit-1.9.0.tgz", + "integrity": "sha512-HkW/8Lrk8jl18kzQHvAw9aTHe1cqsyx5sDnxncx652+CIfhawokEPkeM3BoIC+z/Xv7a0yMr0f3pRRwhGH455g==", + "license": "MIT", + "dependencies": { + "@swc/helpers": "^0.3.13", + "brotli": "^1.3.2", + "clone": "^2.1.2", + "deep-equal": "^2.0.5", + "dfa": "^1.2.0", + "restructure": "^2.0.1", + "tiny-inflate": "^1.0.3", + "unicode-properties": "^1.3.1", + "unicode-trie": "^2.0.0" + } + }, + "node_modules/for-each": { + "version": "0.3.5", + "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.5.tgz", + "integrity": "sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg==", + "license": "MIT", + "dependencies": { + "is-callable": "^1.2.7" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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==", + "dev": true, + "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-constants": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz", + "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==", + "license": "MIT" + }, + "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==", + "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/fstream": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/fstream/-/fstream-1.0.12.tgz", + "integrity": "sha512-WvJ193OHa0GHPEL+AycEJgxvBEwyfRkN1vhjca23OaPVMCaLCXTd5qAu82AjTcgP1UJmytkOKb63Ypde7raDIg==", + "deprecated": "This package is no longer supported.", + "license": "ISC", + "dependencies": { + "graceful-fs": "^4.1.2", + "inherits": "~2.0.0", + "mkdirp": ">=0.5 0", + "rimraf": "2" + }, + "engines": { + "node": ">=0.6" + } + }, + "node_modules/fstream/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==", + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/fstream/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", + "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/fstream/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/fstream/node_modules/rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "deprecated": "Rimraf versions prior to v4 are no longer supported", + "license": "ISC", + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "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/functions-have-names": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz", + "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", + "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==", + "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-bigints": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.1.0.tgz", + "integrity": "sha512-R3pbpkcIqv2Pm3dUwgjclDRVmWpTJW2DcMzcIhEXEx1oh/CEMObMm3KLmRJOdvhM7o4uQBnwr8pzRK2sJWIqfg==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "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==", + "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/immediate": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/immediate/-/immediate-3.0.6.tgz", + "integrity": "sha512-XXOFtyqDjNDAQxVfYxuF7g9Il/IbWmmlQg2MYKOH8ExIT1qg6xc4zyS3HaEEATgs1btfzxq15ciUiY7gjSXRGQ==", + "license": "MIT" + }, + "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.", + "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/internal-slot": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.1.0.tgz", + "integrity": "sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "hasown": "^2.0.2", + "side-channel": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "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-arguments": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.2.0.tgz", + "integrity": "sha512-7bVbi0huj/wrIAOzb8U1aszg9kdi3KN/CyU19CTI7tAoZYEZoL9yCDXpbXN+uPsuWnP02cyug1gleqq+TU+YCA==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-array-buffer": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.5.tgz", + "integrity": "sha512-DDfANUiiG2wC1qawP66qlTugJeL5HyzMpfr8lLK+jMQirGzNod0B12cFB/9q838Ru27sBwfw78/rdoU7RERz6A==", + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "call-bound": "^1.0.3", + "get-intrinsic": "^1.2.6" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", + "dev": true, + "license": "MIT" + }, + "node_modules/is-bigint": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.1.0.tgz", + "integrity": "sha512-n4ZT37wG78iz03xPRKJrHTdZbe3IicyucEtdRsV5yglwc3GyUfbAfpSeD0FJ41NbUNSt5wbhqfp1fS+BgnvDFQ==", + "license": "MIT", + "dependencies": { + "has-bigints": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dev": true, + "license": "MIT", + "dependencies": { + "binary-extensions": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-boolean-object": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.2.2.tgz", + "integrity": "sha512-wa56o2/ElJMYqjCjGkXri7it5FbebW5usLw/nPmCMs5DeZ7eziSYZhSmPRn0txqeW4LnAmQQU7FgqLpsEFKM4A==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-callable": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", + "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-core-module": { + "version": "2.16.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", + "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-date-object": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.1.0.tgz", + "integrity": "sha512-PwwhEakHVKTdRNVOw+/Gyh0+MzlCl4R6qKvkhuvLtPMggI1WAHt9sOwZxQLSGpUaDnrdyDsomoRgNnCfKNSXXg==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-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-map": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.3.tgz", + "integrity": "sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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-number-object": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.1.1.tgz", + "integrity": "sha512-lZhclumE1G6VYD8VHe35wFaIif+CTy5SJIi5+3y4psDgWu4wPDoBhF8NxUOinEc7pHgiTsT6MaBb92rKhhD+Xw==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-path-inside": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", + "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-regex": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.2.1.tgz", + "integrity": "sha512-MjYsKHO5O7mCsmRGxWcLWheFqN9DJ/2TmngvjKXihe6efViPqc274+Fx/4fYj/r03+ESvBdTXK0V6tA3rgez1g==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "gopd": "^1.2.0", + "has-tostringtag": "^1.0.2", + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-set": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.3.tgz", + "integrity": "sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-shared-array-buffer": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.4.tgz", + "integrity": "sha512-ISWac8drv4ZGfwKl5slpHG9OwPNty4jOWPRIhBpxOoD+hqITiwuipOQ2bNthAzwA3B4fIjO4Nln74N0S9byq8A==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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-string": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.1.1.tgz", + "integrity": "sha512-BtEeSsoaQjlSPBemMQIrY1MY0uM6vnS1g5fmufYOtnxLGUZM2178PKbhsk7Ffv58IX+ZtcvoGwccYsh0PglkAA==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-symbol": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.1.1.tgz", + "integrity": "sha512-9gGx6GTtCQM73BgmHQXfDmLtfjjTUDSyoxTCbp5WtoixAhfgsDirWIcVQ/IHpvI5Vgd5i/J5F7B9cN/WlVbC/w==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "has-symbols": "^1.1.0", + "safe-regex-test": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-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/is-weakmap": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.2.tgz", + "integrity": "sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-weakset": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.4.tgz", + "integrity": "sha512-mfcwb6IzQyOKTs84CQMrOwW4gQcaTOAWJ0zzJCl2WSPDrWk/OzDaImWFH3djXhb24g4eudZfLRozAvPGw4d9hQ==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "get-intrinsic": "^1.2.6" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/isarray": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", + "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", + "license": "MIT" + }, + "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/jszip": { + "version": "3.10.1", + "resolved": "https://registry.npmjs.org/jszip/-/jszip-3.10.1.tgz", + "integrity": "sha512-xXDvecyTpGLrqFrvkrUSoxxfJI5AH7U8zxxtVclpsUtMCq4JQ290LY8AW5c7Ggnr/Y/oK+bQMbqK2qmtk3pN4g==", + "license": "(MIT OR GPL-3.0-or-later)", + "dependencies": { + "lie": "~3.3.0", + "pako": "~1.0.2", + "readable-stream": "~2.3.6", + "setimmediate": "^1.0.5" + } + }, + "node_modules/jszip/node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", + "license": "MIT" + }, + "node_modules/jszip/node_modules/readable-stream": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", + "license": "MIT", + "dependencies": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "node_modules/jszip/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "license": "MIT" + }, + "node_modules/jszip/node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "license": "MIT", + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "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/lazystream": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/lazystream/-/lazystream-1.0.1.tgz", + "integrity": "sha512-b94GiNHQNy6JNTrt5w6zNyffMrNkXZb3KTkCZJb2V1xaEGCk093vkZ2jk3tpaeP33/OiXC+WvK9AxUebnf5nbw==", + "license": "MIT", + "dependencies": { + "readable-stream": "^2.0.5" + }, + "engines": { + "node": ">= 0.6.3" + } + }, + "node_modules/lazystream/node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", + "license": "MIT" + }, + "node_modules/lazystream/node_modules/readable-stream": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", + "license": "MIT", + "dependencies": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "node_modules/lazystream/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "license": "MIT" + }, + "node_modules/lazystream/node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "license": "MIT", + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "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/lie": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/lie/-/lie-3.3.0.tgz", + "integrity": "sha512-UaiMJzeWRlEujzAuw5LokY1L5ecNQYZKfmyZ9L7wDHb/p5etKaxXhohBcrw0EYby+G/NA52vRSN4N39dxHAIwQ==", + "license": "MIT", + "dependencies": { + "immediate": "~3.0.5" + } + }, + "node_modules/linebreak": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/linebreak/-/linebreak-1.1.0.tgz", + "integrity": "sha512-MHp03UImeVhB7XZtjd0E4n6+3xr5Dq/9xI/5FptGk5FrbDR3zagPa2DS6U8ks/3HjbKWG9Q1M2ufOzxV2qLYSQ==", + "license": "MIT", + "dependencies": { + "base64-js": "0.0.8", + "unicode-trie": "^2.0.0" + } + }, + "node_modules/linebreak/node_modules/base64-js": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-0.0.8.tgz", + "integrity": "sha512-3XSA2cR/h/73EzlXXdU6YNycmYI7+kicTxks4eJg2g39biHR84slg2+des+p7iHYhbRg/udIS4TD53WabcOUkw==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "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/listenercount": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/listenercount/-/listenercount-1.0.1.tgz", + "integrity": "sha512-3mk/Zag0+IJxeDrxSgaDPy4zZ3w05PRZeJNnlWhzFz5OkX49J4krc+A8X2d2M69vGMBEX0uyl8M+W+8gH+kBqQ==", + "license": "ISC" + }, + "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.difference": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.difference/-/lodash.difference-4.5.0.tgz", + "integrity": "sha512-dS2j+W26TQ7taQBGN8Lbbq04ssV3emRw4NY58WErlTO29pIqS0HmoT5aJ9+TUQ1N3G+JOZSji4eugsWwGp9yPA==", + "license": "MIT" + }, + "node_modules/lodash.escaperegexp": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/lodash.escaperegexp/-/lodash.escaperegexp-4.1.2.tgz", + "integrity": "sha512-TM9YBvyC84ZxE3rgfefxUWiQKLilstD6k7PTGt6wfbtXF8ixIJLOL3VYyV/z+ZiPLsVxAsKAFVwWlWeb2Y8Yyw==", + "license": "MIT" + }, + "node_modules/lodash.flatten": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/lodash.flatten/-/lodash.flatten-4.4.0.tgz", + "integrity": "sha512-C5N2Z3DgnnKr0LOpv/hKCgKdb7ZZwafIrsesve6lmzvZIRZRGaZ/l6Q8+2W7NaT+ZwO3fFlSCzCzrDCFdJfZ4g==", + "license": "MIT" + }, + "node_modules/lodash.groupby": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/lodash.groupby/-/lodash.groupby-4.6.0.tgz", + "integrity": "sha512-5dcWxm23+VAoz+awKmBaiBvzox8+RqMgFhi7UvX9DHZr2HdxHXM/Wrf8cfKpsW37RNrvtPn6hSwNqurSILbmJw==", + "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.isequal": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz", + "integrity": "sha512-pDo3lu8Jhfjqls6GkMgpahsF9kCyayhgykjyLMNFTKWrpVdAQtYyB4muAMWozBB4ig/dtWAmsMxLEI8wuz+DYQ==", + "deprecated": "This package is deprecated. Use require('node:util').isDeepStrictEqual instead.", + "license": "MIT" + }, + "node_modules/lodash.isfunction": { + "version": "3.0.9", + "resolved": "https://registry.npmjs.org/lodash.isfunction/-/lodash.isfunction-3.0.9.tgz", + "integrity": "sha512-AirXNj15uRIMMPihnkInB4i3NHeb4iBtNg9WRWuK2o31S+ePwwNmDPaTL3o7dTJ+VXNZim7rFs4rxN4YU1oUJw==", + "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.isnil": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/lodash.isnil/-/lodash.isnil-4.0.0.tgz", + "integrity": "sha512-up2Mzq3545mwVnMhTDMdfoG1OurpA/s5t88JmQX809eH3C8491iu2sfKhTfhQtKY78oPNhiaHJUpT/dUDAAtng==", + "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.isundefined": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/lodash.isundefined/-/lodash.isundefined-3.0.1.tgz", + "integrity": "sha512-MXB1is3s899/cD8jheYYE2V9qTHwKvt+npCwpD+1Sxm3Q3cECXCiYHjeHWXNwr6Q0SOBPrYUDxendrO6goVTEA==", + "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/lodash.union": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/lodash.union/-/lodash.union-4.6.0.tgz", + "integrity": "sha512-c4pB2CdGrGdjMKYLA+XiRDO7Y0PRQbm/Gzg8qMj+QH+pFVAoTp5sBpO0odL3FjoPCGjK96p6qsP+yQoiLoOBcw==", + "license": "MIT" + }, + "node_modules/lodash.uniq": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.uniq/-/lodash.uniq-4.5.0.tgz", + "integrity": "sha512-xfBaXQd9ryd9dlSDvnvI0lvxfLJlYAZzXomUYzLKtUeOQvOP5piqAWuGtrhWeqaXK9hhoM/iyJc5AV+XfsX3HQ==", + "license": "MIT" + }, + "node_modules/log-symbols": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz", + "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.1.0", + "is-unicode-supported": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "license": "ISC", + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/luxon": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/luxon/-/luxon-3.5.0.tgz", + "integrity": "sha512-rh+Zjr6DNfUYR3bPwJEnuwDdqMbxZW7LOQfUN4B54+Cl+0o5zaU9RJ6bcidfDtC1cWCZXQ+nvX8bf6bAji37QQ==", + "license": "MIT", + "engines": { + "node": ">=12" + } + }, + "node_modules/magic-string": { + "version": "0.30.8", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.8.tgz", + "integrity": "sha512-ISQTe55T2ao7XtlAStud6qwYPZjE4GK1S/BeVPus4jrq6JuOnQ00YKQC581RWhR122W7msZV263KzVeLoqidyQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.4.15" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/make-dir": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-4.0.0.tgz", + "integrity": "sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==", + "dev": true, + "license": "MIT", + "dependencies": { + "semver": "^7.5.3" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dev": true, + "license": "ISC" + }, + "node_modules/makeerror": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/makeerror/-/makeerror-1.0.12.tgz", + "integrity": "sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "tmpl": "1.0.5" + } + }, + "node_modules/math-intrinsics": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", + "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/memfs": { + "version": "3.5.3", + "resolved": "https://registry.npmjs.org/memfs/-/memfs-3.5.3.tgz", + "integrity": "sha512-UERzLsxzllchadvbPs5aolHh65ISpKpM+ccLbOJ8/vvpBKmAWf+la7dXFy7Mr0ySHbdHrFv5kGFCUHHe6GFEmw==", + "dev": true, + "license": "Unlicense", + "dependencies": { + "fs-monkey": "^1.0.4" + }, + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/merge-descriptors": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz", + "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/merge-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", + "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", + "dev": true, + "license": "MIT" + }, + "node_modules/merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 8" + } + }, + "node_modules/methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/micromatch": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", + "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", + "dev": true, + "license": "MIT", + "dependencies": { + "braces": "^3.0.3", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/micromatch/node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "license": "MIT", + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "license": "MIT", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/minimatch": { + "version": "9.0.3", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.3.tgz", + "integrity": "sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/minimist": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/minipass": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz", + "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=16 || 14 >=14.17" + } + }, + "node_modules/mkdirp": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz", + "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==", + "license": "MIT", + "dependencies": { + "minimist": "^1.2.6" + }, + "bin": { + "mkdirp": "bin/cmd.js" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "license": "MIT" + }, + "node_modules/multer": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/multer/-/multer-2.0.2.tgz", + "integrity": "sha512-u7f2xaZ/UG8oLXHvtF/oWTRvT44p9ecwBBqTwgJVq0+4BW1g8OW01TyMEGWBHbyMOYVHXslaut7qEQ1meATXgw==", + "license": "MIT", + "dependencies": { + "append-field": "^1.0.0", + "busboy": "^1.6.0", + "concat-stream": "^2.0.0", + "mkdirp": "^0.5.6", + "object-assign": "^4.1.1", + "type-is": "^1.6.18", + "xtend": "^4.0.2" + }, + "engines": { + "node": ">= 10.16.0" + } + }, + "node_modules/mute-stream": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz", + "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", + "dev": true, + "license": "ISC" + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true, + "license": "MIT" + }, + "node_modules/negotiator": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", + "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-abort-controller": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/node-abort-controller/-/node-abort-controller-3.1.1.tgz", + "integrity": "sha512-AGK2yQKIjRuqnc6VkX2Xj5d+QW8xZ87pa1UK6yA6ouUyuxfHuMP6umE5QK7UmTeOAymo+Zx1Fxiuw9rVx8taHQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-emoji": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/node-emoji/-/node-emoji-1.11.0.tgz", + "integrity": "sha512-wo2DpQkQp7Sjm2A0cq+sN7EHKO6Sl0ctXeBdFZrL9T9+UywORbufTcTZxom8YqpLQt/FqNMUkOpkZrJVYSKD3A==", + "dev": true, + "license": "MIT", + "dependencies": { + "lodash": "^4.17.21" + } + }, + "node_modules/node-fetch": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz", + "integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==", + "license": "MIT", + "dependencies": { + "whatwg-url": "^5.0.0" + }, + "engines": { + "node": "4.x || >=6.0.0" + }, + "peerDependencies": { + "encoding": "^0.1.0" + }, + "peerDependenciesMeta": { + "encoding": { + "optional": true + } + } + }, + "node_modules/node-int64": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", + "integrity": "sha512-O5lz91xSOeoXP6DulyHfllpq+Eg00MWitZIbtPfoSEvqIHdl5gfcY6hYzDWnj0qD5tz52PI08u9qUvSVeUBeHw==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-releases": { + "version": "2.0.27", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.27.tgz", + "integrity": "sha512-nmh3lCkYZ3grZvqcCH+fjmQ7X+H0OeZgP40OierEaAptX4XofMh5kwNbWh7lBduUzCcV/8kZ+NDLCwm2iorIlA==", + "dev": true, + "license": "MIT" + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "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/object-is": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/object-is/-/object-is-1.1.6.tgz", + "integrity": "sha512-F8cZ+KfGlSGi09lJT7/Nd6KJZ9ygtvYC0/UYYLI9nmQKLMnydpB9yvbv9K1uSkEu7FU9vYPmVwLg328tX+ot3Q==", + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object-keys": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", + "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object.assign": { + "version": "4.1.7", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.7.tgz", + "integrity": "sha512-nK28WOo+QIjBkDduTINE4JkF/UJJKyf2EJxvJKfblDpyg0Q+pkOHNTL0Qwy6NP6FhE/EnzV73BxxqcJaXY9anw==", + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "call-bound": "^1.0.3", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0", + "has-symbols": "^1.1.0", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/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==", + "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/pako": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz", + "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==", + "license": "(MIT AND Zlib)" + }, + "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==", + "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/pdfkit": { + "version": "0.14.0", + "resolved": "https://registry.npmjs.org/pdfkit/-/pdfkit-0.14.0.tgz", + "integrity": "sha512-Hnor8/78jhHm6ONrxWhrqOwAVALlBnFyWOF8sstBZMiqHZgZ5A6RU+Q3yahhw82plxpT7LOfH3b3qcOX6rzMQg==", + "license": "MIT", + "dependencies": { + "crypto-js": "^4.2.0", + "fontkit": "^1.8.1", + "linebreak": "^1.0.2", + "png-js": "^1.0.0" + } + }, + "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/png-js": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/png-js/-/png-js-1.0.0.tgz", + "integrity": "sha512-k+YsbhpA9e+EFfKjTCH3VW6aoKlyNYI6NYdTfDL4CIvFnvsuO84ttonmZE7rc+v23SLTH8XX+5w/Ak9v0xGY4g==" + }, + "node_modules/possible-typed-array-names": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.1.0.tgz", + "integrity": "sha512-/+5VFTchJDoVj3bhoqi6UeymcD00DAwb1nJwamzPvHEszJ4FpF6SNNbUbOS8yI56qHzdV8eK0qEfOSiodkTdxg==", + "license": "MIT", + "engines": { + "node": ">= 0.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/process-nextick-args": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", + "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", + "license": "MIT" + }, + "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/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/readdir-glob": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/readdir-glob/-/readdir-glob-1.1.3.tgz", + "integrity": "sha512-v05I2k7xN8zXvPD9N+z/uhXPaj0sUFCe2rcWZIpBsqxfP7xXFQ0tipAd/wjj1YxWyWtUS5IDJpOG82JKt2EAVA==", + "license": "Apache-2.0", + "dependencies": { + "minimatch": "^5.1.0" + } + }, + "node_modules/readdir-glob/node_modules/minimatch": { + "version": "5.1.6", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.6.tgz", + "integrity": "sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==", + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "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/regexp.prototype.flags": { + "version": "1.5.4", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.4.tgz", + "integrity": "sha512-dYqgNSZbDwkaJ2ceRd9ojCGjBq+mOm9LmtXnAnEGyHhN/5R7iDW2TRw3h+o/jCFxus3P2LfWIIiwowAjANm7IA==", + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "define-properties": "^1.2.1", + "es-errors": "^1.3.0", + "get-proto": "^1.0.1", + "gopd": "^1.2.0", + "set-function-name": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/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/restructure": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/restructure/-/restructure-2.0.1.tgz", + "integrity": "sha512-e0dOpjm5DseomnXx2M5lpdZ5zoHqF1+bqdMJUohoYVVQa7cBdnk7fdmeI6byNWP/kiME72EeTiSypTCVnpLiDg==", + "license": "MIT" + }, + "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/safe-regex-test": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.1.0.tgz", + "integrity": "sha512-x/+Cz4YrimQxQccJf5mKEbIa1NzeCRNI5Ecl/ekmlYaampdNLPalVyIcCZNNH3MvmqBugV5TMYZXv0ljslUlaw==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "is-regex": "^1.2.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/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/saxes": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/saxes/-/saxes-5.0.1.tgz", + "integrity": "sha512-5LBh1Tls8c9xgGjw3QrMwETmTMVk0oFgvrFSvWx62llR2hcEInrKNZ2GZCCuuy2lvWrdl5jhbpeqc5hRYKFOcw==", + "license": "ISC", + "dependencies": { + "xmlchars": "^2.2.0" + }, + "engines": { + "node": ">=10" + } + }, + "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==", + "license": "MIT", + "dependencies": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "gopd": "^1.0.1", + "has-property-descriptors": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/set-function-name": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", + "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", + "license": "MIT", + "dependencies": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "functions-have-names": "^1.2.3", + "has-property-descriptors": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/setimmediate": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.5.tgz", + "integrity": "sha512-MATJdZp8sLqDl/68LfQmbP8zKPLQNV6BIZoIgrscFDQ+RsvK/BxeDQOgyxKKoh0y/8h3BqVFnCqQ/gd+reiIXA==", + "license": "MIT" + }, + "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/stop-iteration-iterator": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/stop-iteration-iterator/-/stop-iteration-iterator-1.1.0.tgz", + "integrity": "sha512-eLoXW/DHyl62zxY4SCaIgnRhuMr6ri4juEYARS8E6sCEqzKpOiE521Ucofdx+KnDZl5xmvGYaaKCk5FEOxJCoQ==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "internal-slot": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "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/tar-stream": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", + "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", + "license": "MIT", + "dependencies": { + "bl": "^4.0.3", + "end-of-stream": "^1.4.1", + "fs-constants": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^3.1.1" + }, + "engines": { + "node": ">=6" + } + }, + "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/tiny-inflate": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/tiny-inflate/-/tiny-inflate-1.0.3.tgz", + "integrity": "sha512-pkY1fj1cKHb2seWDy0B16HeWyczlJA9/WW3u3c4z/NiWDsO3DOU5D7nhTLE9CF0yXv/QZFY7sEJmj24dK+Rrqw==", + "license": "MIT" + }, + "node_modules/tmp": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.2.5.tgz", + "integrity": "sha512-voyz6MApa1rQGUxT3E+BK7/ROe8itEx7vD8/HEvt4xwXucvQ5G5oeEiHkmHZJuBO21RpOf+YYm9MOivj709jow==", + "license": "MIT", + "engines": { + "node": ">=14.14" + } + }, + "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/traverse": { + "version": "0.3.9", + "resolved": "https://registry.npmjs.org/traverse/-/traverse-0.3.9.tgz", + "integrity": "sha512-iawgk0hLP3SxGKDfnDJf8wTz4p2qImnyihM5Hh/sGvQ3K37dPi/w8sRhdNIxYA1TwFwc5mDhIJq+O0RsvXBKdQ==", + "license": "MIT/X11", + "engines": { + "node": "*" + } + }, + "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/unicode-properties": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/unicode-properties/-/unicode-properties-1.4.1.tgz", + "integrity": "sha512-CLjCCLQ6UuMxWnbIylkisbRj31qxHPAurvena/0iwSVbQ2G1VY5/HjV0IRabOEbDHlzZlRdCrD4NhB0JtU40Pg==", + "license": "MIT", + "dependencies": { + "base64-js": "^1.3.0", + "unicode-trie": "^2.0.0" + } + }, + "node_modules/unicode-trie": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/unicode-trie/-/unicode-trie-2.0.0.tgz", + "integrity": "sha512-x7bc76x0bm4prf1VLg79uhAzKw8DVboClSN5VxJuQ+LKDOVEW9CdH+VY7SP+vX7xCYQqzzgQpFqz15zeLvAtZQ==", + "license": "MIT", + "dependencies": { + "pako": "^0.2.5", + "tiny-inflate": "^1.0.0" + } + }, + "node_modules/unicode-trie/node_modules/pako": { + "version": "0.2.9", + "resolved": "https://registry.npmjs.org/pako/-/pako-0.2.9.tgz", + "integrity": "sha512-NUcwaKxUxWrZLpDG+z/xZaCgQITkA/Dv4V/T6bw7VON6l1Xz/VnrBqrYjZQ12TamKHzITTfOEIYUj48y2KXImA==", + "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/unzipper": { + "version": "0.10.14", + "resolved": "https://registry.npmjs.org/unzipper/-/unzipper-0.10.14.tgz", + "integrity": "sha512-ti4wZj+0bQTiX2KmKWuwj7lhV+2n//uXEotUmGuQqrbVZSEGFMbI68+c6JCQ8aAmUWYvtHEz2A8K6wXvueR/6g==", + "license": "MIT", + "dependencies": { + "big-integer": "^1.6.17", + "binary": "~0.3.0", + "bluebird": "~3.4.1", + "buffer-indexof-polyfill": "~1.0.0", + "duplexer2": "~0.1.4", + "fstream": "^1.0.12", + "graceful-fs": "^4.2.2", + "listenercount": "~1.0.1", + "readable-stream": "~2.3.6", + "setimmediate": "~1.0.4" + } + }, + "node_modules/unzipper/node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", + "license": "MIT" + }, + "node_modules/unzipper/node_modules/readable-stream": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", + "license": "MIT", + "dependencies": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "node_modules/unzipper/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "license": "MIT" + }, + "node_modules/unzipper/node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "license": "MIT", + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "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/which-boxed-primitive": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.1.1.tgz", + "integrity": "sha512-TbX3mj8n0odCBFVlY8AxkqcHASw3L60jIuF8jFP78az3C2YhmGvqbHBpAjTRH2/xqYunrJ9g1jSyjCjpoWzIAA==", + "license": "MIT", + "dependencies": { + "is-bigint": "^1.1.0", + "is-boolean-object": "^1.2.1", + "is-number-object": "^1.1.1", + "is-string": "^1.1.1", + "is-symbol": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/which-collection": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.2.tgz", + "integrity": "sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==", + "license": "MIT", + "dependencies": { + "is-map": "^2.0.3", + "is-set": "^2.0.3", + "is-weakmap": "^2.0.2", + "is-weakset": "^2.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/which-typed-array": { + "version": "1.1.19", + "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.19.tgz", + "integrity": "sha512-rEvr90Bck4WZt9HHFC4DJMsjvu7x+r6bImz0/BrbWb7A2djJ8hnZMrWnHo9F8ssv0OMErasDhftrfROTyqSDrw==", + "license": "MIT", + "dependencies": { + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.8", + "call-bound": "^1.0.4", + "for-each": "^0.3.5", + "get-proto": "^1.0.1", + "gopd": "^1.2.0", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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==", + "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/xmlchars": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", + "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==", + "license": "MIT" + }, + "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" + } + }, + "node_modules/zip-stream": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/zip-stream/-/zip-stream-4.1.1.tgz", + "integrity": "sha512-9qv4rlDiopXg4E69k+vMHjNN63YFMe9sZMrdlvKnCjlCRWeCBswPPMPUfx+ipsAWq1LXHe70RcbaHdJJpS6hyQ==", + "license": "MIT", + "dependencies": { + "archiver-utils": "^3.0.4", + "compress-commons": "^4.1.2", + "readable-stream": "^3.6.0" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/zip-stream/node_modules/archiver-utils": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/archiver-utils/-/archiver-utils-3.0.4.tgz", + "integrity": "sha512-KVgf4XQVrTjhyWmx6cte4RxonPLR9onExufI1jhvw/MQ4BB6IsZD5gT8Lq+u/+pRkWna/6JoHpiQioaqFP5Rzw==", + "license": "MIT", + "dependencies": { + "glob": "^7.2.3", + "graceful-fs": "^4.2.0", + "lazystream": "^1.0.0", + "lodash.defaults": "^4.2.0", + "lodash.difference": "^4.5.0", + "lodash.flatten": "^4.4.0", + "lodash.isplainobject": "^4.0.6", + "lodash.union": "^4.6.0", + "normalize-path": "^3.0.0", + "readable-stream": "^3.6.0" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/zip-stream/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==", + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/zip-stream/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", + "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/zip-stream/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + } + } +} diff --git a/backend/services/reporting-service/package.json b/backend/services/reporting-service/package.json index e69de29b..e242f4dc 100644 --- a/backend/services/reporting-service/package.json +++ b/backend/services/reporting-service/package.json @@ -0,0 +1,105 @@ +{ + "name": "reporting-service", + "version": "1.0.0", + "description": "Reporting & Analytics Service for RWA Durian Platform", + "author": "", + "private": true, + "license": "UNLICENSED", + "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:seed": "ts-node prisma/seed.ts", + "prisma:studio": "prisma studio" + }, + "dependencies": { + "@nestjs/common": "^10.0.0", + "@nestjs/config": "^3.1.1", + "@nestjs/core": "^10.0.0", + "@nestjs/jwt": "^10.2.0", + "@nestjs/microservices": "^10.3.0", + "@nestjs/passport": "^10.0.3", + "@nestjs/platform-express": "^10.0.0", + "@nestjs/schedule": "^4.0.0", + "@nestjs/swagger": "^7.1.17", + "@prisma/client": "^5.7.0", + "class-transformer": "^0.5.1", + "class-validator": "^0.14.0", + "csv-stringify": "^6.4.4", + "exceljs": "^4.4.0", + "ioredis": "^5.3.2", + "kafkajs": "^2.2.4", + "passport": "^0.7.0", + "passport-jwt": "^4.0.1", + "pdfkit": "^0.14.0", + "reflect-metadata": "^0.1.13", + "rxjs": "^7.8.1", + "uuid": "^9.0.1" + }, + "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/pdfkit": "^0.13.3", + "@types/supertest": "^2.0.12", + "@types/uuid": "^9.0.7", + "@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|integration\\.spec)\\.ts$", + "transform": { + "^.+\\.(t|j)s$": "ts-jest" + }, + "collectCoverageFrom": [ + "src/**/*.(t|j)s" + ], + "coverageDirectory": "./coverage", + "testEnvironment": "node", + "moduleNameMapper": { + "^@/(.*)$": "/src/$1", + "^@domain/(.*)$": "/src/domain/$1", + "^@application/(.*)$": "/src/application/$1", + "^@infrastructure/(.*)$": "/src/infrastructure/$1", + "^@api/(.*)$": "/src/api/$1", + "^@shared/(.*)$": "/src/shared/$1", + "^@config/(.*)$": "/src/config/$1" + } + } +} diff --git a/backend/services/reporting-service/prisma/schema.prisma b/backend/services/reporting-service/prisma/schema.prisma new file mode 100644 index 00000000..d9d33e91 --- /dev/null +++ b/backend/services/reporting-service/prisma/schema.prisma @@ -0,0 +1,322 @@ +generator client { + provider = "prisma-client-js" +} + +datasource db { + provider = "postgresql" + url = env("DATABASE_URL") +} + +// ============================================ +// 报表定义表 (聚合根1) +// 定义各类报表的配置和调度规则 +// ============================================ +model ReportDefinition { + id BigInt @id @default(autoincrement()) @map("definition_id") + + // === 报表基本信息 === + reportType String @map("report_type") @db.VarChar(50) + reportName String @map("report_name") @db.VarChar(200) + reportCode String @unique @map("report_code") @db.VarChar(50) + description String? @map("description") @db.Text + + // === 报表参数 === + parameters Json @map("parameters") + + // === 调度配置 === + scheduleCron String? @map("schedule_cron") @db.VarChar(100) + scheduleTimezone String? @map("schedule_timezone") @db.VarChar(50) @default("Asia/Shanghai") + scheduleEnabled Boolean @default(false) @map("schedule_enabled") + + // === 输出格式 === + outputFormats String[] @map("output_formats") + + // === 状态 === + isActive Boolean @default(true) @map("is_active") + + // === 时间戳 === + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + lastGeneratedAt DateTime? @map("last_generated_at") + + @@map("report_definitions") + @@index([reportType], name: "idx_def_type") + @@index([isActive], name: "idx_def_active") + @@index([scheduleEnabled], name: "idx_def_scheduled") +} + +// ============================================ +// 报表快照表 (聚合根2 - 读模型) +// 存储已生成的报表数据快照 +// ============================================ +model ReportSnapshot { + id BigInt @id @default(autoincrement()) @map("snapshot_id") + + // === 报表信息 === + reportType String @map("report_type") @db.VarChar(50) + reportCode String @map("report_code") @db.VarChar(50) + reportPeriod String @map("report_period") @db.VarChar(20) + periodKey String @map("period_key") @db.VarChar(30) + + // === 快照数据 === + snapshotData Json @map("snapshot_data") + summaryData Json? @map("summary_data") + + // === 数据来源 === + dataSources String[] @map("data_sources") + dataFreshness Int @default(0) @map("data_freshness") + + // === 过滤条件 === + filterParams Json? @map("filter_params") + + // === 统计信息 === + rowCount Int @default(0) @map("row_count") + + // === 时间戳 === + periodStartAt DateTime @map("period_start_at") + periodEndAt DateTime @map("period_end_at") + generatedAt DateTime @default(now()) @map("generated_at") + expiresAt DateTime? @map("expires_at") + + // === 关联 === + files ReportFile[] + + @@map("report_snapshots") + @@unique([reportCode, periodKey], name: "uk_report_period") + @@index([reportType], name: "idx_snapshot_type") + @@index([reportCode], name: "idx_snapshot_code") + @@index([periodKey], name: "idx_snapshot_period") + @@index([generatedAt(sort: Desc)], name: "idx_snapshot_generated") + @@index([expiresAt], name: "idx_snapshot_expires") +} + +// ============================================ +// 报表文件表 +// 存储已导出的报表文件信息 +// ============================================ +model ReportFile { + id BigInt @id @default(autoincrement()) @map("file_id") + snapshotId BigInt @map("snapshot_id") + + // === 文件信息 === + fileName String @map("file_name") @db.VarChar(500) + filePath String @map("file_path") @db.VarChar(1000) + fileUrl String? @map("file_url") @db.VarChar(1000) + fileSize BigInt @map("file_size") + fileFormat String @map("file_format") @db.VarChar(20) + mimeType String @map("mime_type") @db.VarChar(100) + + // === 访问信息 === + downloadCount Int @default(0) @map("download_count") + lastDownloadAt DateTime? @map("last_download_at") + + // === 时间戳 === + createdAt DateTime @default(now()) @map("created_at") + expiresAt DateTime? @map("expires_at") + + // === 关联 === + snapshot ReportSnapshot @relation(fields: [snapshotId], references: [id], onDelete: Cascade) + + @@map("report_files") + @@index([snapshotId], name: "idx_file_snapshot") + @@index([fileFormat], name: "idx_file_format") + @@index([createdAt(sort: Desc)], name: "idx_file_created") +} + +// ============================================ +// 分析指标表 (聚合数据) +// 存储预聚合的分析指标数据 +// ============================================ +model AnalyticsMetric { + id BigInt @id @default(autoincrement()) @map("metric_id") + + // === 指标信息 === + metricType String @map("metric_type") @db.VarChar(50) + metricCode String @map("metric_code") @db.VarChar(50) + + // === 维度 === + dimensionTime DateTime? @map("dimension_time") @db.Date + dimensionRegion String? @map("dimension_region") @db.VarChar(100) + dimensionUserType String? @map("dimension_user_type") @db.VarChar(50) + dimensionRightType String? @map("dimension_right_type") @db.VarChar(50) + + // === 指标值 === + metricValue Decimal @map("metric_value") @db.Decimal(20, 8) + metricData Json? @map("metric_data") + + // === 时间戳 === + calculatedAt DateTime @default(now()) @map("calculated_at") + + @@map("analytics_metrics") + @@unique([metricCode, dimensionTime, dimensionRegion, dimensionUserType, dimensionRightType], name: "uk_metric_dimensions") + @@index([metricType], name: "idx_metric_type") + @@index([metricCode], name: "idx_metric_code") + @@index([dimensionTime], name: "idx_metric_time") + @@index([dimensionRegion], name: "idx_metric_region") +} + +// ============================================ +// 认种统计日表 (每日聚合) +// ============================================ +model PlantingDailyStat { + id BigInt @id @default(autoincrement()) @map("stat_id") + + // === 统计日期 === + statDate DateTime @map("stat_date") @db.Date + + // === 区域维度 === + provinceCode String? @map("province_code") @db.VarChar(10) + cityCode String? @map("city_code") @db.VarChar(10) + + // === 统计数据 === + orderCount Int @default(0) @map("order_count") + treeCount Int @default(0) @map("tree_count") + totalAmount Decimal @default(0) @map("total_amount") @db.Decimal(20, 8) + newUserCount Int @default(0) @map("new_user_count") + activeUserCount Int @default(0) @map("active_user_count") + + // === 时间戳 === + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + @@map("planting_daily_stats") + @@unique([statDate, provinceCode, cityCode], name: "uk_daily_stat") + @@index([statDate], name: "idx_pds_date") + @@index([provinceCode], name: "idx_pds_province") + @@index([cityCode], name: "idx_pds_city") +} + +// ============================================ +// 社区统计表 +// ============================================ +model CommunityStat { + id BigInt @id @default(autoincrement()) @map("stat_id") + + // === 社区信息 === + communityId BigInt @map("community_id") + communityName String @map("community_name") @db.VarChar(200) + parentCommunityId BigInt? @map("parent_community_id") + + // === 统计日期 === + statDate DateTime @map("stat_date") @db.Date + + // === 统计数据 === + totalPlanting Int @default(0) @map("total_planting") + dailyPlanting Int @default(0) @map("daily_planting") + weeklyPlanting Int @default(0) @map("weekly_planting") + monthlyPlanting Int @default(0) @map("monthly_planting") + memberCount Int @default(0) @map("member_count") + + // === 时间戳 === + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + @@map("community_stats") + @@unique([communityId, statDate], name: "uk_community_stat") + @@index([communityId], name: "idx_cs_community") + @@index([communityName], name: "idx_cs_name") + @@index([statDate], name: "idx_cs_date") + @@index([parentCommunityId], name: "idx_cs_parent") +} + +// ============================================ +// 系统账户月度统计表 +// 省公司/市公司账户的月度数据 +// ============================================ +model SystemAccountMonthlyStat { + id BigInt @id @default(autoincrement()) @map("stat_id") + + // === 账户信息 === + accountId BigInt @map("account_id") + accountType String @map("account_type") @db.VarChar(30) + accountName String @map("account_name") @db.VarChar(200) + regionCode String @map("region_code") @db.VarChar(10) + + // === 统计月份 === + statMonth String @map("stat_month") @db.VarChar(7) + + // === 月度数据 === + monthlyHashpower Decimal @default(0) @map("monthly_hashpower") @db.Decimal(20, 8) + cumulativeHashpower Decimal @default(0) @map("cumulative_hashpower") @db.Decimal(20, 8) + monthlyMining Decimal @default(0) @map("monthly_mining") @db.Decimal(20, 8) + cumulativeMining Decimal @default(0) @map("cumulative_mining") @db.Decimal(20, 8) + monthlyCommission Decimal @default(0) @map("monthly_commission") @db.Decimal(20, 8) + cumulativeCommission Decimal @default(0) @map("cumulative_commission") @db.Decimal(20, 8) + monthlyPlantingBonus Decimal @default(0) @map("monthly_planting_bonus") @db.Decimal(20, 8) + cumulativePlantingBonus Decimal @default(0) @map("cumulative_planting_bonus") @db.Decimal(20, 8) + + // === 时间戳 === + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + @@map("system_account_monthly_stats") + @@unique([accountId, statMonth], name: "uk_account_month") + @@index([accountType], name: "idx_sams_type") + @@index([statMonth], name: "idx_sams_month") + @@index([regionCode], name: "idx_sams_region") +} + +// ============================================ +// 系统账户收益流水表 +// 记录每笔收益的来源和时间 +// ============================================ +model SystemAccountIncomeRecord { + id BigInt @id @default(autoincrement()) @map("record_id") + + // === 账户信息 === + accountId BigInt @map("account_id") + accountType String @map("account_type") @db.VarChar(30) + + // === 收益信息 === + incomeType String @map("income_type") @db.VarChar(50) + incomeAmount Decimal @map("income_amount") @db.Decimal(20, 8) + currency String @map("currency") @db.VarChar(10) + + // === 来源信息 === + sourceType String @map("source_type") @db.VarChar(50) + sourceId String? @map("source_id") @db.VarChar(100) + sourceUserId BigInt? @map("source_user_id") + sourceAddress String? @map("source_address") @db.VarChar(200) + transactionNo String? @map("transaction_no") @db.VarChar(100) + + // === 备注 === + memo String? @map("memo") @db.Text + + // === 时间戳 === + occurredAt DateTime @map("occurred_at") + createdAt DateTime @default(now()) @map("created_at") + + @@map("system_account_income_records") + @@index([accountId], name: "idx_sair_account") + @@index([accountType], name: "idx_sair_type") + @@index([incomeType], name: "idx_sair_income_type") + @@index([sourceType], name: "idx_sair_source_type") + @@index([sourceAddress], name: "idx_sair_address") + @@index([transactionNo], name: "idx_sair_txno") + @@index([occurredAt(sort: Desc)], name: "idx_sair_occurred") +} + +// ============================================ +// 报表事件表 +// ============================================ +model ReportEvent { + 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("report_events") + @@index([aggregateType, aggregateId], name: "idx_report_event_aggregate") + @@index([eventType], name: "idx_report_event_type") + @@index([occurredAt], name: "idx_report_event_occurred") +} diff --git a/backend/services/reporting-service/prisma/seed.ts b/backend/services/reporting-service/prisma/seed.ts new file mode 100644 index 00000000..a710fea0 --- /dev/null +++ b/backend/services/reporting-service/prisma/seed.ts @@ -0,0 +1,129 @@ +import { PrismaClient } from '@prisma/client'; + +const prisma = new PrismaClient(); + +async function main() { + // 初始化报表定义 + const reportDefinitions = [ + { + reportType: 'LEADERBOARD_REPORT', + reportName: '龙虎榜数据报表', + reportCode: 'RPT_LEADERBOARD', + description: '龙虎榜日榜/周榜/月榜排名数据统计', + parameters: { + dimensions: ['TIME', 'USER'], + defaultPeriod: 'DAILY', + }, + outputFormats: ['EXCEL', 'CSV'], + isActive: true, + }, + { + reportType: 'PLANTING_REPORT', + reportName: '榴莲树认种报表', + reportCode: 'RPT_PLANTING', + description: '榴莲树认种日/周/月/季度/年度报表', + parameters: { + dimensions: ['TIME', 'REGION'], + defaultPeriod: 'DAILY', + }, + scheduleCron: '0 1 * * *', + scheduleEnabled: true, + outputFormats: ['EXCEL', 'CSV', 'PDF'], + isActive: true, + }, + { + reportType: 'REGIONAL_PLANTING_REPORT', + reportName: '区域认种报表', + reportCode: 'RPT_REGIONAL_PLANTING', + description: '按省/市统计的认种报表', + parameters: { + dimensions: ['REGION', 'TIME'], + defaultPeriod: 'DAILY', + }, + scheduleCron: '0 2 * * *', + scheduleEnabled: true, + outputFormats: ['EXCEL', 'CSV'], + isActive: true, + }, + { + reportType: 'AUTHORIZED_COMPANY_TOP_REPORT', + reportName: '授权公司第1名统计', + reportCode: 'RPT_COMPANY_TOP', + description: '各授权省公司和市公司的第1名及完成数据', + parameters: { + dimensions: ['REGION', 'USER'], + includeProvince: true, + includeCity: true, + }, + outputFormats: ['EXCEL', 'CSV'], + isActive: true, + }, + { + reportType: 'COMMUNITY_REPORT', + reportName: '社区数据统计', + reportCode: 'RPT_COMMUNITY', + description: '社区认种总量、日/周/月新增、上下级社区统计', + parameters: { + dimensions: ['COMMUNITY', 'TIME'], + supportFuzzySearch: true, + }, + outputFormats: ['EXCEL', 'CSV'], + isActive: true, + }, + { + reportType: 'SYSTEM_ACCOUNT_MONTHLY_REPORT', + reportName: '系统账户月度报表', + reportCode: 'RPT_SYSTEM_ACCOUNT_MONTHLY', + description: '系统省/市公司账户每月各项数据统计', + parameters: { + dimensions: ['ACCOUNT', 'TIME'], + metrics: [ + 'monthlyHashpower', + 'cumulativeHashpower', + 'monthlyMining', + 'cumulativeMining', + 'monthlyCommission', + 'cumulativeCommission', + 'monthlyPlantingBonus', + 'cumulativePlantingBonus', + ], + }, + scheduleCron: '0 0 1 * *', + scheduleEnabled: true, + outputFormats: ['EXCEL', 'CSV'], + isActive: true, + }, + { + reportType: 'SYSTEM_ACCOUNT_INCOME_REPORT', + reportName: '系统账户收益来源报表', + reportCode: 'RPT_SYSTEM_ACCOUNT_INCOME', + description: '系统省/市公司账户收益来源细分统计及时间轴', + parameters: { + dimensions: ['ACCOUNT', 'TIME', 'SOURCE'], + supportTimeFilter: true, + supportKeywordSearch: true, + }, + outputFormats: ['EXCEL', 'CSV'], + isActive: true, + }, + ]; + + for (const def of reportDefinitions) { + await prisma.reportDefinition.upsert({ + where: { reportCode: def.reportCode }, + update: def, + create: def, + }); + } + + console.log('Seed completed: Report definitions initialized'); +} + +main() + .catch((e) => { + console.error(e); + process.exit(1); + }) + .finally(async () => { + await prisma.$disconnect(); + }); diff --git a/backend/services/reporting-service/src/api/api.module.ts b/backend/services/reporting-service/src/api/api.module.ts new file mode 100644 index 00000000..6a9a958c --- /dev/null +++ b/backend/services/reporting-service/src/api/api.module.ts @@ -0,0 +1,11 @@ +import { Module } from '@nestjs/common'; +import { ApplicationModule } from '../application/application.module'; +import { HealthController } from './controllers/health.controller'; +import { ReportController } from './controllers/report.controller'; +import { ExportController } from './controllers/export.controller'; + +@Module({ + imports: [ApplicationModule], + controllers: [HealthController, ReportController, ExportController], +}) +export class ApiModule {} diff --git a/backend/services/reporting-service/src/api/controllers/export.controller.ts b/backend/services/reporting-service/src/api/controllers/export.controller.ts new file mode 100644 index 00000000..c8da70ae --- /dev/null +++ b/backend/services/reporting-service/src/api/controllers/export.controller.ts @@ -0,0 +1,73 @@ +import { + Controller, + Post, + Body, + Get, + Param, + Res, + HttpCode, + HttpStatus, + NotFoundException, +} from '@nestjs/common'; +import { + ApiTags, + ApiOperation, + ApiResponse, + ApiBearerAuth, +} from '@nestjs/swagger'; +import { Response } from 'express'; +import * as fs from 'fs'; +import { ExportReportDto } from '../dto/request/export-report.dto'; +import { ReportFileResponseDto } from '../dto/response/report-file.dto'; +import { ExportReportHandler } from '../../application/commands/export-report/export-report.handler'; +import { ExportReportCommand } from '../../application/commands/export-report/export-report.command'; + +@ApiTags('Export') +@Controller('export') +@ApiBearerAuth() +export class ExportController { + constructor(private readonly exportReportHandler: ExportReportHandler) {} + + @Post() + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: '导出报表' }) + @ApiResponse({ + status: 200, + description: '导出成功', + type: ReportFileResponseDto, + }) + async exportReport( + @Body() dto: ExportReportDto, + ): Promise { + const command = new ExportReportCommand(BigInt(dto.snapshotId), dto.format); + + const file = await this.exportReportHandler.execute(command); + + return { + id: file.id?.toString() || '', + fileName: file.fileName, + fileUrl: file.fileUrl || '', + fileSize: file.fileSize.toString(), + fileFormat: file.fileFormat, + mimeType: file.mimeType, + createdAt: file.createdAt, + }; + } + + @Get('download/:fileId') + @ApiOperation({ summary: '下载报表文件' }) + @ApiResponse({ status: 200, description: '文件下载' }) + @ApiResponse({ status: 404, description: '文件不存在' }) + async downloadFile( + @Param('fileId') fileId: string, + @Res() res: Response, + ): Promise { + // In a real implementation, you would: + // 1. Look up the file record from database + // 2. Verify user has access + // 3. Stream the file + + // For now, return a placeholder response + throw new NotFoundException(`File not found: ${fileId}`); + } +} diff --git a/backend/services/reporting-service/src/api/controllers/health.controller.ts b/backend/services/reporting-service/src/api/controllers/health.controller.ts new file mode 100644 index 00000000..f413a3d9 --- /dev/null +++ b/backend/services/reporting-service/src/api/controllers/health.controller.ts @@ -0,0 +1,28 @@ +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() { + return { + status: 'ok', + service: 'reporting-service', + timestamp: new Date().toISOString(), + }; + } + + @Get('ready') + @ApiOperation({ summary: '就绪检查' }) + @ApiResponse({ status: 200, description: '服务就绪' }) + ready() { + return { + status: 'ready', + service: 'reporting-service', + timestamp: new Date().toISOString(), + }; + } +} diff --git a/backend/services/reporting-service/src/api/controllers/report.controller.ts b/backend/services/reporting-service/src/api/controllers/report.controller.ts new file mode 100644 index 00000000..4eaa649b --- /dev/null +++ b/backend/services/reporting-service/src/api/controllers/report.controller.ts @@ -0,0 +1,154 @@ +import { + Controller, + Get, + Post, + Body, + Param, + Query, + HttpCode, + HttpStatus, + NotFoundException, +} from '@nestjs/common'; +import { + ApiTags, + ApiOperation, + ApiResponse, + ApiBearerAuth, +} from '@nestjs/swagger'; +import { GenerateReportDto } from '../dto/request/generate-report.dto'; +import { QueryReportDto } from '../dto/request/query-report.dto'; +import { ReportSnapshotResponseDto } from '../dto/response/report-snapshot.dto'; +import { ReportDefinitionResponseDto } from '../dto/response/report-definition.dto'; +import { GenerateReportHandler } from '../../application/commands/generate-report/generate-report.handler'; +import { GenerateReportCommand } from '../../application/commands/generate-report/generate-report.command'; +import { ReportingApplicationService } from '../../application/services/reporting-application.service'; + +@ApiTags('Reports') +@Controller('reports') +@ApiBearerAuth() +export class ReportController { + constructor( + private readonly generateReportHandler: GenerateReportHandler, + private readonly reportingService: ReportingApplicationService, + ) {} + + @Get('definitions') + @ApiOperation({ summary: '获取所有报表定义' }) + @ApiResponse({ + status: 200, + description: '报表定义列表', + type: [ReportDefinitionResponseDto], + }) + async getDefinitions(): Promise { + return this.reportingService.getReportDefinitions(); + } + + @Get('definitions/:code') + @ApiOperation({ summary: '获取报表定义详情' }) + @ApiResponse({ + status: 200, + description: '报表定义详情', + type: ReportDefinitionResponseDto, + }) + @ApiResponse({ status: 404, description: '报表定义不存在' }) + async getDefinition( + @Param('code') code: string, + ): Promise { + const definition = await this.reportingService.getReportDefinitionByCode(code); + if (!definition) { + throw new NotFoundException(`Report definition not found: ${code}`); + } + return definition; + } + + @Post('generate') + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: '生成报表' }) + @ApiResponse({ + status: 200, + description: '报表生成成功', + type: ReportSnapshotResponseDto, + }) + async generateReport( + @Body() dto: GenerateReportDto, + ): Promise { + const command = new GenerateReportCommand( + dto.reportCode, + dto.reportPeriod, + new Date(dto.startDate), + new Date(dto.endDate), + dto.dimensions, + dto.filters, + ); + + const snapshot = await this.generateReportHandler.execute(command); + + return { + id: snapshot.id?.toString() || '', + reportType: snapshot.reportType, + reportCode: snapshot.reportCode, + reportPeriod: snapshot.reportPeriod, + periodKey: snapshot.periodKey, + rowCount: snapshot.rowCount, + summary: snapshot.snapshotData.summary, + rows: snapshot.snapshotData.rows, + periodStartAt: snapshot.periodStartAt, + periodEndAt: snapshot.periodEndAt, + generatedAt: snapshot.generatedAt, + }; + } + + @Get('snapshots') + @ApiOperation({ summary: '查询报表快照' }) + @ApiResponse({ + status: 200, + description: '报表快照列表', + type: [ReportSnapshotResponseDto], + }) + async getSnapshots( + @Query() query: QueryReportDto, + ): Promise { + if (query.reportCode) { + const snapshots = await this.reportingService.getReportSnapshots( + query.reportCode, + ); + return snapshots.map((s) => ({ + ...s, + rows: undefined, + })); + } + + if (query.reportPeriod) { + const snapshots = await this.reportingService.getSnapshotsByPeriod( + query.reportPeriod, + ); + return snapshots.map((s) => ({ + ...s, + rows: undefined, + })); + } + + return []; + } + + @Get('snapshots/:code/latest') + @ApiOperation({ summary: '获取最新报表快照' }) + @ApiResponse({ + status: 200, + description: '最新报表快照', + type: ReportSnapshotResponseDto, + }) + @ApiResponse({ status: 404, description: '快照不存在' }) + async getLatestSnapshot( + @Param('code') code: string, + ): Promise { + const snapshot = await this.reportingService.getLatestSnapshot(code); + if (!snapshot) { + throw new NotFoundException(`No snapshot found for report: ${code}`); + } + return { + ...snapshot, + rows: undefined, + }; + } +} diff --git a/backend/services/reporting-service/src/api/dto/request/export-report.dto.ts b/backend/services/reporting-service/src/api/dto/request/export-report.dto.ts new file mode 100644 index 00000000..7c769d93 --- /dev/null +++ b/backend/services/reporting-service/src/api/dto/request/export-report.dto.ts @@ -0,0 +1,17 @@ +import { ApiProperty } from '@nestjs/swagger'; +import { IsString, IsEnum } from 'class-validator'; +import { OutputFormat } from '../../../domain/value-objects'; + +export class ExportReportDto { + @ApiProperty({ description: '快照ID', example: '1' }) + @IsString() + snapshotId: string; + + @ApiProperty({ + description: '导出格式', + enum: OutputFormat, + example: OutputFormat.EXCEL, + }) + @IsEnum(OutputFormat) + format: OutputFormat; +} diff --git a/backend/services/reporting-service/src/api/dto/request/generate-report.dto.ts b/backend/services/reporting-service/src/api/dto/request/generate-report.dto.ts new file mode 100644 index 00000000..cf27c745 --- /dev/null +++ b/backend/services/reporting-service/src/api/dto/request/generate-report.dto.ts @@ -0,0 +1,47 @@ +import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; +import { + IsString, + IsEnum, + IsDateString, + IsOptional, + IsObject, + IsArray, +} from 'class-validator'; +import { ReportPeriod, ReportDimension } from '../../../domain/value-objects'; + +export class GenerateReportDto { + @ApiProperty({ description: '报表代码', example: 'RPT_LEADERBOARD' }) + @IsString() + reportCode: string; + + @ApiProperty({ + description: '报表周期', + enum: ReportPeriod, + example: ReportPeriod.DAILY, + }) + @IsEnum(ReportPeriod) + reportPeriod: ReportPeriod; + + @ApiProperty({ description: '开始日期', example: '2024-01-01' }) + @IsDateString() + startDate: string; + + @ApiProperty({ description: '结束日期', example: '2024-01-31' }) + @IsDateString() + endDate: string; + + @ApiPropertyOptional({ + description: '报表维度', + enum: ReportDimension, + isArray: true, + }) + @IsOptional() + @IsArray() + @IsEnum(ReportDimension, { each: true }) + dimensions?: ReportDimension[]; + + @ApiPropertyOptional({ description: '筛选条件' }) + @IsOptional() + @IsObject() + filters?: Record; +} diff --git a/backend/services/reporting-service/src/api/dto/request/query-report.dto.ts b/backend/services/reporting-service/src/api/dto/request/query-report.dto.ts new file mode 100644 index 00000000..58640da4 --- /dev/null +++ b/backend/services/reporting-service/src/api/dto/request/query-report.dto.ts @@ -0,0 +1,33 @@ +import { ApiPropertyOptional } from '@nestjs/swagger'; +import { IsOptional, IsString, IsEnum, IsDateString } from 'class-validator'; +import { ReportPeriod } from '../../../domain/value-objects'; + +export class QueryReportDto { + @ApiPropertyOptional({ description: '报表代码' }) + @IsOptional() + @IsString() + reportCode?: string; + + @ApiPropertyOptional({ description: '周期键', example: '2024-01-15' }) + @IsOptional() + @IsString() + periodKey?: string; + + @ApiPropertyOptional({ + description: '报表周期', + enum: ReportPeriod, + }) + @IsOptional() + @IsEnum(ReportPeriod) + reportPeriod?: ReportPeriod; + + @ApiPropertyOptional({ description: '开始日期' }) + @IsOptional() + @IsDateString() + startDate?: string; + + @ApiPropertyOptional({ description: '结束日期' }) + @IsOptional() + @IsDateString() + endDate?: string; +} diff --git a/backend/services/reporting-service/src/api/dto/response/report-definition.dto.ts b/backend/services/reporting-service/src/api/dto/response/report-definition.dto.ts new file mode 100644 index 00000000..4f69a2be --- /dev/null +++ b/backend/services/reporting-service/src/api/dto/response/report-definition.dto.ts @@ -0,0 +1,36 @@ +import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; + +export class ReportDefinitionResponseDto { + @ApiProperty({ description: '定义ID' }) + id: string; + + @ApiProperty({ description: '报表类型' }) + reportType: string; + + @ApiProperty({ description: '报表名称' }) + reportName: string; + + @ApiProperty({ description: '报表代码' }) + reportCode: string; + + @ApiProperty({ description: '报表描述' }) + description: string; + + @ApiProperty({ description: '报表参数' }) + parameters: Record; + + @ApiPropertyOptional({ description: '调度Cron表达式' }) + scheduleCron?: string | null; + + @ApiProperty({ description: '调度是否启用' }) + scheduleEnabled: boolean; + + @ApiProperty({ description: '支持的输出格式', isArray: true }) + outputFormats: string[]; + + @ApiProperty({ description: '是否激活' }) + isActive: boolean; + + @ApiPropertyOptional({ description: '最后生成时间' }) + lastGeneratedAt?: Date | null; +} diff --git a/backend/services/reporting-service/src/api/dto/response/report-file.dto.ts b/backend/services/reporting-service/src/api/dto/response/report-file.dto.ts new file mode 100644 index 00000000..aaf4b092 --- /dev/null +++ b/backend/services/reporting-service/src/api/dto/response/report-file.dto.ts @@ -0,0 +1,24 @@ +import { ApiProperty } from '@nestjs/swagger'; + +export class ReportFileResponseDto { + @ApiProperty({ description: '文件ID' }) + id: string; + + @ApiProperty({ description: '文件名' }) + fileName: string; + + @ApiProperty({ description: '文件URL' }) + fileUrl: string; + + @ApiProperty({ description: '文件大小(字节)' }) + fileSize: string; + + @ApiProperty({ description: '文件格式' }) + fileFormat: string; + + @ApiProperty({ description: 'MIME类型' }) + mimeType: string; + + @ApiProperty({ description: '创建时间' }) + createdAt: Date; +} diff --git a/backend/services/reporting-service/src/api/dto/response/report-snapshot.dto.ts b/backend/services/reporting-service/src/api/dto/response/report-snapshot.dto.ts new file mode 100644 index 00000000..894cff6e --- /dev/null +++ b/backend/services/reporting-service/src/api/dto/response/report-snapshot.dto.ts @@ -0,0 +1,36 @@ +import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; + +export class ReportSnapshotResponseDto { + @ApiProperty({ description: '快照ID' }) + id: string; + + @ApiProperty({ description: '报表类型' }) + reportType: string; + + @ApiProperty({ description: '报表代码' }) + reportCode: string; + + @ApiProperty({ description: '报表周期' }) + reportPeriod: string; + + @ApiProperty({ description: '周期键' }) + periodKey: string; + + @ApiProperty({ description: '数据行数' }) + rowCount: number; + + @ApiPropertyOptional({ description: '汇总数据' }) + summary?: Record; + + @ApiPropertyOptional({ description: '数据行' }) + rows?: any[]; + + @ApiProperty({ description: '周期开始时间' }) + periodStartAt: Date; + + @ApiProperty({ description: '周期结束时间' }) + periodEndAt: Date; + + @ApiProperty({ description: '生成时间' }) + generatedAt: Date; +} diff --git a/backend/services/reporting-service/src/app.module.ts b/backend/services/reporting-service/src/app.module.ts new file mode 100644 index 00000000..9335120d --- /dev/null +++ b/backend/services/reporting-service/src/app.module.ts @@ -0,0 +1,42 @@ +import { Module } from '@nestjs/common'; +import { ConfigModule } from '@nestjs/config'; +import { APP_FILTER, APP_INTERCEPTOR, APP_GUARD } from '@nestjs/core'; +import { ApiModule } from './api/api.module'; +import { GlobalExceptionFilter } from './shared/filters/global-exception.filter'; +import { TransformInterceptor } from './shared/interceptors/transform.interceptor'; +import { JwtAuthGuard } from './shared/guards/jwt-auth.guard'; +import { JwtStrategy } from './shared/strategies/jwt.strategy'; +import { + appConfig, + databaseConfig, + jwtConfig, + redisConfig, +} from './config'; + +@Module({ + imports: [ + ConfigModule.forRoot({ + isGlobal: true, + envFilePath: ['.env.development', '.env'], + load: [appConfig, databaseConfig, jwtConfig, redisConfig], + }), + ApiModule, + ], + providers: [ + JwtStrategy, + { + provide: APP_FILTER, + useClass: GlobalExceptionFilter, + }, + { + provide: APP_INTERCEPTOR, + useClass: TransformInterceptor, + }, + // Uncomment to enable JWT auth globally + // { + // provide: APP_GUARD, + // useClass: JwtAuthGuard, + // }, + ], +}) +export class AppModule {} diff --git a/backend/services/reporting-service/src/application/application.module.ts b/backend/services/reporting-service/src/application/application.module.ts new file mode 100644 index 00000000..9ab4a724 --- /dev/null +++ b/backend/services/reporting-service/src/application/application.module.ts @@ -0,0 +1,27 @@ +import { Module } from '@nestjs/common'; +import { ScheduleModule } from '@nestjs/schedule'; +import { DomainModule } from '../domain/domain.module'; +import { InfrastructureModule } from '../infrastructure/infrastructure.module'; +import { GenerateReportHandler } from './commands/generate-report/generate-report.handler'; +import { ExportReportHandler } from './commands/export-report/export-report.handler'; +import { GetReportSnapshotHandler } from './queries/get-report-snapshot/get-report-snapshot.handler'; +import { ReportingApplicationService } from './services/reporting-application.service'; +import { ReportGenerationScheduler } from './schedulers/report-generation.scheduler'; + +@Module({ + imports: [ScheduleModule.forRoot(), DomainModule, InfrastructureModule], + providers: [ + GenerateReportHandler, + ExportReportHandler, + GetReportSnapshotHandler, + ReportingApplicationService, + ReportGenerationScheduler, + ], + exports: [ + GenerateReportHandler, + ExportReportHandler, + GetReportSnapshotHandler, + ReportingApplicationService, + ], +}) +export class ApplicationModule {} diff --git a/backend/services/reporting-service/src/application/commands/export-report/export-report.command.ts b/backend/services/reporting-service/src/application/commands/export-report/export-report.command.ts new file mode 100644 index 00000000..06c09b7b --- /dev/null +++ b/backend/services/reporting-service/src/application/commands/export-report/export-report.command.ts @@ -0,0 +1,8 @@ +import { OutputFormat } from '../../../domain/value-objects'; + +export class ExportReportCommand { + constructor( + public readonly snapshotId: bigint, + public readonly format: OutputFormat, + ) {} +} diff --git a/backend/services/reporting-service/src/application/commands/export-report/export-report.handler.ts b/backend/services/reporting-service/src/application/commands/export-report/export-report.handler.ts new file mode 100644 index 00000000..09b4fae1 --- /dev/null +++ b/backend/services/reporting-service/src/application/commands/export-report/export-report.handler.ts @@ -0,0 +1,181 @@ +import { Injectable, Inject, Logger, NotFoundException } from '@nestjs/common'; +import { ConfigService } from '@nestjs/config'; +import { ExportReportCommand } from './export-report.command'; +import { + IReportSnapshotRepository, + IReportFileRepository, + REPORT_SNAPSHOT_REPOSITORY, + REPORT_FILE_REPOSITORY, +} from '../../../domain/repositories'; +import { ReportFile } from '../../../domain/entities/report-file.entity'; +import { + OutputFormat, + OutputFormatMimeTypes, + OutputFormatExtensions, +} from '../../../domain/value-objects'; +import { ExcelExportService } from '../../../infrastructure/export/excel-export.service'; +import { CsvExportService } from '../../../infrastructure/export/csv-export.service'; +import { PdfExportService } from '../../../infrastructure/export/pdf-export.service'; + +@Injectable() +export class ExportReportHandler { + private readonly logger = new Logger(ExportReportHandler.name); + private readonly fileUrlPrefix: string; + + constructor( + @Inject(REPORT_SNAPSHOT_REPOSITORY) + private readonly snapshotRepo: IReportSnapshotRepository, + @Inject(REPORT_FILE_REPOSITORY) + private readonly fileRepo: IReportFileRepository, + private readonly excelExportService: ExcelExportService, + private readonly csvExportService: CsvExportService, + private readonly pdfExportService: PdfExportService, + private readonly configService: ConfigService, + ) { + this.fileUrlPrefix = this.configService.get( + 'FILE_STORAGE_URL_PREFIX', + 'http://localhost:3008/files', + ); + } + + async execute(command: ExportReportCommand): Promise { + this.logger.log( + `Exporting report snapshot ${command.snapshotId} to ${command.format}`, + ); + + // Find snapshot + const snapshot = await this.snapshotRepo.findById(command.snapshotId); + if (!snapshot) { + throw new NotFoundException( + `Report snapshot not found: ${command.snapshotId}`, + ); + } + + // Check if file already exists + const existingFile = await this.fileRepo.findBySnapshotIdAndFormat( + command.snapshotId, + command.format, + ); + + if (existingFile && !existingFile.isExpired()) { + this.logger.debug(`Using existing file for snapshot ${command.snapshotId}`); + return existingFile; + } + + // Generate file name + const extension = OutputFormatExtensions[command.format]; + const fileName = `${snapshot.reportCode}_${snapshot.periodKey}_${Date.now()}.${extension}`; + + // Get columns from snapshot data + const rows = snapshot.snapshotData.rows; + const columns = this.inferColumns(rows); + + // Export based on format + let exportResult: { filePath: string; fileSize: number }; + + switch (command.format) { + case OutputFormat.EXCEL: + exportResult = await this.excelExportService.export(fileName, { + sheetName: snapshot.reportCode, + columns, + data: rows, + title: `${snapshot.reportCode} Report`, + subtitle: `Period: ${snapshot.periodKey}`, + }); + break; + + case OutputFormat.CSV: + exportResult = await this.csvExportService.export(fileName, { + columns, + data: rows, + }); + break; + + case OutputFormat.PDF: + exportResult = await this.pdfExportService.export(fileName, { + columns, + data: rows, + title: `${snapshot.reportCode} Report`, + subtitle: `Period: ${snapshot.periodKey}`, + }); + break; + + case OutputFormat.JSON: + exportResult = await this.exportJson(fileName, rows); + break; + + default: + throw new Error(`Unsupported format: ${command.format}`); + } + + // Create report file entity + const reportFile = ReportFile.create({ + snapshotId: command.snapshotId, + fileName, + filePath: exportResult.filePath, + fileUrl: `${this.fileUrlPrefix}/${fileName}`, + fileSize: BigInt(exportResult.fileSize), + fileFormat: command.format, + mimeType: OutputFormatMimeTypes[command.format], + }); + + // Save file record + const savedFile = await this.fileRepo.save(reportFile); + + this.logger.log( + `Report exported: ${fileName} (${exportResult.fileSize} bytes)`, + ); + + return savedFile; + } + + private inferColumns( + rows: any[], + ): { header: string; key: string; width?: number }[] { + if (rows.length === 0) { + return []; + } + + const firstRow = rows[0]; + return Object.keys(firstRow).map((key) => ({ + header: this.formatHeader(key), + key, + width: 15, + })); + } + + private formatHeader(key: string): string { + return key + .replace(/([A-Z])/g, ' $1') + .replace(/^./, (str) => str.toUpperCase()) + .trim(); + } + + private async exportJson( + fileName: string, + data: any[], + ): Promise<{ filePath: string; fileSize: number }> { + const fs = await import('fs'); + const path = await import('path'); + const storagePath = this.configService.get( + 'FILE_STORAGE_PATH', + './storage/reports', + ); + + const fullPath = path.join(storagePath, fileName); + const dir = path.dirname(fullPath); + if (!fs.existsSync(dir)) { + fs.mkdirSync(dir, { recursive: true }); + } + + const content = JSON.stringify(data, null, 2); + fs.writeFileSync(fullPath, content, 'utf8'); + + const stats = fs.statSync(fullPath); + + return { + filePath: fullPath, + fileSize: stats.size, + }; + } +} diff --git a/backend/services/reporting-service/src/application/commands/generate-report/generate-report.command.ts b/backend/services/reporting-service/src/application/commands/generate-report/generate-report.command.ts new file mode 100644 index 00000000..01351c76 --- /dev/null +++ b/backend/services/reporting-service/src/application/commands/generate-report/generate-report.command.ts @@ -0,0 +1,12 @@ +import { ReportType, ReportPeriod, ReportDimension } from '../../../domain/value-objects'; + +export class GenerateReportCommand { + constructor( + public readonly reportCode: string, + public readonly reportPeriod: ReportPeriod, + public readonly startDate: Date, + public readonly endDate: Date, + public readonly dimensions?: ReportDimension[], + public readonly filters?: Record, + ) {} +} diff --git a/backend/services/reporting-service/src/application/commands/generate-report/generate-report.handler.ts b/backend/services/reporting-service/src/application/commands/generate-report/generate-report.handler.ts new file mode 100644 index 00000000..f48bd78d --- /dev/null +++ b/backend/services/reporting-service/src/application/commands/generate-report/generate-report.handler.ts @@ -0,0 +1,223 @@ +import { Injectable, Inject, Logger, NotFoundException } from '@nestjs/common'; +import { GenerateReportCommand } from './generate-report.command'; +import { + IReportDefinitionRepository, + IReportSnapshotRepository, + REPORT_DEFINITION_REPOSITORY, + REPORT_SNAPSHOT_REPOSITORY, +} from '../../../domain/repositories'; +import { ReportGenerationDomainService } from '../../../domain/services'; +import { ReportSnapshot } from '../../../domain/aggregates/report-snapshot'; +import { DateRange, ReportType } from '../../../domain/value-objects'; +import { LeaderboardServiceClient } from '../../../infrastructure/external/leaderboard-service/leaderboard-service.client'; +import { PlantingServiceClient } from '../../../infrastructure/external/planting-service/planting-service.client'; +import { ReportCacheService } from '../../../infrastructure/redis/report-cache.service'; + +@Injectable() +export class GenerateReportHandler { + private readonly logger = new Logger(GenerateReportHandler.name); + + constructor( + @Inject(REPORT_DEFINITION_REPOSITORY) + private readonly definitionRepo: IReportDefinitionRepository, + @Inject(REPORT_SNAPSHOT_REPOSITORY) + private readonly snapshotRepo: IReportSnapshotRepository, + private readonly generationService: ReportGenerationDomainService, + private readonly leaderboardClient: LeaderboardServiceClient, + private readonly plantingClient: PlantingServiceClient, + private readonly cacheService: ReportCacheService, + ) {} + + async execute(command: GenerateReportCommand): Promise { + this.logger.log(`Generating report: ${command.reportCode}`); + + // Find report definition + const definition = await this.definitionRepo.findByCode(command.reportCode); + if (!definition) { + throw new NotFoundException( + `Report definition not found: ${command.reportCode}`, + ); + } + + // Create date range and period key + const dateRange = DateRange.create(command.startDate, command.endDate); + const periodKey = dateRange.toPeriodKey(command.reportPeriod); + + // Check cache first + const cached = await this.cacheService.getCachedSnapshot( + command.reportCode, + periodKey, + ); + + if (cached) { + this.logger.debug(`Using cached snapshot for ${command.reportCode}:${periodKey}`); + // Return cached snapshot if exists + const existingSnapshot = await this.snapshotRepo.findByCodeAndPeriodKey( + command.reportCode, + periodKey, + ); + if (existingSnapshot) { + return existingSnapshot; + } + } + + // Generate report data based on type + const { rows, summary, dataSources } = await this.generateReportData( + definition.reportType, + command, + ); + + // Create snapshot + const snapshot = this.generationService.generateSnapshot({ + reportType: definition.reportType, + reportCode: command.reportCode, + reportPeriod: command.reportPeriod, + periodKey, + rows, + summary, + dataSources, + filterParams: command.filters, + periodStartAt: command.startDate, + periodEndAt: command.endDate, + }); + + // Save snapshot + const savedSnapshot = await this.snapshotRepo.save(snapshot); + + // Update definition's last generated timestamp + definition.markGenerated(); + await this.definitionRepo.save(definition); + + // Cache the result + await this.cacheService.cacheSnapshot( + command.reportCode, + periodKey, + savedSnapshot.snapshotData.toJSON(), + ); + + this.logger.log( + `Report generated: ${command.reportCode}:${periodKey} with ${savedSnapshot.rowCount} rows`, + ); + + return savedSnapshot; + } + + private async generateReportData( + reportType: ReportType, + command: GenerateReportCommand, + ): Promise<{ + rows: any[]; + summary: Record; + dataSources: string[]; + }> { + switch (reportType) { + case ReportType.LEADERBOARD_REPORT: + return this.generateLeaderboardReport(command); + + case ReportType.PLANTING_REPORT: + return this.generatePlantingReport(command); + + case ReportType.REGIONAL_PLANTING_REPORT: + return this.generateRegionalPlantingReport(command); + + default: + return { + rows: [], + summary: {}, + dataSources: [], + }; + } + } + + private async generateLeaderboardReport( + command: GenerateReportCommand, + ): Promise<{ + rows: any[]; + summary: Record; + dataSources: string[]; + }> { + let leaderboardData; + + switch (command.reportPeriod) { + case 'DAILY': + leaderboardData = await this.leaderboardClient.getDailyLeaderboard( + command.startDate, + ); + break; + case 'WEEKLY': + leaderboardData = await this.leaderboardClient.getWeeklyLeaderboard(); + break; + case 'MONTHLY': + leaderboardData = await this.leaderboardClient.getMonthlyLeaderboard(); + break; + default: + leaderboardData = await this.leaderboardClient.getDailyLeaderboard( + command.startDate, + ); + } + + return { + rows: leaderboardData.entries, + summary: { + totalEntries: leaderboardData.entries.length, + period: leaderboardData.period, + type: leaderboardData.type, + }, + dataSources: ['leaderboard-service'], + }; + } + + private async generatePlantingReport( + command: GenerateReportCommand, + ): Promise<{ + rows: any[]; + summary: Record; + dataSources: string[]; + }> { + const stats = await this.plantingClient.getStatsForDateRange( + command.startDate, + command.endDate, + ); + + return { + rows: [stats], + summary: { + totalOrders: stats.totalOrders, + totalTrees: stats.totalTrees, + totalAmount: stats.totalAmount, + newUsers: stats.newUsers, + activeUsers: stats.activeUsers, + }, + dataSources: ['planting-service'], + }; + } + + private async generateRegionalPlantingReport( + command: GenerateReportCommand, + ): Promise<{ + rows: any[]; + summary: Record; + dataSources: string[]; + }> { + const provincesStats = await this.plantingClient.getAllProvincesStats( + command.startDate, + command.endDate, + ); + + const totalOrders = provincesStats.reduce( + (sum, s) => sum + s.totalOrders, + 0, + ); + const totalTrees = provincesStats.reduce((sum, s) => sum + s.totalTrees, 0); + + return { + rows: provincesStats, + summary: { + totalProvinces: provincesStats.length, + totalOrders, + totalTrees, + }, + dataSources: ['planting-service'], + }; + } +} diff --git a/backend/services/reporting-service/src/application/commands/index.ts b/backend/services/reporting-service/src/application/commands/index.ts new file mode 100644 index 00000000..5a4ab0e0 --- /dev/null +++ b/backend/services/reporting-service/src/application/commands/index.ts @@ -0,0 +1,4 @@ +export * from './generate-report/generate-report.command'; +export * from './generate-report/generate-report.handler'; +export * from './export-report/export-report.command'; +export * from './export-report/export-report.handler'; diff --git a/backend/services/reporting-service/src/application/queries/get-report-snapshot/get-report-snapshot.handler.ts b/backend/services/reporting-service/src/application/queries/get-report-snapshot/get-report-snapshot.handler.ts new file mode 100644 index 00000000..6f9601eb --- /dev/null +++ b/backend/services/reporting-service/src/application/queries/get-report-snapshot/get-report-snapshot.handler.ts @@ -0,0 +1,38 @@ +import { Injectable, Inject } from '@nestjs/common'; +import { + GetReportSnapshotQuery, + GetReportSnapshotByIdQuery, + GetLatestReportSnapshotQuery, +} from './get-report-snapshot.query'; +import { + IReportSnapshotRepository, + REPORT_SNAPSHOT_REPOSITORY, +} from '../../../domain/repositories'; +import { ReportSnapshot } from '../../../domain/aggregates/report-snapshot'; + +@Injectable() +export class GetReportSnapshotHandler { + constructor( + @Inject(REPORT_SNAPSHOT_REPOSITORY) + private readonly snapshotRepo: IReportSnapshotRepository, + ) {} + + async execute(query: GetReportSnapshotQuery): Promise { + return this.snapshotRepo.findByCodeAndPeriodKey( + query.reportCode, + query.periodKey, + ); + } + + async executeById( + query: GetReportSnapshotByIdQuery, + ): Promise { + return this.snapshotRepo.findById(query.id); + } + + async executeLatest( + query: GetLatestReportSnapshotQuery, + ): Promise { + return this.snapshotRepo.findLatestByCode(query.reportCode); + } +} diff --git a/backend/services/reporting-service/src/application/queries/get-report-snapshot/get-report-snapshot.query.ts b/backend/services/reporting-service/src/application/queries/get-report-snapshot/get-report-snapshot.query.ts new file mode 100644 index 00000000..3fbb7781 --- /dev/null +++ b/backend/services/reporting-service/src/application/queries/get-report-snapshot/get-report-snapshot.query.ts @@ -0,0 +1,14 @@ +export class GetReportSnapshotQuery { + constructor( + public readonly reportCode: string, + public readonly periodKey: string, + ) {} +} + +export class GetReportSnapshotByIdQuery { + constructor(public readonly id: bigint) {} +} + +export class GetLatestReportSnapshotQuery { + constructor(public readonly reportCode: string) {} +} diff --git a/backend/services/reporting-service/src/application/queries/index.ts b/backend/services/reporting-service/src/application/queries/index.ts new file mode 100644 index 00000000..4b9cbca0 --- /dev/null +++ b/backend/services/reporting-service/src/application/queries/index.ts @@ -0,0 +1,2 @@ +export * from './get-report-snapshot/get-report-snapshot.query'; +export * from './get-report-snapshot/get-report-snapshot.handler'; diff --git a/backend/services/reporting-service/src/application/schedulers/report-generation.scheduler.ts b/backend/services/reporting-service/src/application/schedulers/report-generation.scheduler.ts new file mode 100644 index 00000000..ea4179e9 --- /dev/null +++ b/backend/services/reporting-service/src/application/schedulers/report-generation.scheduler.ts @@ -0,0 +1,170 @@ +import { Injectable, Inject, Logger } from '@nestjs/common'; +import { Cron, CronExpression } from '@nestjs/schedule'; +import { + IReportDefinitionRepository, + REPORT_DEFINITION_REPOSITORY, +} from '../../domain/repositories'; +import { GenerateReportHandler } from '../commands/generate-report/generate-report.handler'; +import { GenerateReportCommand } from '../commands/generate-report/generate-report.command'; +import { DateRange, ReportPeriod } from '../../domain/value-objects'; + +@Injectable() +export class ReportGenerationScheduler { + private readonly logger = new Logger(ReportGenerationScheduler.name); + + constructor( + @Inject(REPORT_DEFINITION_REPOSITORY) + private readonly definitionRepo: IReportDefinitionRepository, + private readonly generateReportHandler: GenerateReportHandler, + ) {} + + @Cron(CronExpression.EVERY_DAY_AT_1AM) + async generateDailyReports(): Promise { + this.logger.log('Starting daily report generation...'); + + try { + const definitions = await this.definitionRepo.findScheduled(); + const yesterday = this.getYesterday(); + const dateRange = DateRange.create(yesterday.start, yesterday.end); + + for (const definition of definitions) { + try { + const command = new GenerateReportCommand( + definition.reportCode, + ReportPeriod.DAILY, + yesterday.start, + yesterday.end, + ); + + await this.generateReportHandler.execute(command); + this.logger.log(`Daily report generated: ${definition.reportCode}`); + } catch (error) { + this.logger.error( + `Failed to generate daily report: ${definition.reportCode}`, + error, + ); + } + } + + this.logger.log('Daily report generation completed'); + } catch (error) { + this.logger.error('Daily report generation failed', error); + } + } + + @Cron(CronExpression.EVERY_WEEK) + async generateWeeklyReports(): Promise { + this.logger.log('Starting weekly report generation...'); + + try { + const definitions = await this.definitionRepo.findScheduled(); + const lastWeek = this.getLastWeek(); + + for (const definition of definitions) { + try { + const command = new GenerateReportCommand( + definition.reportCode, + ReportPeriod.WEEKLY, + lastWeek.start, + lastWeek.end, + ); + + await this.generateReportHandler.execute(command); + this.logger.log(`Weekly report generated: ${definition.reportCode}`); + } catch (error) { + this.logger.error( + `Failed to generate weekly report: ${definition.reportCode}`, + error, + ); + } + } + + this.logger.log('Weekly report generation completed'); + } catch (error) { + this.logger.error('Weekly report generation failed', error); + } + } + + @Cron('0 0 1 * *') // First day of month at midnight + async generateMonthlyReports(): Promise { + this.logger.log('Starting monthly report generation...'); + + try { + const definitions = await this.definitionRepo.findScheduled(); + const lastMonth = this.getLastMonth(); + + for (const definition of definitions) { + try { + const command = new GenerateReportCommand( + definition.reportCode, + ReportPeriod.MONTHLY, + lastMonth.start, + lastMonth.end, + ); + + await this.generateReportHandler.execute(command); + this.logger.log(`Monthly report generated: ${definition.reportCode}`); + } catch (error) { + this.logger.error( + `Failed to generate monthly report: ${definition.reportCode}`, + error, + ); + } + } + + this.logger.log('Monthly report generation completed'); + } catch (error) { + this.logger.error('Monthly report generation failed', error); + } + } + + private getYesterday(): { start: Date; end: Date } { + const now = new Date(); + const yesterday = new Date(now); + yesterday.setDate(yesterday.getDate() - 1); + + const start = new Date( + yesterday.getFullYear(), + yesterday.getMonth(), + yesterday.getDate(), + 0, + 0, + 0, + ); + const end = new Date( + yesterday.getFullYear(), + yesterday.getMonth(), + yesterday.getDate(), + 23, + 59, + 59, + 999, + ); + + return { start, end }; + } + + private getLastWeek(): { start: Date; end: Date } { + const now = new Date(); + const dayOfWeek = now.getDay(); + const diffToLastMonday = dayOfWeek === 0 ? -13 : -6 - dayOfWeek; + + const start = new Date(now); + start.setDate(now.getDate() + diffToLastMonday); + start.setHours(0, 0, 0, 0); + + const end = new Date(start); + end.setDate(start.getDate() + 6); + end.setHours(23, 59, 59, 999); + + return { start, end }; + } + + private getLastMonth(): { start: Date; end: Date } { + const now = new Date(); + const start = new Date(now.getFullYear(), now.getMonth() - 1, 1, 0, 0, 0); + const end = new Date(now.getFullYear(), now.getMonth(), 0, 23, 59, 59, 999); + + return { start, end }; + } +} diff --git a/backend/services/reporting-service/src/application/services/reporting-application.service.ts b/backend/services/reporting-service/src/application/services/reporting-application.service.ts new file mode 100644 index 00000000..1d4d10fc --- /dev/null +++ b/backend/services/reporting-service/src/application/services/reporting-application.service.ts @@ -0,0 +1,107 @@ +import { Injectable, Inject, Logger } from '@nestjs/common'; +import { + IReportDefinitionRepository, + IReportSnapshotRepository, + REPORT_DEFINITION_REPOSITORY, + REPORT_SNAPSHOT_REPOSITORY, +} from '../../domain/repositories'; +import { ReportDefinition } from '../../domain/aggregates/report-definition'; +import { ReportSnapshot } from '../../domain/aggregates/report-snapshot'; +import { ReportType, ReportPeriod } from '../../domain/value-objects'; + +export interface ReportDefinitionDto { + id: string; + reportType: string; + reportName: string; + reportCode: string; + description: string; + parameters: Record; + scheduleCron: string | null; + scheduleEnabled: boolean; + outputFormats: string[]; + isActive: boolean; + lastGeneratedAt: Date | null; +} + +export interface ReportSnapshotDto { + id: string; + reportType: string; + reportCode: string; + reportPeriod: string; + periodKey: string; + rowCount: number; + summary: Record; + periodStartAt: Date; + periodEndAt: Date; + generatedAt: Date; +} + +@Injectable() +export class ReportingApplicationService { + private readonly logger = new Logger(ReportingApplicationService.name); + + constructor( + @Inject(REPORT_DEFINITION_REPOSITORY) + private readonly definitionRepo: IReportDefinitionRepository, + @Inject(REPORT_SNAPSHOT_REPOSITORY) + private readonly snapshotRepo: IReportSnapshotRepository, + ) {} + + async getReportDefinitions(): Promise { + const definitions = await this.definitionRepo.findActive(); + return definitions.map(this.mapDefinitionToDto); + } + + async getReportDefinitionByCode( + code: string, + ): Promise { + const definition = await this.definitionRepo.findByCode(code); + return definition ? this.mapDefinitionToDto(definition) : null; + } + + async getReportSnapshots(reportCode: string): Promise { + const snapshots = await this.snapshotRepo.findByCode(reportCode); + return snapshots.map(this.mapSnapshotToDto); + } + + async getLatestSnapshot(reportCode: string): Promise { + const snapshot = await this.snapshotRepo.findLatestByCode(reportCode); + return snapshot ? this.mapSnapshotToDto(snapshot) : null; + } + + async getSnapshotsByPeriod(period: ReportPeriod): Promise { + const snapshots = await this.snapshotRepo.findByPeriod(period); + return snapshots.map(this.mapSnapshotToDto); + } + + private mapDefinitionToDto(definition: ReportDefinition): ReportDefinitionDto { + return { + id: definition.id?.toString() || '', + reportType: definition.reportType, + reportName: definition.reportName, + reportCode: definition.reportCode, + description: definition.description, + parameters: definition.parameters, + scheduleCron: definition.schedule?.cronExpression || null, + scheduleEnabled: definition.schedule?.enabled || false, + outputFormats: definition.outputFormats, + isActive: definition.isActive, + lastGeneratedAt: definition.lastGeneratedAt, + }; + } + + private mapSnapshotToDto(snapshot: ReportSnapshot): ReportSnapshotDto { + return { + id: snapshot.id?.toString() || '', + reportType: snapshot.reportType, + reportCode: snapshot.reportCode, + reportPeriod: snapshot.reportPeriod, + periodKey: snapshot.periodKey, + rowCount: snapshot.rowCount, + summary: snapshot.snapshotData.summary, + periodStartAt: snapshot.periodStartAt, + periodEndAt: snapshot.periodEndAt, + generatedAt: snapshot.generatedAt, + }; + } +} diff --git a/backend/services/reporting-service/src/config/app.config.ts b/backend/services/reporting-service/src/config/app.config.ts new file mode 100644 index 00000000..543d9160 --- /dev/null +++ b/backend/services/reporting-service/src/config/app.config.ts @@ -0,0 +1,7 @@ +import { registerAs } from '@nestjs/config'; + +export default registerAs('app', () => ({ + name: process.env.APP_NAME || 'reporting-service', + port: parseInt(process.env.PORT || '3008', 10), + env: process.env.NODE_ENV || 'development', +})); diff --git a/backend/services/reporting-service/src/config/database.config.ts b/backend/services/reporting-service/src/config/database.config.ts new file mode 100644 index 00000000..4bc50472 --- /dev/null +++ b/backend/services/reporting-service/src/config/database.config.ts @@ -0,0 +1,5 @@ +import { registerAs } from '@nestjs/config'; + +export default registerAs('database', () => ({ + url: process.env.DATABASE_URL, +})); diff --git a/backend/services/reporting-service/src/config/index.ts b/backend/services/reporting-service/src/config/index.ts new file mode 100644 index 00000000..dd269b57 --- /dev/null +++ b/backend/services/reporting-service/src/config/index.ts @@ -0,0 +1,4 @@ +export { default as appConfig } from './app.config'; +export { default as databaseConfig } from './database.config'; +export { default as jwtConfig } from './jwt.config'; +export { default as redisConfig } from './redis.config'; diff --git a/backend/services/reporting-service/src/config/jwt.config.ts b/backend/services/reporting-service/src/config/jwt.config.ts new file mode 100644 index 00000000..5a506a71 --- /dev/null +++ b/backend/services/reporting-service/src/config/jwt.config.ts @@ -0,0 +1,6 @@ +import { registerAs } from '@nestjs/config'; + +export default registerAs('jwt', () => ({ + secret: process.env.JWT_SECRET || 'default-secret-change-in-production', + accessExpiresIn: process.env.JWT_ACCESS_EXPIRES_IN || '2h', +})); diff --git a/backend/services/reporting-service/src/config/redis.config.ts b/backend/services/reporting-service/src/config/redis.config.ts new file mode 100644 index 00000000..8de085fc --- /dev/null +++ b/backend/services/reporting-service/src/config/redis.config.ts @@ -0,0 +1,7 @@ +import { registerAs } from '@nestjs/config'; + +export default registerAs('redis', () => ({ + host: process.env.REDIS_HOST || 'localhost', + port: parseInt(process.env.REDIS_PORT || '6379', 10), + password: process.env.REDIS_PASSWORD || undefined, +})); diff --git a/backend/services/reporting-service/src/domain/aggregates/report-definition/index.ts b/backend/services/reporting-service/src/domain/aggregates/report-definition/index.ts new file mode 100644 index 00000000..7efdd47d --- /dev/null +++ b/backend/services/reporting-service/src/domain/aggregates/report-definition/index.ts @@ -0,0 +1 @@ +export * from './report-definition.aggregate'; diff --git a/backend/services/reporting-service/src/domain/aggregates/report-definition/report-definition.aggregate.ts b/backend/services/reporting-service/src/domain/aggregates/report-definition/report-definition.aggregate.ts new file mode 100644 index 00000000..d419329e --- /dev/null +++ b/backend/services/reporting-service/src/domain/aggregates/report-definition/report-definition.aggregate.ts @@ -0,0 +1,233 @@ +import { DomainEvent } from '../../events/domain-event.base'; +import { ReportType } from '../../value-objects/report-type.enum'; +import { ReportSchedule } from '../../value-objects/report-schedule.vo'; +import { OutputFormat } from '../../value-objects/output-format.enum'; + +export class ReportDefinition { + private _id: bigint | null = null; + private readonly _reportType: ReportType; + private _reportName: string; + private readonly _reportCode: string; + private _description: string; + private _parameters: Record; + private _schedule: ReportSchedule | null; + private _outputFormats: OutputFormat[]; + private _isActive: boolean; + private readonly _createdAt: Date; + private _lastGeneratedAt: Date | null; + + private _domainEvents: DomainEvent[] = []; + + private constructor( + reportType: ReportType, + reportName: string, + reportCode: string, + description: string, + parameters: Record, + schedule: ReportSchedule | null, + outputFormats: OutputFormat[], + isActive: boolean, + createdAt: Date, + lastGeneratedAt: Date | null, + ) { + this._reportType = reportType; + this._reportName = reportName; + this._reportCode = reportCode; + this._description = description; + this._parameters = parameters; + this._schedule = schedule; + this._outputFormats = outputFormats; + this._isActive = isActive; + this._createdAt = createdAt; + this._lastGeneratedAt = lastGeneratedAt; + + this.validateInvariants(); + } + + static create(params: { + reportType: ReportType; + reportName: string; + reportCode: string; + description?: string; + parameters?: Record; + schedule?: ReportSchedule; + outputFormats?: OutputFormat[]; + }): ReportDefinition { + const outputFormats = + params.outputFormats && params.outputFormats.length > 0 + ? params.outputFormats + : [OutputFormat.EXCEL]; + + return new ReportDefinition( + params.reportType, + params.reportName, + params.reportCode, + params.description || '', + params.parameters || {}, + params.schedule || null, + outputFormats, + true, + new Date(), + null, + ); + } + + static reconstitute(params: { + id: bigint; + reportType: ReportType; + reportName: string; + reportCode: string; + description: string; + parameters: Record; + schedule: ReportSchedule | null; + outputFormats: OutputFormat[]; + isActive: boolean; + createdAt: Date; + lastGeneratedAt: Date | null; + }): ReportDefinition { + const definition = new ReportDefinition( + params.reportType, + params.reportName, + params.reportCode, + params.description, + params.parameters, + params.schedule, + params.outputFormats, + params.isActive, + params.createdAt, + params.lastGeneratedAt, + ); + definition._id = params.id; + return definition; + } + + private validateInvariants(): void { + if (this._outputFormats.length === 0) { + throw new Error('报表定义至少需要支持一种输出格式'); + } + + if (this._schedule?.enabled && !this._schedule.cronExpression) { + throw new Error('启用调度时必须有有效的 cron 表达式'); + } + } + + // Getters + get id(): bigint | null { + return this._id; + } + + get reportType(): ReportType { + return this._reportType; + } + + get reportName(): string { + return this._reportName; + } + + get reportCode(): string { + return this._reportCode; + } + + get description(): string { + return this._description; + } + + get parameters(): Record { + return { ...this._parameters }; + } + + get schedule(): ReportSchedule | null { + return this._schedule; + } + + get outputFormats(): OutputFormat[] { + return [...this._outputFormats]; + } + + get isActive(): boolean { + return this._isActive; + } + + get createdAt(): Date { + return this._createdAt; + } + + get lastGeneratedAt(): Date | null { + return this._lastGeneratedAt; + } + + get domainEvents(): DomainEvent[] { + return [...this._domainEvents]; + } + + // Commands + updateName(name: string): void { + this._reportName = name; + } + + updateDescription(description: string): void { + this._description = description; + } + + updateParameters(parameters: Record): void { + this._parameters = parameters; + } + + setSchedule(schedule: ReportSchedule): void { + this._schedule = schedule; + this.validateInvariants(); + } + + enableSchedule(): void { + if (this._schedule) { + this._schedule = this._schedule.enable(); + } + } + + disableSchedule(): void { + if (this._schedule) { + this._schedule = this._schedule.disable(); + } + } + + addOutputFormat(format: OutputFormat): void { + if (!this._outputFormats.includes(format)) { + this._outputFormats.push(format); + } + } + + removeOutputFormat(format: OutputFormat): void { + if (this._outputFormats.length <= 1) { + throw new Error('报表定义至少需要支持一种输出格式'); + } + this._outputFormats = this._outputFormats.filter((f) => f !== format); + } + + activate(): void { + this._isActive = true; + } + + deactivate(): void { + this._isActive = false; + } + + markGenerated(): void { + this._lastGeneratedAt = new Date(); + } + + supportsFormat(format: OutputFormat): boolean { + return this._outputFormats.includes(format); + } + + isScheduled(): boolean { + return this._schedule !== null && this._schedule.enabled; + } + + clearDomainEvents(): void { + this._domainEvents = []; + } + + protected addDomainEvent(event: DomainEvent): void { + this._domainEvents.push(event); + } +} diff --git a/backend/services/reporting-service/src/domain/aggregates/report-definition/report-definition.spec.ts b/backend/services/reporting-service/src/domain/aggregates/report-definition/report-definition.spec.ts new file mode 100644 index 00000000..74c09eeb --- /dev/null +++ b/backend/services/reporting-service/src/domain/aggregates/report-definition/report-definition.spec.ts @@ -0,0 +1,127 @@ +import { ReportDefinition } from './report-definition.aggregate'; +import { ReportType } from '../../value-objects/report-type.enum'; +import { OutputFormat } from '../../value-objects/output-format.enum'; +import { ReportSchedule } from '../../value-objects/report-schedule.vo'; + +describe('ReportDefinition Aggregate', () => { + describe('create', () => { + it('should create a report definition with required fields', () => { + const definition = ReportDefinition.create({ + reportType: ReportType.LEADERBOARD_REPORT, + reportName: 'Test Report', + reportCode: 'TEST_001', + }); + + expect(definition.reportType).toBe(ReportType.LEADERBOARD_REPORT); + expect(definition.reportName).toBe('Test Report'); + expect(definition.reportCode).toBe('TEST_001'); + expect(definition.isActive).toBe(true); + expect(definition.outputFormats).toContain(OutputFormat.EXCEL); + }); + + it('should create a report definition with custom output formats', () => { + const definition = ReportDefinition.create({ + reportType: ReportType.PLANTING_REPORT, + reportName: 'Planting Report', + reportCode: 'PLANTING_001', + outputFormats: [OutputFormat.CSV, OutputFormat.PDF], + }); + + expect(definition.outputFormats).toHaveLength(2); + expect(definition.outputFormats).toContain(OutputFormat.CSV); + expect(definition.outputFormats).toContain(OutputFormat.PDF); + }); + + it('should create a report definition with schedule', () => { + const schedule = ReportSchedule.daily(1, 0); + const definition = ReportDefinition.create({ + reportType: ReportType.PLANTING_REPORT, + reportName: 'Daily Planting Report', + reportCode: 'PLANTING_DAILY', + schedule, + }); + + expect(definition.schedule).toBeDefined(); + expect(definition.schedule?.enabled).toBe(true); + expect(definition.isScheduled()).toBe(true); + }); + }); + + describe('invariants', () => { + it('should require at least one output format', () => { + expect(() => { + const def = ReportDefinition.create({ + reportType: ReportType.LEADERBOARD_REPORT, + reportName: 'Test', + reportCode: 'TEST', + outputFormats: [OutputFormat.EXCEL], + }); + def.removeOutputFormat(OutputFormat.EXCEL); + }).toThrow('报表定义至少需要支持一种输出格式'); + }); + }); + + describe('commands', () => { + it('should update name', () => { + const definition = ReportDefinition.create({ + reportType: ReportType.LEADERBOARD_REPORT, + reportName: 'Original Name', + reportCode: 'TEST', + }); + + definition.updateName('New Name'); + expect(definition.reportName).toBe('New Name'); + }); + + it('should add output format', () => { + const definition = ReportDefinition.create({ + reportType: ReportType.LEADERBOARD_REPORT, + reportName: 'Test', + reportCode: 'TEST', + outputFormats: [OutputFormat.EXCEL], + }); + + definition.addOutputFormat(OutputFormat.CSV); + expect(definition.outputFormats).toContain(OutputFormat.CSV); + }); + + it('should not add duplicate output format', () => { + const definition = ReportDefinition.create({ + reportType: ReportType.LEADERBOARD_REPORT, + reportName: 'Test', + reportCode: 'TEST', + outputFormats: [OutputFormat.EXCEL], + }); + + definition.addOutputFormat(OutputFormat.EXCEL); + expect(definition.outputFormats.filter(f => f === OutputFormat.EXCEL)).toHaveLength(1); + }); + + it('should activate and deactivate', () => { + const definition = ReportDefinition.create({ + reportType: ReportType.LEADERBOARD_REPORT, + reportName: 'Test', + reportCode: 'TEST', + }); + + definition.deactivate(); + expect(definition.isActive).toBe(false); + + definition.activate(); + expect(definition.isActive).toBe(true); + }); + + it('should mark as generated', () => { + const definition = ReportDefinition.create({ + reportType: ReportType.LEADERBOARD_REPORT, + reportName: 'Test', + reportCode: 'TEST', + }); + + expect(definition.lastGeneratedAt).toBeNull(); + + definition.markGenerated(); + expect(definition.lastGeneratedAt).toBeInstanceOf(Date); + }); + }); +}); diff --git a/backend/services/reporting-service/src/domain/aggregates/report-snapshot/index.ts b/backend/services/reporting-service/src/domain/aggregates/report-snapshot/index.ts new file mode 100644 index 00000000..f3ffd53e --- /dev/null +++ b/backend/services/reporting-service/src/domain/aggregates/report-snapshot/index.ts @@ -0,0 +1 @@ +export * from './report-snapshot.aggregate'; diff --git a/backend/services/reporting-service/src/domain/aggregates/report-snapshot/report-snapshot.aggregate.ts b/backend/services/reporting-service/src/domain/aggregates/report-snapshot/report-snapshot.aggregate.ts new file mode 100644 index 00000000..3230eca7 --- /dev/null +++ b/backend/services/reporting-service/src/domain/aggregates/report-snapshot/report-snapshot.aggregate.ts @@ -0,0 +1,207 @@ +import { DomainEvent } from '../../events/domain-event.base'; +import { SnapshotCreatedEvent } from '../../events/snapshot-created.event'; +import { ReportType } from '../../value-objects/report-type.enum'; +import { ReportPeriod } from '../../value-objects/report-period.enum'; +import { SnapshotData } from '../../value-objects/snapshot-data.vo'; +import { DataSource } from '../../value-objects/data-source.vo'; + +export class ReportSnapshot { + private _id: bigint | null = null; + private readonly _reportType: ReportType; + private readonly _reportCode: string; + private readonly _reportPeriod: ReportPeriod; + private readonly _periodKey: string; + private _snapshotData: SnapshotData; + private _dataSource: DataSource; + private _filterParams: Record | null; + private readonly _periodStartAt: Date; + private readonly _periodEndAt: Date; + private readonly _generatedAt: Date; + private _expiresAt: Date | null; + + private _domainEvents: DomainEvent[] = []; + + private constructor( + reportType: ReportType, + reportCode: string, + reportPeriod: ReportPeriod, + periodKey: string, + snapshotData: SnapshotData, + dataSource: DataSource, + filterParams: Record | null, + periodStartAt: Date, + periodEndAt: Date, + generatedAt: Date, + expiresAt: Date | null, + ) { + this._reportType = reportType; + this._reportCode = reportCode; + this._reportPeriod = reportPeriod; + this._periodKey = periodKey; + this._snapshotData = snapshotData; + this._dataSource = dataSource; + this._filterParams = filterParams; + this._periodStartAt = periodStartAt; + this._periodEndAt = periodEndAt; + this._generatedAt = generatedAt; + this._expiresAt = expiresAt; + } + + static create(params: { + reportType: ReportType; + reportCode: string; + reportPeriod: ReportPeriod; + periodKey: string; + snapshotData: SnapshotData; + dataSource: DataSource; + filterParams?: Record; + periodStartAt: Date; + periodEndAt: Date; + expiresAt?: Date; + }): ReportSnapshot { + const snapshot = new ReportSnapshot( + params.reportType, + params.reportCode, + params.reportPeriod, + params.periodKey, + params.snapshotData, + params.dataSource, + params.filterParams || null, + params.periodStartAt, + params.periodEndAt, + new Date(), + params.expiresAt || null, + ); + + snapshot.addDomainEvent( + new SnapshotCreatedEvent({ + snapshotId: params.periodKey, + reportType: params.reportType, + reportCode: params.reportCode, + reportPeriod: params.reportPeriod, + periodKey: params.periodKey, + rowCount: params.snapshotData.getRowCount(), + periodStartAt: params.periodStartAt, + periodEndAt: params.periodEndAt, + }), + ); + + return snapshot; + } + + static reconstitute(params: { + id: bigint; + reportType: ReportType; + reportCode: string; + reportPeriod: ReportPeriod; + periodKey: string; + snapshotData: SnapshotData; + dataSource: DataSource; + filterParams: Record | null; + periodStartAt: Date; + periodEndAt: Date; + generatedAt: Date; + expiresAt: Date | null; + }): ReportSnapshot { + const snapshot = new ReportSnapshot( + params.reportType, + params.reportCode, + params.reportPeriod, + params.periodKey, + params.snapshotData, + params.dataSource, + params.filterParams, + params.periodStartAt, + params.periodEndAt, + params.generatedAt, + params.expiresAt, + ); + snapshot._id = params.id; + return snapshot; + } + + // Getters + get id(): bigint | null { + return this._id; + } + + get reportType(): ReportType { + return this._reportType; + } + + get reportCode(): string { + return this._reportCode; + } + + get reportPeriod(): ReportPeriod { + return this._reportPeriod; + } + + get periodKey(): string { + return this._periodKey; + } + + get snapshotData(): SnapshotData { + return this._snapshotData; + } + + get dataSource(): DataSource { + return this._dataSource; + } + + get filterParams(): Record | null { + return this._filterParams ? { ...this._filterParams } : null; + } + + get periodStartAt(): Date { + return this._periodStartAt; + } + + get periodEndAt(): Date { + return this._periodEndAt; + } + + get generatedAt(): Date { + return this._generatedAt; + } + + get expiresAt(): Date | null { + return this._expiresAt; + } + + get rowCount(): number { + return this._snapshotData.getRowCount(); + } + + get domainEvents(): DomainEvent[] { + return [...this._domainEvents]; + } + + // Commands + updateSnapshotData(data: SnapshotData): void { + this._snapshotData = data; + } + + setExpiration(expiresAt: Date): void { + this._expiresAt = expiresAt; + } + + isExpired(): boolean { + if (!this._expiresAt) { + return false; + } + return new Date() > this._expiresAt; + } + + isFresh(maxAgeSeconds: number): boolean { + return this._dataSource.isFresh(maxAgeSeconds); + } + + clearDomainEvents(): void { + this._domainEvents = []; + } + + protected addDomainEvent(event: DomainEvent): void { + this._domainEvents.push(event); + } +} diff --git a/backend/services/reporting-service/src/domain/domain.module.ts b/backend/services/reporting-service/src/domain/domain.module.ts new file mode 100644 index 00000000..7ccd58cf --- /dev/null +++ b/backend/services/reporting-service/src/domain/domain.module.ts @@ -0,0 +1,8 @@ +import { Module } from '@nestjs/common'; +import { ReportGenerationDomainService } from './services/report-generation.service'; + +@Module({ + providers: [ReportGenerationDomainService], + exports: [ReportGenerationDomainService], +}) +export class DomainModule {} diff --git a/backend/services/reporting-service/src/domain/entities/analytics-metric.entity.ts b/backend/services/reporting-service/src/domain/entities/analytics-metric.entity.ts new file mode 100644 index 00000000..992e77e0 --- /dev/null +++ b/backend/services/reporting-service/src/domain/entities/analytics-metric.entity.ts @@ -0,0 +1,135 @@ +import { Decimal } from '@prisma/client/runtime/library'; + +export class AnalyticsMetric { + private _id: bigint | null = null; + private readonly _metricType: string; + private readonly _metricCode: string; + private readonly _dimensionTime: Date | null; + private readonly _dimensionRegion: string | null; + private readonly _dimensionUserType: string | null; + private readonly _dimensionRightType: string | null; + private _metricValue: Decimal; + private _metricData: Record | null; + private readonly _calculatedAt: Date; + + private constructor( + metricType: string, + metricCode: string, + dimensionTime: Date | null, + dimensionRegion: string | null, + dimensionUserType: string | null, + dimensionRightType: string | null, + metricValue: Decimal, + metricData: Record | null, + calculatedAt: Date, + ) { + this._metricType = metricType; + this._metricCode = metricCode; + this._dimensionTime = dimensionTime; + this._dimensionRegion = dimensionRegion; + this._dimensionUserType = dimensionUserType; + this._dimensionRightType = dimensionRightType; + this._metricValue = metricValue; + this._metricData = metricData; + this._calculatedAt = calculatedAt; + } + + static create(params: { + metricType: string; + metricCode: string; + dimensionTime?: Date; + dimensionRegion?: string; + dimensionUserType?: string; + dimensionRightType?: string; + metricValue: Decimal; + metricData?: Record; + }): AnalyticsMetric { + return new AnalyticsMetric( + params.metricType, + params.metricCode, + params.dimensionTime || null, + params.dimensionRegion || null, + params.dimensionUserType || null, + params.dimensionRightType || null, + params.metricValue, + params.metricData || null, + new Date(), + ); + } + + static reconstitute(params: { + id: bigint; + metricType: string; + metricCode: string; + dimensionTime: Date | null; + dimensionRegion: string | null; + dimensionUserType: string | null; + dimensionRightType: string | null; + metricValue: Decimal; + metricData: Record | null; + calculatedAt: Date; + }): AnalyticsMetric { + const metric = new AnalyticsMetric( + params.metricType, + params.metricCode, + params.dimensionTime, + params.dimensionRegion, + params.dimensionUserType, + params.dimensionRightType, + params.metricValue, + params.metricData, + params.calculatedAt, + ); + metric._id = params.id; + return metric; + } + + // Getters + get id(): bigint | null { + return this._id; + } + + get metricType(): string { + return this._metricType; + } + + get metricCode(): string { + return this._metricCode; + } + + get dimensionTime(): Date | null { + return this._dimensionTime; + } + + get dimensionRegion(): string | null { + return this._dimensionRegion; + } + + get dimensionUserType(): string | null { + return this._dimensionUserType; + } + + get dimensionRightType(): string | null { + return this._dimensionRightType; + } + + get metricValue(): Decimal { + return this._metricValue; + } + + get metricData(): Record | null { + return this._metricData ? { ...this._metricData } : null; + } + + get calculatedAt(): Date { + return this._calculatedAt; + } + + // Commands + updateValue(value: Decimal, data?: Record): void { + this._metricValue = value; + if (data) { + this._metricData = data; + } + } +} diff --git a/backend/services/reporting-service/src/domain/entities/index.ts b/backend/services/reporting-service/src/domain/entities/index.ts new file mode 100644 index 00000000..0fb4ec9a --- /dev/null +++ b/backend/services/reporting-service/src/domain/entities/index.ts @@ -0,0 +1,2 @@ +export * from './report-file.entity'; +export * from './analytics-metric.entity'; diff --git a/backend/services/reporting-service/src/domain/entities/report-file.entity.ts b/backend/services/reporting-service/src/domain/entities/report-file.entity.ts new file mode 100644 index 00000000..3c739073 --- /dev/null +++ b/backend/services/reporting-service/src/domain/entities/report-file.entity.ts @@ -0,0 +1,168 @@ +import { OutputFormat } from '../value-objects/output-format.enum'; + +export class ReportFile { + private _id: bigint | null = null; + private readonly _snapshotId: bigint; + private readonly _fileName: string; + private readonly _filePath: string; + private _fileUrl: string | null; + private readonly _fileSize: bigint; + private readonly _fileFormat: OutputFormat; + private readonly _mimeType: string; + private _downloadCount: number; + private _lastDownloadAt: Date | null; + private readonly _createdAt: Date; + private _expiresAt: Date | null; + + private constructor( + snapshotId: bigint, + fileName: string, + filePath: string, + fileUrl: string | null, + fileSize: bigint, + fileFormat: OutputFormat, + mimeType: string, + downloadCount: number, + lastDownloadAt: Date | null, + createdAt: Date, + expiresAt: Date | null, + ) { + this._snapshotId = snapshotId; + this._fileName = fileName; + this._filePath = filePath; + this._fileUrl = fileUrl; + this._fileSize = fileSize; + this._fileFormat = fileFormat; + this._mimeType = mimeType; + this._downloadCount = downloadCount; + this._lastDownloadAt = lastDownloadAt; + this._createdAt = createdAt; + this._expiresAt = expiresAt; + } + + static create(params: { + snapshotId: bigint; + fileName: string; + filePath: string; + fileUrl?: string; + fileSize: bigint; + fileFormat: OutputFormat; + mimeType: string; + expiresAt?: Date; + }): ReportFile { + return new ReportFile( + params.snapshotId, + params.fileName, + params.filePath, + params.fileUrl || null, + params.fileSize, + params.fileFormat, + params.mimeType, + 0, + null, + new Date(), + params.expiresAt || null, + ); + } + + static reconstitute(params: { + id: bigint; + snapshotId: bigint; + fileName: string; + filePath: string; + fileUrl: string | null; + fileSize: bigint; + fileFormat: OutputFormat; + mimeType: string; + downloadCount: number; + lastDownloadAt: Date | null; + createdAt: Date; + expiresAt: Date | null; + }): ReportFile { + const file = new ReportFile( + params.snapshotId, + params.fileName, + params.filePath, + params.fileUrl, + params.fileSize, + params.fileFormat, + params.mimeType, + params.downloadCount, + params.lastDownloadAt, + params.createdAt, + params.expiresAt, + ); + file._id = params.id; + return file; + } + + // Getters + get id(): bigint | null { + return this._id; + } + + get snapshotId(): bigint { + return this._snapshotId; + } + + get fileName(): string { + return this._fileName; + } + + get filePath(): string { + return this._filePath; + } + + get fileUrl(): string | null { + return this._fileUrl; + } + + get fileSize(): bigint { + return this._fileSize; + } + + get fileFormat(): OutputFormat { + return this._fileFormat; + } + + get mimeType(): string { + return this._mimeType; + } + + get downloadCount(): number { + return this._downloadCount; + } + + get lastDownloadAt(): Date | null { + return this._lastDownloadAt; + } + + get createdAt(): Date { + return this._createdAt; + } + + get expiresAt(): Date | null { + return this._expiresAt; + } + + // Commands + recordDownload(): void { + this._downloadCount++; + this._lastDownloadAt = new Date(); + } + + setFileUrl(url: string): void { + this._fileUrl = url; + } + + setExpiration(expiresAt: Date): void { + this._expiresAt = expiresAt; + } + + isExpired(): boolean { + if (!this._expiresAt) { + return false; + } + return new Date() > this._expiresAt; + } +} diff --git a/backend/services/reporting-service/src/domain/events/domain-event.base.ts b/backend/services/reporting-service/src/domain/events/domain-event.base.ts new file mode 100644 index 00000000..ce09b9b6 --- /dev/null +++ b/backend/services/reporting-service/src/domain/events/domain-event.base.ts @@ -0,0 +1,18 @@ +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; +} diff --git a/backend/services/reporting-service/src/domain/events/index.ts b/backend/services/reporting-service/src/domain/events/index.ts new file mode 100644 index 00000000..960397b3 --- /dev/null +++ b/backend/services/reporting-service/src/domain/events/index.ts @@ -0,0 +1,4 @@ +export * from './domain-event.base'; +export * from './report-generated.event'; +export * from './report-exported.event'; +export * from './snapshot-created.event'; diff --git a/backend/services/reporting-service/src/domain/events/report-exported.event.ts b/backend/services/reporting-service/src/domain/events/report-exported.event.ts new file mode 100644 index 00000000..da10dfe1 --- /dev/null +++ b/backend/services/reporting-service/src/domain/events/report-exported.event.ts @@ -0,0 +1,33 @@ +import { DomainEvent } from './domain-event.base'; +import { OutputFormat } from '../value-objects/output-format.enum'; + +export interface ReportExportedPayload { + fileId: string; + snapshotId: string; + format: OutputFormat; + fileName: string; + fileSize: number; + exportedAt: Date; +} + +export class ReportExportedEvent extends DomainEvent { + constructor(private readonly payload: ReportExportedPayload) { + super(); + } + + get eventType(): string { + return 'ReportExported'; + } + + get aggregateId(): string { + return this.payload.fileId; + } + + get aggregateType(): string { + return 'ReportFile'; + } + + toPayload(): ReportExportedPayload { + return { ...this.payload }; + } +} diff --git a/backend/services/reporting-service/src/domain/events/report-generated.event.ts b/backend/services/reporting-service/src/domain/events/report-generated.event.ts new file mode 100644 index 00000000..dceafca0 --- /dev/null +++ b/backend/services/reporting-service/src/domain/events/report-generated.event.ts @@ -0,0 +1,33 @@ +import { DomainEvent } from './domain-event.base'; +import { ReportType } from '../value-objects/report-type.enum'; + +export interface ReportGeneratedPayload { + snapshotId: string; + reportType: ReportType; + reportCode: string; + periodKey: string; + rowCount: number; + generatedAt: Date; +} + +export class ReportGeneratedEvent extends DomainEvent { + constructor(private readonly payload: ReportGeneratedPayload) { + super(); + } + + get eventType(): string { + return 'ReportGenerated'; + } + + get aggregateId(): string { + return this.payload.snapshotId; + } + + get aggregateType(): string { + return 'ReportSnapshot'; + } + + toPayload(): ReportGeneratedPayload { + return { ...this.payload }; + } +} diff --git a/backend/services/reporting-service/src/domain/events/snapshot-created.event.ts b/backend/services/reporting-service/src/domain/events/snapshot-created.event.ts new file mode 100644 index 00000000..867121c3 --- /dev/null +++ b/backend/services/reporting-service/src/domain/events/snapshot-created.event.ts @@ -0,0 +1,36 @@ +import { DomainEvent } from './domain-event.base'; +import { ReportType } from '../value-objects/report-type.enum'; +import { ReportPeriod } from '../value-objects/report-period.enum'; + +export interface SnapshotCreatedPayload { + snapshotId: string; + reportType: ReportType; + reportCode: string; + reportPeriod: ReportPeriod; + periodKey: string; + rowCount: number; + periodStartAt: Date; + periodEndAt: Date; +} + +export class SnapshotCreatedEvent extends DomainEvent { + constructor(private readonly payload: SnapshotCreatedPayload) { + super(); + } + + get eventType(): string { + return 'SnapshotCreated'; + } + + get aggregateId(): string { + return this.payload.snapshotId; + } + + get aggregateType(): string { + return 'ReportSnapshot'; + } + + toPayload(): SnapshotCreatedPayload { + return { ...this.payload }; + } +} diff --git a/backend/services/reporting-service/src/domain/repositories/index.ts b/backend/services/reporting-service/src/domain/repositories/index.ts new file mode 100644 index 00000000..bafa37b0 --- /dev/null +++ b/backend/services/reporting-service/src/domain/repositories/index.ts @@ -0,0 +1,3 @@ +export * from './report-definition.repository.interface'; +export * from './report-snapshot.repository.interface'; +export * from './report-file.repository.interface'; diff --git a/backend/services/reporting-service/src/domain/repositories/report-definition.repository.interface.ts b/backend/services/reporting-service/src/domain/repositories/report-definition.repository.interface.ts new file mode 100644 index 00000000..f7306c4f --- /dev/null +++ b/backend/services/reporting-service/src/domain/repositories/report-definition.repository.interface.ts @@ -0,0 +1,15 @@ +import { ReportDefinition } from '../aggregates/report-definition'; +import { ReportType } from '../value-objects'; + +export const REPORT_DEFINITION_REPOSITORY = Symbol('REPORT_DEFINITION_REPOSITORY'); + +export interface IReportDefinitionRepository { + save(definition: ReportDefinition): Promise; + findById(id: bigint): Promise; + findByCode(code: string): Promise; + findByType(type: ReportType): Promise; + findActive(): Promise; + findScheduled(): Promise; + findAll(): Promise; + delete(id: bigint): Promise; +} diff --git a/backend/services/reporting-service/src/domain/repositories/report-file.repository.interface.ts b/backend/services/reporting-service/src/domain/repositories/report-file.repository.interface.ts new file mode 100644 index 00000000..36c7110a --- /dev/null +++ b/backend/services/reporting-service/src/domain/repositories/report-file.repository.interface.ts @@ -0,0 +1,17 @@ +import { ReportFile } from '../entities/report-file.entity'; +import { OutputFormat } from '../value-objects'; + +export const REPORT_FILE_REPOSITORY = Symbol('REPORT_FILE_REPOSITORY'); + +export interface IReportFileRepository { + save(file: ReportFile): Promise; + findById(id: bigint): Promise; + findBySnapshotId(snapshotId: bigint): Promise; + findBySnapshotIdAndFormat( + snapshotId: bigint, + format: OutputFormat, + ): Promise; + findExpired(): Promise; + deleteExpired(): Promise; + delete(id: bigint): Promise; +} diff --git a/backend/services/reporting-service/src/domain/repositories/report-snapshot.repository.interface.ts b/backend/services/reporting-service/src/domain/repositories/report-snapshot.repository.interface.ts new file mode 100644 index 00000000..d2c3d83c --- /dev/null +++ b/backend/services/reporting-service/src/domain/repositories/report-snapshot.repository.interface.ts @@ -0,0 +1,20 @@ +import { ReportSnapshot } from '../aggregates/report-snapshot'; +import { ReportType, ReportPeriod } from '../value-objects'; + +export const REPORT_SNAPSHOT_REPOSITORY = Symbol('REPORT_SNAPSHOT_REPOSITORY'); + +export interface IReportSnapshotRepository { + save(snapshot: ReportSnapshot): Promise; + findById(id: bigint): Promise; + findByCodeAndPeriodKey( + reportCode: string, + periodKey: string, + ): Promise; + findByType(type: ReportType): Promise; + findByCode(reportCode: string): Promise; + findByPeriod(period: ReportPeriod): Promise; + findLatestByCode(reportCode: string): Promise; + findExpired(): Promise; + deleteExpired(): Promise; + delete(id: bigint): Promise; +} diff --git a/backend/services/reporting-service/src/domain/services/index.ts b/backend/services/reporting-service/src/domain/services/index.ts new file mode 100644 index 00000000..5907d503 --- /dev/null +++ b/backend/services/reporting-service/src/domain/services/index.ts @@ -0,0 +1 @@ +export * from './report-generation.service'; diff --git a/backend/services/reporting-service/src/domain/services/report-generation.service.ts b/backend/services/reporting-service/src/domain/services/report-generation.service.ts new file mode 100644 index 00000000..82326547 --- /dev/null +++ b/backend/services/reporting-service/src/domain/services/report-generation.service.ts @@ -0,0 +1,66 @@ +import { Injectable } from '@nestjs/common'; +import { ReportSnapshot } from '../aggregates/report-snapshot'; +import { ReportType, ReportPeriod, SnapshotData, DataSource } from '../value-objects'; + +@Injectable() +export class ReportGenerationDomainService { + generateSnapshot(params: { + reportType: ReportType; + reportCode: string; + reportPeriod: ReportPeriod; + periodKey: string; + rows: any[]; + summary?: Record; + metadata?: Record; + dataSources: string[]; + filterParams?: Record; + periodStartAt: Date; + periodEndAt: Date; + expiresAt?: Date; + }): ReportSnapshot { + const snapshotData = SnapshotData.create({ + rows: params.rows, + summary: params.summary, + metadata: params.metadata, + }); + + const dataSource = DataSource.create(params.dataSources); + + return ReportSnapshot.create({ + reportType: params.reportType, + reportCode: params.reportCode, + reportPeriod: params.reportPeriod, + periodKey: params.periodKey, + snapshotData, + dataSource, + filterParams: params.filterParams, + periodStartAt: params.periodStartAt, + periodEndAt: params.periodEndAt, + expiresAt: params.expiresAt, + }); + } + + calculateGrowthRate( + currentValue: number, + previousValue: number, + ): number { + if (previousValue === 0) { + return currentValue > 0 ? 100 : 0; + } + return ((currentValue - previousValue) / previousValue) * 100; + } + + calculateRank( + items: T[], + valueGetter: (item: T) => number, + ): (T & { rank: number })[] { + const sorted = [...items].sort( + (a, b) => valueGetter(b) - valueGetter(a), + ); + + return sorted.map((item, index) => ({ + ...item, + rank: index + 1, + })); + } +} diff --git a/backend/services/reporting-service/src/domain/value-objects/data-source.vo.ts b/backend/services/reporting-service/src/domain/value-objects/data-source.vo.ts new file mode 100644 index 00000000..d91ac8fc --- /dev/null +++ b/backend/services/reporting-service/src/domain/value-objects/data-source.vo.ts @@ -0,0 +1,28 @@ +export class DataSource { + private constructor( + public readonly sources: string[], + public readonly queryTime: Date, + public readonly dataFreshness: number, + ) {} + + static create(sources: string[]): DataSource { + return new DataSource(sources, new Date(), 0); + } + + static withFreshness(sources: string[], freshnessSeconds: number): DataSource { + return new DataSource(sources, new Date(), freshnessSeconds); + } + + isFresh(maxAgeSeconds: number): boolean { + const age = (Date.now() - this.queryTime.getTime()) / 1000; + return age <= maxAgeSeconds; + } + + addSource(source: string): DataSource { + return new DataSource( + [...this.sources, source], + this.queryTime, + this.dataFreshness, + ); + } +} diff --git a/backend/services/reporting-service/src/domain/value-objects/date-range.spec.ts b/backend/services/reporting-service/src/domain/value-objects/date-range.spec.ts new file mode 100644 index 00000000..52153ac1 --- /dev/null +++ b/backend/services/reporting-service/src/domain/value-objects/date-range.spec.ts @@ -0,0 +1,98 @@ +import { DateRange } from './date-range.vo'; +import { ReportPeriod } from './report-period.enum'; + +describe('DateRange Value Object', () => { + describe('create', () => { + it('should create a date range', () => { + const start = new Date(2024, 0, 1); // Jan 1, 2024 + const end = new Date(2024, 0, 31); // Jan 31, 2024 + const range = DateRange.create(start, end); + + expect(range.startDate).toEqual(start); + expect(range.endDate).toEqual(end); + }); + + it('should throw error if start date is after end date', () => { + const start = new Date(2024, 1, 1); // Feb 1, 2024 + const end = new Date(2024, 0, 31); // Jan 31, 2024 + + expect(() => DateRange.create(start, end)).toThrow('开始日期不能大于结束日期'); + }); + }); + + describe('static factory methods', () => { + it('should create today range', () => { + const range = DateRange.today(); + const now = new Date(); + + expect(range.startDate.getDate()).toBe(now.getDate()); + expect(range.endDate.getDate()).toBe(now.getDate()); + }); + + it('should create this week range', () => { + const range = DateRange.thisWeek(); + + expect(range.getDays()).toBe(7); + }); + + it('should create this month range', () => { + const range = DateRange.thisMonth(); + const now = new Date(); + + expect(range.startDate.getMonth()).toBe(now.getMonth()); + expect(range.startDate.getDate()).toBe(1); + }); + + it('should create this year range', () => { + const range = DateRange.thisYear(); + const now = new Date(); + + expect(range.startDate.getFullYear()).toBe(now.getFullYear()); + expect(range.startDate.getMonth()).toBe(0); + expect(range.startDate.getDate()).toBe(1); + }); + }); + + describe('methods', () => { + it('should calculate days correctly', () => { + const start = new Date(2024, 0, 1, 0, 0, 0); + const end = new Date(2024, 0, 10, 23, 59, 59); + const range = DateRange.create(start, end); + + expect(range.getDays()).toBe(10); + }); + + it('should check if date is contained', () => { + const start = new Date(2024, 0, 1); + const end = new Date(2024, 0, 31); + const range = DateRange.create(start, end); + + expect(range.contains(new Date(2024, 0, 15))).toBe(true); + expect(range.contains(new Date(2024, 1, 1))).toBe(false); + }); + + it('should generate period key for daily', () => { + const start = new Date(2024, 0, 15); + const end = new Date(2024, 0, 15); + const range = DateRange.create(start, end); + + expect(range.toPeriodKey(ReportPeriod.DAILY)).toBe('2024-01-15'); + }); + + it('should generate period key for monthly', () => { + const start = new Date(2024, 0, 1); + const end = new Date(2024, 0, 31); + const range = DateRange.create(start, end); + + expect(range.toPeriodKey(ReportPeriod.MONTHLY)).toBe('2024-01'); + }); + + it('should generate period key for yearly', () => { + const start = new Date(2024, 0, 1); + const end = new Date(2024, 11, 31); + const range = DateRange.create(start, end); + + expect(range.toPeriodKey(ReportPeriod.YEARLY)).toBe('2024'); + }); + }); +}); diff --git a/backend/services/reporting-service/src/domain/value-objects/date-range.vo.ts b/backend/services/reporting-service/src/domain/value-objects/date-range.vo.ts new file mode 100644 index 00000000..a602db9f --- /dev/null +++ b/backend/services/reporting-service/src/domain/value-objects/date-range.vo.ts @@ -0,0 +1,131 @@ +import { ReportPeriod } from './report-period.enum'; + +export class DateRange { + private constructor( + public readonly startDate: Date, + public readonly endDate: Date, + ) { + if (startDate > endDate) { + throw new Error('开始日期不能大于结束日期'); + } + } + + static create(startDate: Date, endDate: Date): DateRange { + return new DateRange(startDate, endDate); + } + + static today(): DateRange { + const now = new Date(); + const start = new Date( + now.getFullYear(), + now.getMonth(), + now.getDate(), + 0, + 0, + 0, + ); + const end = new Date( + now.getFullYear(), + now.getMonth(), + now.getDate(), + 23, + 59, + 59, + 999, + ); + return new DateRange(start, end); + } + + static thisWeek(): DateRange { + const now = new Date(); + const dayOfWeek = now.getDay(); + const diffToMonday = dayOfWeek === 0 ? -6 : 1 - dayOfWeek; + + const monday = new Date(now); + monday.setDate(now.getDate() + diffToMonday); + monday.setHours(0, 0, 0, 0); + + const sunday = new Date(monday); + sunday.setDate(monday.getDate() + 6); + sunday.setHours(23, 59, 59, 999); + + return new DateRange(monday, sunday); + } + + static thisMonth(): DateRange { + const now = new Date(); + const start = new Date(now.getFullYear(), now.getMonth(), 1, 0, 0, 0); + const end = new Date( + now.getFullYear(), + now.getMonth() + 1, + 0, + 23, + 59, + 59, + 999, + ); + return new DateRange(start, end); + } + + static thisQuarter(): DateRange { + const now = new Date(); + const quarter = Math.floor(now.getMonth() / 3); + const start = new Date(now.getFullYear(), quarter * 3, 1, 0, 0, 0); + const end = new Date( + now.getFullYear(), + quarter * 3 + 3, + 0, + 23, + 59, + 59, + 999, + ); + return new DateRange(start, end); + } + + static thisYear(): DateRange { + const now = new Date(); + const start = new Date(now.getFullYear(), 0, 1, 0, 0, 0); + const end = new Date(now.getFullYear(), 11, 31, 23, 59, 59, 999); + return new DateRange(start, end); + } + + getDays(): number { + const diff = this.endDate.getTime() - this.startDate.getTime(); + return Math.ceil(diff / (1000 * 60 * 60 * 24)); + } + + contains(date: Date): boolean { + return date >= this.startDate && date <= this.endDate; + } + + toPeriodKey(period: ReportPeriod): string { + const year = this.startDate.getFullYear(); + const month = (this.startDate.getMonth() + 1).toString().padStart(2, '0'); + const day = this.startDate.getDate().toString().padStart(2, '0'); + + switch (period) { + case ReportPeriod.DAILY: + return `${year}-${month}-${day}`; + case ReportPeriod.WEEKLY: + const weekNumber = this.getWeekNumber(this.startDate); + return `${year}-W${weekNumber.toString().padStart(2, '0')}`; + case ReportPeriod.MONTHLY: + return `${year}-${month}`; + case ReportPeriod.QUARTERLY: + const quarter = Math.floor(this.startDate.getMonth() / 3) + 1; + return `${year}-Q${quarter}`; + case ReportPeriod.YEARLY: + return `${year}`; + default: + return `${year}-${month}-${day}_to_${this.endDate.getFullYear()}-${(this.endDate.getMonth() + 1).toString().padStart(2, '0')}-${this.endDate.getDate().toString().padStart(2, '0')}`; + } + } + + private getWeekNumber(date: Date): number { + const firstDayOfYear = new Date(date.getFullYear(), 0, 1); + const pastDaysOfYear = + (date.getTime() - firstDayOfYear.getTime()) / 86400000; + return Math.ceil((pastDaysOfYear + firstDayOfYear.getDay() + 1) / 7); + } +} diff --git a/backend/services/reporting-service/src/domain/value-objects/index.ts b/backend/services/reporting-service/src/domain/value-objects/index.ts new file mode 100644 index 00000000..7f6bcfc7 --- /dev/null +++ b/backend/services/reporting-service/src/domain/value-objects/index.ts @@ -0,0 +1,9 @@ +export * from './report-type.enum'; +export * from './report-period.enum'; +export * from './report-dimension.enum'; +export * from './output-format.enum'; +export * from './date-range.vo'; +export * from './report-parameters.vo'; +export * from './report-schedule.vo'; +export * from './snapshot-data.vo'; +export * from './data-source.vo'; diff --git a/backend/services/reporting-service/src/domain/value-objects/output-format.enum.ts b/backend/services/reporting-service/src/domain/value-objects/output-format.enum.ts new file mode 100644 index 00000000..b365cc29 --- /dev/null +++ b/backend/services/reporting-service/src/domain/value-objects/output-format.enum.ts @@ -0,0 +1,21 @@ +export enum OutputFormat { + EXCEL = 'EXCEL', + PDF = 'PDF', + CSV = 'CSV', + JSON = 'JSON', +} + +export const OutputFormatMimeTypes: Record = { + [OutputFormat.EXCEL]: + 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet', + [OutputFormat.PDF]: 'application/pdf', + [OutputFormat.CSV]: 'text/csv', + [OutputFormat.JSON]: 'application/json', +}; + +export const OutputFormatExtensions: Record = { + [OutputFormat.EXCEL]: 'xlsx', + [OutputFormat.PDF]: 'pdf', + [OutputFormat.CSV]: 'csv', + [OutputFormat.JSON]: 'json', +}; diff --git a/backend/services/reporting-service/src/domain/value-objects/report-dimension.enum.ts b/backend/services/reporting-service/src/domain/value-objects/report-dimension.enum.ts new file mode 100644 index 00000000..c1f802ac --- /dev/null +++ b/backend/services/reporting-service/src/domain/value-objects/report-dimension.enum.ts @@ -0,0 +1,11 @@ +export enum ReportDimension { + TIME = 'TIME', + REGION = 'REGION', + USER = 'USER', + USER_TYPE = 'USER_TYPE', + RIGHT_TYPE = 'RIGHT_TYPE', + COMMUNITY = 'COMMUNITY', + ACCOUNT = 'ACCOUNT', + SOURCE = 'SOURCE', + PRODUCT = 'PRODUCT', +} diff --git a/backend/services/reporting-service/src/domain/value-objects/report-parameters.vo.ts b/backend/services/reporting-service/src/domain/value-objects/report-parameters.vo.ts new file mode 100644 index 00000000..9b1f654d --- /dev/null +++ b/backend/services/reporting-service/src/domain/value-objects/report-parameters.vo.ts @@ -0,0 +1,65 @@ +import { ReportDimension } from './report-dimension.enum'; +import { DateRange } from './date-range.vo'; + +export class ReportParameters { + private constructor( + public readonly dateRange: DateRange, + public readonly dimensions: ReportDimension[], + public readonly filters: Record, + public readonly groupBy: string[], + public readonly orderBy: { field: string; direction: 'ASC' | 'DESC' }[], + public readonly pagination: { page: number; pageSize: number } | null, + ) {} + + static create(params: { + startDate: Date; + endDate: Date; + dimensions?: ReportDimension[]; + filters?: Record; + groupBy?: string[]; + orderBy?: { field: string; direction: 'ASC' | 'DESC' }[]; + page?: number; + pageSize?: number; + }): ReportParameters { + return new ReportParameters( + DateRange.create(params.startDate, params.endDate), + params.dimensions || [], + params.filters || {}, + params.groupBy || [], + params.orderBy || [], + params.page && params.pageSize + ? { page: params.page, pageSize: params.pageSize } + : null, + ); + } + + withFilter(key: string, value: any): ReportParameters { + return new ReportParameters( + this.dateRange, + this.dimensions, + { ...this.filters, [key]: value }, + this.groupBy, + this.orderBy, + this.pagination, + ); + } + + withDimension(dimension: ReportDimension): ReportParameters { + return new ReportParameters( + this.dateRange, + [...this.dimensions, dimension], + this.filters, + this.groupBy, + this.orderBy, + this.pagination, + ); + } + + hasFilter(key: string): boolean { + return key in this.filters; + } + + getFilter(key: string, defaultValue?: T): T | undefined { + return this.filters[key] ?? defaultValue; + } +} diff --git a/backend/services/reporting-service/src/domain/value-objects/report-period.enum.ts b/backend/services/reporting-service/src/domain/value-objects/report-period.enum.ts new file mode 100644 index 00000000..92e76439 --- /dev/null +++ b/backend/services/reporting-service/src/domain/value-objects/report-period.enum.ts @@ -0,0 +1,17 @@ +export enum ReportPeriod { + DAILY = 'DAILY', + WEEKLY = 'WEEKLY', + MONTHLY = 'MONTHLY', + QUARTERLY = 'QUARTERLY', + YEARLY = 'YEARLY', + CUSTOM = 'CUSTOM', +} + +export const ReportPeriodLabels: Record = { + [ReportPeriod.DAILY]: '日报表', + [ReportPeriod.WEEKLY]: '周报表', + [ReportPeriod.MONTHLY]: '月报表', + [ReportPeriod.QUARTERLY]: '季度报表', + [ReportPeriod.YEARLY]: '年度报表', + [ReportPeriod.CUSTOM]: '自定义周期', +}; diff --git a/backend/services/reporting-service/src/domain/value-objects/report-schedule.vo.ts b/backend/services/reporting-service/src/domain/value-objects/report-schedule.vo.ts new file mode 100644 index 00000000..c097820b --- /dev/null +++ b/backend/services/reporting-service/src/domain/value-objects/report-schedule.vo.ts @@ -0,0 +1,59 @@ +export class ReportSchedule { + private constructor( + public readonly cronExpression: string, + public readonly timezone: string, + public readonly enabled: boolean, + ) {} + + static create( + cronExpression: string, + timezone: string = 'Asia/Shanghai', + enabled: boolean = true, + ): ReportSchedule { + return new ReportSchedule(cronExpression, timezone, enabled); + } + + static daily(hour: number = 0, minute: number = 0): ReportSchedule { + return new ReportSchedule(`${minute} ${hour} * * *`, 'Asia/Shanghai', true); + } + + static weekly( + dayOfWeek: number, + hour: number = 0, + minute: number = 0, + ): ReportSchedule { + return new ReportSchedule( + `${minute} ${hour} * * ${dayOfWeek}`, + 'Asia/Shanghai', + true, + ); + } + + static monthly( + dayOfMonth: number, + hour: number = 0, + minute: number = 0, + ): ReportSchedule { + return new ReportSchedule( + `${minute} ${hour} ${dayOfMonth} * *`, + 'Asia/Shanghai', + true, + ); + } + + static quarterly(dayOfQuarter: number = 1, hour: number = 0): ReportSchedule { + return new ReportSchedule( + `0 ${hour} ${dayOfQuarter} 1,4,7,10 *`, + 'Asia/Shanghai', + true, + ); + } + + enable(): ReportSchedule { + return new ReportSchedule(this.cronExpression, this.timezone, true); + } + + disable(): ReportSchedule { + return new ReportSchedule(this.cronExpression, this.timezone, false); + } +} diff --git a/backend/services/reporting-service/src/domain/value-objects/report-type.enum.ts b/backend/services/reporting-service/src/domain/value-objects/report-type.enum.ts new file mode 100644 index 00000000..b5801306 --- /dev/null +++ b/backend/services/reporting-service/src/domain/value-objects/report-type.enum.ts @@ -0,0 +1,28 @@ +export enum ReportType { + // 龙虎榜报表 + LEADERBOARD_REPORT = 'LEADERBOARD_REPORT', + + // 认种报表 + PLANTING_REPORT = 'PLANTING_REPORT', + REGIONAL_PLANTING_REPORT = 'REGIONAL_PLANTING_REPORT', + + // 授权公司报表 + AUTHORIZED_COMPANY_TOP_REPORT = 'AUTHORIZED_COMPANY_TOP_REPORT', + + // 社区报表 + COMMUNITY_REPORT = 'COMMUNITY_REPORT', + + // 系统账户报表 + SYSTEM_ACCOUNT_MONTHLY_REPORT = 'SYSTEM_ACCOUNT_MONTHLY_REPORT', + SYSTEM_ACCOUNT_INCOME_REPORT = 'SYSTEM_ACCOUNT_INCOME_REPORT', +} + +export const ReportTypeLabels: Record = { + [ReportType.LEADERBOARD_REPORT]: '龙虎榜数据报表', + [ReportType.PLANTING_REPORT]: '榴莲树认种报表', + [ReportType.REGIONAL_PLANTING_REPORT]: '区域认种报表', + [ReportType.AUTHORIZED_COMPANY_TOP_REPORT]: '授权公司第1名统计', + [ReportType.COMMUNITY_REPORT]: '社区数据统计', + [ReportType.SYSTEM_ACCOUNT_MONTHLY_REPORT]: '系统账户月度报表', + [ReportType.SYSTEM_ACCOUNT_INCOME_REPORT]: '系统账户收益来源报表', +}; diff --git a/backend/services/reporting-service/src/domain/value-objects/snapshot-data.vo.ts b/backend/services/reporting-service/src/domain/value-objects/snapshot-data.vo.ts new file mode 100644 index 00000000..e76b153b --- /dev/null +++ b/backend/services/reporting-service/src/domain/value-objects/snapshot-data.vo.ts @@ -0,0 +1,44 @@ +export class SnapshotData { + private constructor( + public readonly rows: any[], + public readonly summary: Record, + public readonly metadata: Record, + ) {} + + static create(params: { + rows: any[]; + summary?: Record; + metadata?: Record; + }): SnapshotData { + return new SnapshotData( + params.rows, + params.summary || {}, + params.metadata || {}, + ); + } + + getRowCount(): number { + return this.rows.length; + } + + isEmpty(): boolean { + return this.rows.length === 0; + } + + getSummary(key: string, defaultValue?: T): T | undefined { + return this.summary[key] ?? defaultValue; + } + + getMetadata(key: string, defaultValue?: T): T | undefined { + return this.metadata[key] ?? defaultValue; + } + + toJSON(): object { + return { + rows: this.rows, + summary: this.summary, + metadata: this.metadata, + rowCount: this.getRowCount(), + }; + } +} diff --git a/backend/services/reporting-service/src/infrastructure/export/csv-export.service.ts b/backend/services/reporting-service/src/infrastructure/export/csv-export.service.ts new file mode 100644 index 00000000..b4d87935 --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/export/csv-export.service.ts @@ -0,0 +1,74 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { stringify } from 'csv-stringify/sync'; +import * as path from 'path'; +import * as fs from 'fs'; +import { ConfigService } from '@nestjs/config'; + +export interface CsvColumn { + header: string; + key: string; +} + +export interface CsvExportOptions { + columns: CsvColumn[]; + data: any[]; +} + +@Injectable() +export class CsvExportService { + private readonly logger = new Logger(CsvExportService.name); + private readonly storagePath: string; + + constructor(private readonly configService: ConfigService) { + this.storagePath = this.configService.get( + 'FILE_STORAGE_PATH', + './storage/reports', + ); + } + + async export( + fileName: string, + options: CsvExportOptions, + ): Promise<{ filePath: string; fileSize: number }> { + const headers = options.columns.map((col) => col.header); + const keys = options.columns.map((col) => col.key); + + const rows = options.data.map((item) => + keys.map((key) => { + const value = item[key]; + if (value === null || value === undefined) { + return ''; + } + if (typeof value === 'object') { + return JSON.stringify(value); + } + return String(value); + }), + ); + + const csvContent = stringify([headers, ...rows]); + + // Add BOM for Excel compatibility with Chinese characters + const bom = '\uFEFF'; + const contentWithBom = bom + csvContent; + + // Ensure storage directory exists + const fullPath = path.join(this.storagePath, fileName); + const dir = path.dirname(fullPath); + if (!fs.existsSync(dir)) { + fs.mkdirSync(dir, { recursive: true }); + } + + // Write file + fs.writeFileSync(fullPath, contentWithBom, 'utf8'); + + const stats = fs.statSync(fullPath); + + this.logger.log(`CSV file exported: ${fullPath} (${stats.size} bytes)`); + + return { + filePath: fullPath, + fileSize: stats.size, + }; + } +} diff --git a/backend/services/reporting-service/src/infrastructure/export/excel-export.service.ts b/backend/services/reporting-service/src/infrastructure/export/excel-export.service.ts new file mode 100644 index 00000000..10839d84 --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/export/excel-export.service.ts @@ -0,0 +1,108 @@ +import { Injectable, Logger } from '@nestjs/common'; +import * as ExcelJS from 'exceljs'; +import * as path from 'path'; +import * as fs from 'fs'; +import { ConfigService } from '@nestjs/config'; + +export interface ExcelColumn { + header: string; + key: string; + width?: number; +} + +export interface ExcelExportOptions { + sheetName?: string; + columns: ExcelColumn[]; + data: any[]; + title?: string; + subtitle?: string; +} + +@Injectable() +export class ExcelExportService { + private readonly logger = new Logger(ExcelExportService.name); + private readonly storagePath: string; + + constructor(private readonly configService: ConfigService) { + this.storagePath = this.configService.get( + 'FILE_STORAGE_PATH', + './storage/reports', + ); + } + + async export( + fileName: string, + options: ExcelExportOptions, + ): Promise<{ filePath: string; fileSize: number }> { + const workbook = new ExcelJS.Workbook(); + workbook.creator = 'Reporting Service'; + workbook.created = new Date(); + + const worksheet = workbook.addWorksheet(options.sheetName || 'Report'); + + // Add title if provided + if (options.title) { + worksheet.addRow([options.title]); + worksheet.mergeCells(1, 1, 1, options.columns.length); + const titleRow = worksheet.getRow(1); + titleRow.font = { bold: true, size: 16 }; + titleRow.alignment = { horizontal: 'center' }; + } + + // Add subtitle if provided + if (options.subtitle) { + const subtitleRowNum = options.title ? 2 : 1; + worksheet.addRow([options.subtitle]); + worksheet.mergeCells(subtitleRowNum, 1, subtitleRowNum, options.columns.length); + const subtitleRow = worksheet.getRow(subtitleRowNum); + subtitleRow.font = { size: 12 }; + subtitleRow.alignment = { horizontal: 'center' }; + } + + // Add empty row before headers + if (options.title || options.subtitle) { + worksheet.addRow([]); + } + + // Setup columns + worksheet.columns = options.columns.map((col) => ({ + header: col.header, + key: col.key, + width: col.width || 15, + })); + + // Style header row + const headerRowNum = (options.title ? 1 : 0) + (options.subtitle ? 1 : 0) + (options.title || options.subtitle ? 1 : 0) + 1; + const headerRow = worksheet.getRow(headerRowNum); + headerRow.font = { bold: true }; + headerRow.fill = { + type: 'pattern', + pattern: 'solid', + fgColor: { argb: 'FFE0E0E0' }, + }; + + // Add data rows + options.data.forEach((row) => { + worksheet.addRow(row); + }); + + // Ensure storage directory exists + const fullPath = path.join(this.storagePath, fileName); + const dir = path.dirname(fullPath); + if (!fs.existsSync(dir)) { + fs.mkdirSync(dir, { recursive: true }); + } + + // Write file + await workbook.xlsx.writeFile(fullPath); + + const stats = fs.statSync(fullPath); + + this.logger.log(`Excel file exported: ${fullPath} (${stats.size} bytes)`); + + return { + filePath: fullPath, + fileSize: stats.size, + }; + } +} diff --git a/backend/services/reporting-service/src/infrastructure/export/export.module.ts b/backend/services/reporting-service/src/infrastructure/export/export.module.ts new file mode 100644 index 00000000..6289a631 --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/export/export.module.ts @@ -0,0 +1,10 @@ +import { Module } from '@nestjs/common'; +import { ExcelExportService } from './excel-export.service'; +import { CsvExportService } from './csv-export.service'; +import { PdfExportService } from './pdf-export.service'; + +@Module({ + providers: [ExcelExportService, CsvExportService, PdfExportService], + exports: [ExcelExportService, CsvExportService, PdfExportService], +}) +export class ExportModule {} diff --git a/backend/services/reporting-service/src/infrastructure/export/pdf-export.service.ts b/backend/services/reporting-service/src/infrastructure/export/pdf-export.service.ts new file mode 100644 index 00000000..eeb4eed4 --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/export/pdf-export.service.ts @@ -0,0 +1,154 @@ +import { Injectable, Logger } from '@nestjs/common'; +import * as PDFDocument from 'pdfkit'; +import * as path from 'path'; +import * as fs from 'fs'; +import { ConfigService } from '@nestjs/config'; + +export interface PdfColumn { + header: string; + key: string; + width?: number; +} + +export interface PdfExportOptions { + columns: PdfColumn[]; + data: any[]; + title?: string; + subtitle?: string; + orientation?: 'portrait' | 'landscape'; +} + +@Injectable() +export class PdfExportService { + private readonly logger = new Logger(PdfExportService.name); + private readonly storagePath: string; + + constructor(private readonly configService: ConfigService) { + this.storagePath = this.configService.get( + 'FILE_STORAGE_PATH', + './storage/reports', + ); + } + + async export( + fileName: string, + options: PdfExportOptions, + ): Promise<{ filePath: string; fileSize: number }> { + return new Promise((resolve, reject) => { + const doc = new PDFDocument({ + size: 'A4', + layout: options.orientation || 'portrait', + margins: { top: 50, bottom: 50, left: 50, right: 50 }, + }); + + // Ensure storage directory exists + const fullPath = path.join(this.storagePath, fileName); + const dir = path.dirname(fullPath); + if (!fs.existsSync(dir)) { + fs.mkdirSync(dir, { recursive: true }); + } + + const stream = fs.createWriteStream(fullPath); + doc.pipe(stream); + + // Title + if (options.title) { + doc.fontSize(18).text(options.title, { align: 'center' }); + doc.moveDown(); + } + + // Subtitle + if (options.subtitle) { + doc.fontSize(12).text(options.subtitle, { align: 'center' }); + doc.moveDown(); + } + + // Calculate column widths + const pageWidth = + doc.page.width - doc.page.margins.left - doc.page.margins.right; + const columnCount = options.columns.length; + const defaultWidth = pageWidth / columnCount; + + // Draw table header + doc.fontSize(10); + let x = doc.page.margins.left; + const headerY = doc.y; + + // Header background + doc + .fillColor('#e0e0e0') + .rect(x, headerY - 5, pageWidth, 20) + .fill(); + + // Header text + doc.fillColor('#000000'); + options.columns.forEach((col) => { + const width = col.width || defaultWidth; + doc.text(col.header, x, headerY, { + width: width - 5, + align: 'left', + }); + x += width; + }); + + doc.moveDown(); + + // Draw data rows + options.data.forEach((row, rowIndex) => { + // Check if we need a new page + if (doc.y > doc.page.height - 100) { + doc.addPage(); + } + + x = doc.page.margins.left; + const rowY = doc.y; + + // Alternate row background + if (rowIndex % 2 === 1) { + doc + .fillColor('#f5f5f5') + .rect(x, rowY - 5, pageWidth, 15) + .fill(); + } + + doc.fillColor('#000000'); + options.columns.forEach((col) => { + const width = col.width || defaultWidth; + const value = row[col.key]; + const displayValue = + value === null || value === undefined ? '' : String(value); + doc.text(displayValue, x, rowY, { + width: width - 5, + align: 'left', + }); + x += width; + }); + + doc.moveDown(0.5); + }); + + // Footer + doc + .fontSize(8) + .text( + `Generated at: ${new Date().toISOString()}`, + doc.page.margins.left, + doc.page.height - 30, + { align: 'center' }, + ); + + doc.end(); + + stream.on('finish', () => { + const stats = fs.statSync(fullPath); + this.logger.log(`PDF file exported: ${fullPath} (${stats.size} bytes)`); + resolve({ + filePath: fullPath, + fileSize: stats.size, + }); + }); + + stream.on('error', reject); + }); + } +} diff --git a/backend/services/reporting-service/src/infrastructure/external/leaderboard-service/leaderboard-service.client.ts b/backend/services/reporting-service/src/infrastructure/external/leaderboard-service/leaderboard-service.client.ts new file mode 100644 index 00000000..cdde13b9 --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/external/leaderboard-service/leaderboard-service.client.ts @@ -0,0 +1,94 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { ConfigService } from '@nestjs/config'; + +export interface LeaderboardEntry { + rank: number; + userId: bigint; + username: string; + score: number; + teamCount?: number; + plantingCount?: number; +} + +export interface LeaderboardData { + type: string; + period: string; + entries: LeaderboardEntry[]; + generatedAt: Date; +} + +@Injectable() +export class LeaderboardServiceClient { + private readonly logger = new Logger(LeaderboardServiceClient.name); + private readonly baseUrl: string; + + constructor(private readonly configService: ConfigService) { + this.baseUrl = this.configService.get( + 'LEADERBOARD_SERVICE_URL', + 'http://localhost:3007', + ); + } + + async getDailyLeaderboard(date?: Date): Promise { + // In production, this would make an HTTP call to leaderboard-service + this.logger.debug(`Fetching daily leaderboard from ${this.baseUrl}`); + + // Mock data for development + return { + type: 'DAILY', + period: (date || new Date()).toISOString().split('T')[0], + entries: [], + generatedAt: new Date(), + }; + } + + async getWeeklyLeaderboard(weekKey?: string): Promise { + this.logger.debug(`Fetching weekly leaderboard from ${this.baseUrl}`); + + return { + type: 'WEEKLY', + period: weekKey || this.getCurrentWeekKey(), + entries: [], + generatedAt: new Date(), + }; + } + + async getMonthlyLeaderboard(monthKey?: string): Promise { + this.logger.debug(`Fetching monthly leaderboard from ${this.baseUrl}`); + + return { + type: 'MONTHLY', + period: monthKey || this.getCurrentMonthKey(), + entries: [], + generatedAt: new Date(), + }; + } + + async getTopByRegion( + regionCode: string, + limit: number = 10, + ): Promise { + this.logger.debug( + `Fetching top ${limit} for region ${regionCode} from ${this.baseUrl}`, + ); + + return []; + } + + private getCurrentWeekKey(): string { + const now = new Date(); + const year = now.getFullYear(); + const weekNumber = Math.ceil( + ((now.getTime() - new Date(year, 0, 1).getTime()) / 86400000 + + new Date(year, 0, 1).getDay() + + 1) / + 7, + ); + return `${year}-W${weekNumber.toString().padStart(2, '0')}`; + } + + private getCurrentMonthKey(): string { + const now = new Date(); + return `${now.getFullYear()}-${(now.getMonth() + 1).toString().padStart(2, '0')}`; + } +} diff --git a/backend/services/reporting-service/src/infrastructure/external/planting-service/planting-service.client.ts b/backend/services/reporting-service/src/infrastructure/external/planting-service/planting-service.client.ts new file mode 100644 index 00000000..095d7924 --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/external/planting-service/planting-service.client.ts @@ -0,0 +1,99 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { ConfigService } from '@nestjs/config'; + +export interface PlantingStats { + totalOrders: number; + totalTrees: number; + totalAmount: string; + newUsers: number; + activeUsers: number; +} + +export interface RegionalPlantingStats extends PlantingStats { + regionCode: string; + regionName: string; +} + +@Injectable() +export class PlantingServiceClient { + private readonly logger = new Logger(PlantingServiceClient.name); + private readonly baseUrl: string; + + constructor(private readonly configService: ConfigService) { + this.baseUrl = this.configService.get( + 'PLANTING_SERVICE_URL', + 'http://localhost:3003', + ); + } + + async getDailyStats(date: Date): Promise { + this.logger.debug(`Fetching daily planting stats from ${this.baseUrl}`); + + // Mock data for development + return { + totalOrders: 0, + totalTrees: 0, + totalAmount: '0', + newUsers: 0, + activeUsers: 0, + }; + } + + async getStatsForDateRange( + startDate: Date, + endDate: Date, + ): Promise { + this.logger.debug( + `Fetching planting stats for range ${startDate} - ${endDate} from ${this.baseUrl}`, + ); + + return { + totalOrders: 0, + totalTrees: 0, + totalAmount: '0', + newUsers: 0, + activeUsers: 0, + }; + } + + async getRegionalStats( + regionCode: string, + startDate: Date, + endDate: Date, + ): Promise { + this.logger.debug( + `Fetching regional planting stats for ${regionCode} from ${this.baseUrl}`, + ); + + return { + regionCode, + regionName: '', + totalOrders: 0, + totalTrees: 0, + totalAmount: '0', + newUsers: 0, + activeUsers: 0, + }; + } + + async getAllProvincesStats( + startDate: Date, + endDate: Date, + ): Promise { + this.logger.debug(`Fetching all provinces planting stats from ${this.baseUrl}`); + + return []; + } + + async getAllCitiesStats( + provinceCode: string, + startDate: Date, + endDate: Date, + ): Promise { + this.logger.debug( + `Fetching all cities planting stats for province ${provinceCode} from ${this.baseUrl}`, + ); + + return []; + } +} diff --git a/backend/services/reporting-service/src/infrastructure/infrastructure.module.ts b/backend/services/reporting-service/src/infrastructure/infrastructure.module.ts new file mode 100644 index 00000000..cc5fd2c0 --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/infrastructure.module.ts @@ -0,0 +1,46 @@ +import { Module } from '@nestjs/common'; +import { PrismaService } from './persistence/prisma/prisma.service'; +import { ReportDefinitionRepository } from './persistence/repositories/report-definition.repository.impl'; +import { ReportSnapshotRepository } from './persistence/repositories/report-snapshot.repository.impl'; +import { ReportFileRepository } from './persistence/repositories/report-file.repository.impl'; +import { + REPORT_DEFINITION_REPOSITORY, + REPORT_SNAPSHOT_REPOSITORY, + REPORT_FILE_REPOSITORY, +} from '../domain/repositories'; +import { LeaderboardServiceClient } from './external/leaderboard-service/leaderboard-service.client'; +import { PlantingServiceClient } from './external/planting-service/planting-service.client'; +import { ExportModule } from './export/export.module'; +import { RedisModule } from './redis/redis.module'; + +@Module({ + imports: [ExportModule, RedisModule], + providers: [ + PrismaService, + { + provide: REPORT_DEFINITION_REPOSITORY, + useClass: ReportDefinitionRepository, + }, + { + provide: REPORT_SNAPSHOT_REPOSITORY, + useClass: ReportSnapshotRepository, + }, + { + provide: REPORT_FILE_REPOSITORY, + useClass: ReportFileRepository, + }, + LeaderboardServiceClient, + PlantingServiceClient, + ], + exports: [ + PrismaService, + REPORT_DEFINITION_REPOSITORY, + REPORT_SNAPSHOT_REPOSITORY, + REPORT_FILE_REPOSITORY, + LeaderboardServiceClient, + PlantingServiceClient, + ExportModule, + RedisModule, + ], +}) +export class InfrastructureModule {} diff --git a/backend/services/reporting-service/src/infrastructure/persistence/mappers/report-definition.mapper.ts b/backend/services/reporting-service/src/infrastructure/persistence/mappers/report-definition.mapper.ts new file mode 100644 index 00000000..cb0faa5b --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/persistence/mappers/report-definition.mapper.ts @@ -0,0 +1,48 @@ +import { ReportDefinition as PrismaReportDefinition } from '@prisma/client'; +import { ReportDefinition } from '../../../domain/aggregates/report-definition'; +import { ReportType, ReportSchedule, OutputFormat } from '../../../domain/value-objects'; + +export class ReportDefinitionMapper { + static toDomain(raw: PrismaReportDefinition): ReportDefinition { + const schedule = + raw.scheduleCron + ? ReportSchedule.create( + raw.scheduleCron, + raw.scheduleTimezone || 'Asia/Shanghai', + raw.scheduleEnabled, + ) + : null; + + return ReportDefinition.reconstitute({ + id: raw.id, + reportType: raw.reportType as ReportType, + reportName: raw.reportName, + reportCode: raw.reportCode, + description: raw.description || '', + parameters: raw.parameters as Record, + schedule, + outputFormats: raw.outputFormats as OutputFormat[], + isActive: raw.isActive, + createdAt: raw.createdAt, + lastGeneratedAt: raw.lastGeneratedAt, + }); + } + + static toPersistence(domain: ReportDefinition): Omit & { id?: bigint } { + return { + id: domain.id || undefined, + reportType: domain.reportType, + reportName: domain.reportName, + reportCode: domain.reportCode, + description: domain.description, + parameters: domain.parameters, + scheduleCron: domain.schedule?.cronExpression || null, + scheduleTimezone: domain.schedule?.timezone || 'Asia/Shanghai', + scheduleEnabled: domain.schedule?.enabled || false, + outputFormats: domain.outputFormats, + isActive: domain.isActive, + createdAt: domain.createdAt, + lastGeneratedAt: domain.lastGeneratedAt, + }; + } +} diff --git a/backend/services/reporting-service/src/infrastructure/persistence/mappers/report-file.mapper.ts b/backend/services/reporting-service/src/infrastructure/persistence/mappers/report-file.mapper.ts new file mode 100644 index 00000000..44d8c99c --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/persistence/mappers/report-file.mapper.ts @@ -0,0 +1,41 @@ +import { ReportFile as PrismaReportFile } from '@prisma/client'; +import { ReportFile } from '../../../domain/entities/report-file.entity'; +import { OutputFormat } from '../../../domain/value-objects'; + +export class ReportFileMapper { + static toDomain(raw: PrismaReportFile): ReportFile { + return ReportFile.reconstitute({ + id: raw.id, + snapshotId: raw.snapshotId, + fileName: raw.fileName, + filePath: raw.filePath, + fileUrl: raw.fileUrl, + fileSize: raw.fileSize, + fileFormat: raw.fileFormat as OutputFormat, + mimeType: raw.mimeType, + downloadCount: raw.downloadCount, + lastDownloadAt: raw.lastDownloadAt, + createdAt: raw.createdAt, + expiresAt: raw.expiresAt, + }); + } + + static toPersistence( + domain: ReportFile, + ): Omit & { id?: bigint } { + return { + id: domain.id || undefined, + snapshotId: domain.snapshotId, + fileName: domain.fileName, + filePath: domain.filePath, + fileUrl: domain.fileUrl, + fileSize: domain.fileSize, + fileFormat: domain.fileFormat, + mimeType: domain.mimeType, + downloadCount: domain.downloadCount, + lastDownloadAt: domain.lastDownloadAt, + createdAt: domain.createdAt, + expiresAt: domain.expiresAt, + }; + } +} diff --git a/backend/services/reporting-service/src/infrastructure/persistence/mappers/report-snapshot.mapper.ts b/backend/services/reporting-service/src/infrastructure/persistence/mappers/report-snapshot.mapper.ts new file mode 100644 index 00000000..4c311fbb --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/persistence/mappers/report-snapshot.mapper.ts @@ -0,0 +1,66 @@ +import { ReportSnapshot as PrismaReportSnapshot } from '@prisma/client'; +import { ReportSnapshot } from '../../../domain/aggregates/report-snapshot'; +import { + ReportType, + ReportPeriod, + SnapshotData, + DataSource, +} from '../../../domain/value-objects'; + +export class ReportSnapshotMapper { + static toDomain(raw: PrismaReportSnapshot): ReportSnapshot { + const snapshotDataRaw = raw.snapshotData as { + rows: any[]; + summary?: Record; + metadata?: Record; + }; + + const snapshotData = SnapshotData.create({ + rows: snapshotDataRaw.rows || [], + summary: snapshotDataRaw.summary, + metadata: snapshotDataRaw.metadata, + }); + + const dataSource = DataSource.withFreshness( + raw.dataSources, + raw.dataFreshness, + ); + + return ReportSnapshot.reconstitute({ + id: raw.id, + reportType: raw.reportType as ReportType, + reportCode: raw.reportCode, + reportPeriod: raw.reportPeriod as ReportPeriod, + periodKey: raw.periodKey, + snapshotData, + dataSource, + filterParams: raw.filterParams as Record | null, + periodStartAt: raw.periodStartAt, + periodEndAt: raw.periodEndAt, + generatedAt: raw.generatedAt, + expiresAt: raw.expiresAt, + }); + } + + static toPersistence( + domain: ReportSnapshot, + ): Omit & { id?: bigint } { + return { + id: domain.id || undefined, + reportType: domain.reportType, + reportCode: domain.reportCode, + reportPeriod: domain.reportPeriod, + periodKey: domain.periodKey, + snapshotData: domain.snapshotData.toJSON(), + summaryData: domain.snapshotData.summary, + dataSources: domain.dataSource.sources, + dataFreshness: domain.dataSource.dataFreshness, + filterParams: domain.filterParams, + rowCount: domain.rowCount, + periodStartAt: domain.periodStartAt, + periodEndAt: domain.periodEndAt, + generatedAt: domain.generatedAt, + expiresAt: domain.expiresAt, + }; + } +} diff --git a/backend/services/reporting-service/src/infrastructure/persistence/prisma/prisma.service.ts b/backend/services/reporting-service/src/infrastructure/persistence/prisma/prisma.service.ts new file mode 100644 index 00000000..7ffd32da --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/persistence/prisma/prisma.service.ts @@ -0,0 +1,16 @@ +import { Injectable, OnModuleInit, OnModuleDestroy } from '@nestjs/common'; +import { PrismaClient } from '@prisma/client'; + +@Injectable() +export class PrismaService + extends PrismaClient + implements OnModuleInit, OnModuleDestroy +{ + async onModuleInit() { + await this.$connect(); + } + + async onModuleDestroy() { + await this.$disconnect(); + } +} diff --git a/backend/services/reporting-service/src/infrastructure/persistence/repositories/report-definition.repository.impl.ts b/backend/services/reporting-service/src/infrastructure/persistence/repositories/report-definition.repository.impl.ts new file mode 100644 index 00000000..279af2c4 --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/persistence/repositories/report-definition.repository.impl.ts @@ -0,0 +1,99 @@ +import { Injectable } from '@nestjs/common'; +import { Prisma } from '@prisma/client'; +import { PrismaService } from '../prisma/prisma.service'; +import { ReportDefinition } from '../../../domain/aggregates/report-definition'; +import { IReportDefinitionRepository } from '../../../domain/repositories'; +import { ReportType } from '../../../domain/value-objects'; +import { ReportDefinitionMapper } from '../mappers/report-definition.mapper'; + +@Injectable() +export class ReportDefinitionRepository implements IReportDefinitionRepository { + constructor(private readonly prisma: PrismaService) {} + + async save(definition: ReportDefinition): Promise { + const data = ReportDefinitionMapper.toPersistence(definition); + + if (definition.id) { + const updated = await this.prisma.reportDefinition.update({ + where: { id: definition.id }, + data: { + reportName: data.reportName, + description: data.description, + parameters: data.parameters as Prisma.InputJsonValue, + scheduleCron: data.scheduleCron, + scheduleTimezone: data.scheduleTimezone, + scheduleEnabled: data.scheduleEnabled, + outputFormats: data.outputFormats, + isActive: data.isActive, + lastGeneratedAt: data.lastGeneratedAt, + }, + }); + return ReportDefinitionMapper.toDomain(updated); + } + + const created = await this.prisma.reportDefinition.create({ + data: { + reportType: data.reportType, + reportName: data.reportName, + reportCode: data.reportCode, + description: data.description, + parameters: data.parameters as Prisma.InputJsonValue, + scheduleCron: data.scheduleCron, + scheduleTimezone: data.scheduleTimezone, + scheduleEnabled: data.scheduleEnabled, + outputFormats: data.outputFormats, + isActive: data.isActive, + }, + }); + return ReportDefinitionMapper.toDomain(created); + } + + async findById(id: bigint): Promise { + const found = await this.prisma.reportDefinition.findUnique({ + where: { id }, + }); + return found ? ReportDefinitionMapper.toDomain(found) : null; + } + + async findByCode(code: string): Promise { + const found = await this.prisma.reportDefinition.findUnique({ + where: { reportCode: code }, + }); + return found ? ReportDefinitionMapper.toDomain(found) : null; + } + + async findByType(type: ReportType): Promise { + const found = await this.prisma.reportDefinition.findMany({ + where: { reportType: type }, + }); + return found.map(ReportDefinitionMapper.toDomain); + } + + async findActive(): Promise { + const found = await this.prisma.reportDefinition.findMany({ + where: { isActive: true }, + }); + return found.map(ReportDefinitionMapper.toDomain); + } + + async findScheduled(): Promise { + const found = await this.prisma.reportDefinition.findMany({ + where: { + isActive: true, + scheduleEnabled: true, + }, + }); + return found.map(ReportDefinitionMapper.toDomain); + } + + async findAll(): Promise { + const found = await this.prisma.reportDefinition.findMany(); + return found.map(ReportDefinitionMapper.toDomain); + } + + async delete(id: bigint): Promise { + await this.prisma.reportDefinition.delete({ + where: { id }, + }); + } +} diff --git a/backend/services/reporting-service/src/infrastructure/persistence/repositories/report-file.repository.impl.ts b/backend/services/reporting-service/src/infrastructure/persistence/repositories/report-file.repository.impl.ts new file mode 100644 index 00000000..e5c6eeb9 --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/persistence/repositories/report-file.repository.impl.ts @@ -0,0 +1,101 @@ +import { Injectable } from '@nestjs/common'; +import { PrismaService } from '../prisma/prisma.service'; +import { ReportFile } from '../../../domain/entities/report-file.entity'; +import { IReportFileRepository } from '../../../domain/repositories'; +import { OutputFormat } from '../../../domain/value-objects'; +import { ReportFileMapper } from '../mappers/report-file.mapper'; + +@Injectable() +export class ReportFileRepository implements IReportFileRepository { + constructor(private readonly prisma: PrismaService) {} + + async save(file: ReportFile): Promise { + const data = ReportFileMapper.toPersistence(file); + + if (file.id) { + const updated = await this.prisma.reportFile.update({ + where: { id: file.id }, + data: { + fileUrl: data.fileUrl, + downloadCount: data.downloadCount, + lastDownloadAt: data.lastDownloadAt, + expiresAt: data.expiresAt, + }, + }); + return ReportFileMapper.toDomain(updated); + } + + const created = await this.prisma.reportFile.create({ + data: { + snapshotId: data.snapshotId, + fileName: data.fileName, + filePath: data.filePath, + fileUrl: data.fileUrl, + fileSize: data.fileSize, + fileFormat: data.fileFormat, + mimeType: data.mimeType, + downloadCount: data.downloadCount, + lastDownloadAt: data.lastDownloadAt, + expiresAt: data.expiresAt, + }, + }); + return ReportFileMapper.toDomain(created); + } + + async findById(id: bigint): Promise { + const found = await this.prisma.reportFile.findUnique({ + where: { id }, + }); + return found ? ReportFileMapper.toDomain(found) : null; + } + + async findBySnapshotId(snapshotId: bigint): Promise { + const found = await this.prisma.reportFile.findMany({ + where: { snapshotId }, + orderBy: { createdAt: 'desc' }, + }); + return found.map(ReportFileMapper.toDomain); + } + + async findBySnapshotIdAndFormat( + snapshotId: bigint, + format: OutputFormat, + ): Promise { + const found = await this.prisma.reportFile.findFirst({ + where: { + snapshotId, + fileFormat: format, + }, + orderBy: { createdAt: 'desc' }, + }); + return found ? ReportFileMapper.toDomain(found) : null; + } + + async findExpired(): Promise { + const found = await this.prisma.reportFile.findMany({ + where: { + expiresAt: { + lt: new Date(), + }, + }, + }); + return found.map(ReportFileMapper.toDomain); + } + + async deleteExpired(): Promise { + const result = await this.prisma.reportFile.deleteMany({ + where: { + expiresAt: { + lt: new Date(), + }, + }, + }); + return result.count; + } + + async delete(id: bigint): Promise { + await this.prisma.reportFile.delete({ + where: { id }, + }); + } +} diff --git a/backend/services/reporting-service/src/infrastructure/persistence/repositories/report-snapshot.repository.impl.ts b/backend/services/reporting-service/src/infrastructure/persistence/repositories/report-snapshot.repository.impl.ts new file mode 100644 index 00000000..ae444764 --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/persistence/repositories/report-snapshot.repository.impl.ts @@ -0,0 +1,133 @@ +import { Injectable } from '@nestjs/common'; +import { Prisma } from '@prisma/client'; +import { PrismaService } from '../prisma/prisma.service'; +import { ReportSnapshot } from '../../../domain/aggregates/report-snapshot'; +import { IReportSnapshotRepository } from '../../../domain/repositories'; +import { ReportType, ReportPeriod } from '../../../domain/value-objects'; +import { ReportSnapshotMapper } from '../mappers/report-snapshot.mapper'; + +@Injectable() +export class ReportSnapshotRepository implements IReportSnapshotRepository { + constructor(private readonly prisma: PrismaService) {} + + async save(snapshot: ReportSnapshot): Promise { + const data = ReportSnapshotMapper.toPersistence(snapshot); + + if (snapshot.id) { + const updated = await this.prisma.reportSnapshot.update({ + where: { id: snapshot.id }, + data: { + snapshotData: data.snapshotData as Prisma.InputJsonValue, + summaryData: data.summaryData as Prisma.InputJsonValue | undefined, + dataSources: data.dataSources, + dataFreshness: data.dataFreshness, + filterParams: data.filterParams as Prisma.InputJsonValue | undefined, + rowCount: data.rowCount, + expiresAt: data.expiresAt, + }, + }); + return ReportSnapshotMapper.toDomain(updated); + } + + const created = await this.prisma.reportSnapshot.create({ + data: { + reportType: data.reportType, + reportCode: data.reportCode, + reportPeriod: data.reportPeriod, + periodKey: data.periodKey, + snapshotData: data.snapshotData as Prisma.InputJsonValue, + summaryData: data.summaryData as Prisma.InputJsonValue | undefined, + dataSources: data.dataSources, + dataFreshness: data.dataFreshness, + filterParams: data.filterParams as Prisma.InputJsonValue | undefined, + rowCount: data.rowCount, + periodStartAt: data.periodStartAt, + periodEndAt: data.periodEndAt, + expiresAt: data.expiresAt, + }, + }); + return ReportSnapshotMapper.toDomain(created); + } + + async findById(id: bigint): Promise { + const found = await this.prisma.reportSnapshot.findUnique({ + where: { id }, + }); + return found ? ReportSnapshotMapper.toDomain(found) : null; + } + + async findByCodeAndPeriodKey( + reportCode: string, + periodKey: string, + ): Promise { + const found = await this.prisma.reportSnapshot.findUnique({ + where: { + uk_report_period: { + reportCode, + periodKey, + }, + }, + }); + return found ? ReportSnapshotMapper.toDomain(found) : null; + } + + async findByType(type: ReportType): Promise { + const found = await this.prisma.reportSnapshot.findMany({ + where: { reportType: type }, + orderBy: { generatedAt: 'desc' }, + }); + return found.map(ReportSnapshotMapper.toDomain); + } + + async findByCode(reportCode: string): Promise { + const found = await this.prisma.reportSnapshot.findMany({ + where: { reportCode }, + orderBy: { generatedAt: 'desc' }, + }); + return found.map(ReportSnapshotMapper.toDomain); + } + + async findByPeriod(period: ReportPeriod): Promise { + const found = await this.prisma.reportSnapshot.findMany({ + where: { reportPeriod: period }, + orderBy: { generatedAt: 'desc' }, + }); + return found.map(ReportSnapshotMapper.toDomain); + } + + async findLatestByCode(reportCode: string): Promise { + const found = await this.prisma.reportSnapshot.findFirst({ + where: { reportCode }, + orderBy: { generatedAt: 'desc' }, + }); + return found ? ReportSnapshotMapper.toDomain(found) : null; + } + + async findExpired(): Promise { + const found = await this.prisma.reportSnapshot.findMany({ + where: { + expiresAt: { + lt: new Date(), + }, + }, + }); + return found.map(ReportSnapshotMapper.toDomain); + } + + async deleteExpired(): Promise { + const result = await this.prisma.reportSnapshot.deleteMany({ + where: { + expiresAt: { + lt: new Date(), + }, + }, + }); + return result.count; + } + + async delete(id: bigint): Promise { + await this.prisma.reportSnapshot.delete({ + where: { id }, + }); + } +} diff --git a/backend/services/reporting-service/src/infrastructure/redis/redis.module.ts b/backend/services/reporting-service/src/infrastructure/redis/redis.module.ts new file mode 100644 index 00000000..4b7ae129 --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/redis/redis.module.ts @@ -0,0 +1,9 @@ +import { Module } from '@nestjs/common'; +import { RedisService } from './redis.service'; +import { ReportCacheService } from './report-cache.service'; + +@Module({ + providers: [RedisService, ReportCacheService], + exports: [RedisService, ReportCacheService], +}) +export class RedisModule {} diff --git a/backend/services/reporting-service/src/infrastructure/redis/redis.service.ts b/backend/services/reporting-service/src/infrastructure/redis/redis.service.ts new file mode 100644 index 00000000..9c2b906c --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/redis/redis.service.ts @@ -0,0 +1,72 @@ +import { Injectable, OnModuleDestroy, Logger } from '@nestjs/common'; +import { ConfigService } from '@nestjs/config'; +import Redis from 'ioredis'; + +@Injectable() +export class RedisService implements OnModuleDestroy { + private readonly logger = new Logger(RedisService.name); + private readonly 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') || undefined, + retryStrategy: (times) => { + const delay = Math.min(times * 50, 2000); + return delay; + }, + }); + + this.client.on('connect', () => { + this.logger.log('Redis connected'); + }); + + this.client.on('error', (err) => { + this.logger.error('Redis error', err); + }); + } + + async onModuleDestroy() { + await this.client.quit(); + } + + async get(key: string): Promise { + return this.client.get(key); + } + + async set(key: string, value: string, ttlSeconds?: number): Promise { + if (ttlSeconds) { + await this.client.setex(key, ttlSeconds, value); + } else { + await this.client.set(key, value); + } + } + + async del(key: string): Promise { + await this.client.del(key); + } + + async exists(key: string): Promise { + const result = await this.client.exists(key); + return result === 1; + } + + async keys(pattern: string): Promise { + return this.client.keys(pattern); + } + + async getJson(key: string): Promise { + const value = await this.get(key); + if (!value) return null; + try { + return JSON.parse(value) as T; + } catch { + return null; + } + } + + async setJson(key: string, value: T, ttlSeconds?: number): Promise { + await this.set(key, JSON.stringify(value), ttlSeconds); + } +} diff --git a/backend/services/reporting-service/src/infrastructure/redis/report-cache.service.ts b/backend/services/reporting-service/src/infrastructure/redis/report-cache.service.ts new file mode 100644 index 00000000..3ab54588 --- /dev/null +++ b/backend/services/reporting-service/src/infrastructure/redis/report-cache.service.ts @@ -0,0 +1,59 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { ConfigService } from '@nestjs/config'; +import { RedisService } from './redis.service'; + +@Injectable() +export class ReportCacheService { + private readonly logger = new Logger(ReportCacheService.name); + private readonly cachePrefix = 'report:'; + private readonly defaultTtl: number; + + constructor( + private readonly redisService: RedisService, + private readonly configService: ConfigService, + ) { + this.defaultTtl = this.configService.get('REPORT_CACHE_TTL', 3600); + } + + async cacheSnapshot( + reportCode: string, + periodKey: string, + data: any, + ttlSeconds?: number, + ): Promise { + const key = this.buildKey(reportCode, periodKey); + await this.redisService.setJson(key, data, ttlSeconds || this.defaultTtl); + this.logger.debug(`Cached report snapshot: ${key}`); + } + + async getCachedSnapshot( + reportCode: string, + periodKey: string, + ): Promise { + const key = this.buildKey(reportCode, periodKey); + const cached = await this.redisService.getJson(key); + if (cached) { + this.logger.debug(`Cache hit for report snapshot: ${key}`); + } + return cached; + } + + async invalidateSnapshot(reportCode: string, periodKey: string): Promise { + const key = this.buildKey(reportCode, periodKey); + await this.redisService.del(key); + this.logger.debug(`Invalidated report cache: ${key}`); + } + + async invalidateReportCache(reportCode: string): Promise { + const pattern = `${this.cachePrefix}${reportCode}:*`; + const keys = await this.redisService.keys(pattern); + for (const key of keys) { + await this.redisService.del(key); + } + this.logger.debug(`Invalidated all cache for report: ${reportCode}`); + } + + private buildKey(reportCode: string, periodKey: string): string { + return `${this.cachePrefix}${reportCode}:${periodKey}`; + } +} diff --git a/backend/services/reporting-service/src/main.ts b/backend/services/reporting-service/src/main.ts new file mode 100644 index 00000000..5e7db987 --- /dev/null +++ b/backend/services/reporting-service/src/main.ts @@ -0,0 +1,53 @@ +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); + + // Global prefix + app.setGlobalPrefix('api/v1'); + + // CORS + app.enableCors({ + origin: true, + credentials: true, + }); + + // Validation pipe + app.useGlobalPipes( + new ValidationPipe({ + whitelist: true, + forbidNonWhitelisted: true, + transform: true, + transformOptions: { + enableImplicitConversion: true, + }, + }), + ); + + // Swagger setup + const config = new DocumentBuilder() + .setTitle('Reporting & Analytics Service') + .setDescription('RWA Durian Platform - Reporting & Analytics API') + .setVersion('1.0') + .addBearerAuth() + .addTag('Health', '健康检查') + .addTag('Reports', '报表管理') + .addTag('Export', '报表导出') + .build(); + + const document = SwaggerModule.createDocument(app, config); + SwaggerModule.setup('api/docs', app, document); + + const port = process.env.PORT || 3008; + await app.listen(port); + + logger.log(`🚀 Reporting Service is running on: http://localhost:${port}`); + logger.log(`📚 Swagger API docs: http://localhost:${port}/api/docs`); +} + +bootstrap(); diff --git a/backend/services/reporting-service/src/shared/decorators/current-user.decorator.ts b/backend/services/reporting-service/src/shared/decorators/current-user.decorator.ts new file mode 100644 index 00000000..42088952 --- /dev/null +++ b/backend/services/reporting-service/src/shared/decorators/current-user.decorator.ts @@ -0,0 +1,14 @@ +import { createParamDecorator, ExecutionContext } from '@nestjs/common'; + +export interface CurrentUserPayload { + userId: bigint; + email: string; + roles: string[]; +} + +export const CurrentUser = createParamDecorator( + (data: unknown, ctx: ExecutionContext): CurrentUserPayload | null => { + const request = ctx.switchToHttp().getRequest(); + return request.user || null; + }, +); diff --git a/backend/services/reporting-service/src/shared/decorators/index.ts b/backend/services/reporting-service/src/shared/decorators/index.ts new file mode 100644 index 00000000..8760ff1d --- /dev/null +++ b/backend/services/reporting-service/src/shared/decorators/index.ts @@ -0,0 +1,2 @@ +export * from './public.decorator'; +export * from './current-user.decorator'; diff --git a/backend/services/reporting-service/src/shared/decorators/public.decorator.ts b/backend/services/reporting-service/src/shared/decorators/public.decorator.ts new file mode 100644 index 00000000..b3845e12 --- /dev/null +++ b/backend/services/reporting-service/src/shared/decorators/public.decorator.ts @@ -0,0 +1,4 @@ +import { SetMetadata } from '@nestjs/common'; + +export const IS_PUBLIC_KEY = 'isPublic'; +export const Public = () => SetMetadata(IS_PUBLIC_KEY, true); diff --git a/backend/services/reporting-service/src/shared/exceptions/domain.exception.ts b/backend/services/reporting-service/src/shared/exceptions/domain.exception.ts new file mode 100644 index 00000000..5e6a6105 --- /dev/null +++ b/backend/services/reporting-service/src/shared/exceptions/domain.exception.ts @@ -0,0 +1,23 @@ +export class DomainException extends Error { + constructor( + message: string, + public readonly code?: string, + ) { + super(message); + this.name = 'DomainException'; + } +} + +export class EntityNotFoundException extends DomainException { + constructor(entityName: string, id: string | bigint) { + super(`${entityName} not found: ${id}`, 'ENTITY_NOT_FOUND'); + this.name = 'EntityNotFoundException'; + } +} + +export class InvalidOperationException extends DomainException { + constructor(message: string) { + super(message, 'INVALID_OPERATION'); + this.name = 'InvalidOperationException'; + } +} diff --git a/backend/services/reporting-service/src/shared/exceptions/index.ts b/backend/services/reporting-service/src/shared/exceptions/index.ts new file mode 100644 index 00000000..ca0c7823 --- /dev/null +++ b/backend/services/reporting-service/src/shared/exceptions/index.ts @@ -0,0 +1 @@ +export * from './domain.exception'; diff --git a/backend/services/reporting-service/src/shared/filters/global-exception.filter.ts b/backend/services/reporting-service/src/shared/filters/global-exception.filter.ts new file mode 100644 index 00000000..def9d880 --- /dev/null +++ b/backend/services/reporting-service/src/shared/filters/global-exception.filter.ts @@ -0,0 +1,52 @@ +import { + ExceptionFilter, + Catch, + ArgumentsHost, + HttpException, + HttpStatus, + Logger, +} from '@nestjs/common'; +import { Response } from 'express'; +import { DomainException } from '../exceptions/domain.exception'; + +@Catch() +export class GlobalExceptionFilter implements ExceptionFilter { + private readonly logger = new Logger(GlobalExceptionFilter.name); + + catch(exception: unknown, host: ArgumentsHost) { + const ctx = host.switchToHttp(); + const response = ctx.getResponse(); + + let status = HttpStatus.INTERNAL_SERVER_ERROR; + let message = 'Internal server error'; + let code = 'INTERNAL_ERROR'; + + if (exception instanceof HttpException) { + status = exception.getStatus(); + const exceptionResponse = exception.getResponse(); + message = + typeof exceptionResponse === 'string' + ? exceptionResponse + : (exceptionResponse as any).message || exception.message; + code = (exceptionResponse as any).error || 'HTTP_ERROR'; + } else if (exception instanceof DomainException) { + status = HttpStatus.BAD_REQUEST; + message = exception.message; + code = exception.code || 'DOMAIN_ERROR'; + } else if (exception instanceof Error) { + message = exception.message; + } + + this.logger.error( + `Exception: ${message}`, + exception instanceof Error ? exception.stack : undefined, + ); + + response.status(status).json({ + statusCode: status, + code, + message, + timestamp: new Date().toISOString(), + }); + } +} diff --git a/backend/services/reporting-service/src/shared/guards/jwt-auth.guard.ts b/backend/services/reporting-service/src/shared/guards/jwt-auth.guard.ts new file mode 100644 index 00000000..f75ddaa2 --- /dev/null +++ b/backend/services/reporting-service/src/shared/guards/jwt-auth.guard.ts @@ -0,0 +1,24 @@ +import { Injectable, ExecutionContext } from '@nestjs/common'; +import { AuthGuard } from '@nestjs/passport'; +import { Reflector } from '@nestjs/core'; +import { IS_PUBLIC_KEY } from '../decorators/public.decorator'; + +@Injectable() +export class JwtAuthGuard extends AuthGuard('jwt') { + constructor(private reflector: Reflector) { + super(); + } + + canActivate(context: ExecutionContext) { + const isPublic = this.reflector.getAllAndOverride(IS_PUBLIC_KEY, [ + context.getHandler(), + context.getClass(), + ]); + + if (isPublic) { + return true; + } + + return super.canActivate(context); + } +} diff --git a/backend/services/reporting-service/src/shared/interceptors/transform.interceptor.ts b/backend/services/reporting-service/src/shared/interceptors/transform.interceptor.ts new file mode 100644 index 00000000..403c1cbb --- /dev/null +++ b/backend/services/reporting-service/src/shared/interceptors/transform.interceptor.ts @@ -0,0 +1,32 @@ +import { + Injectable, + NestInterceptor, + ExecutionContext, + CallHandler, +} from '@nestjs/common'; +import { Observable } from 'rxjs'; +import { map } from 'rxjs/operators'; + +export interface ApiResponse { + success: boolean; + data: T; + timestamp: string; +} + +@Injectable() +export class TransformInterceptor + implements NestInterceptor> +{ + intercept( + context: ExecutionContext, + next: CallHandler, + ): Observable> { + return next.handle().pipe( + map((data) => ({ + success: true, + data, + timestamp: new Date().toISOString(), + })), + ); + } +} diff --git a/backend/services/reporting-service/src/shared/strategies/jwt.strategy.ts b/backend/services/reporting-service/src/shared/strategies/jwt.strategy.ts new file mode 100644 index 00000000..04d01de2 --- /dev/null +++ b/backend/services/reporting-service/src/shared/strategies/jwt.strategy.ts @@ -0,0 +1,31 @@ +import { Injectable } from '@nestjs/common'; +import { PassportStrategy } from '@nestjs/passport'; +import { ExtractJwt, Strategy } from 'passport-jwt'; +import { ConfigService } from '@nestjs/config'; + +interface JwtPayload { + sub: string; + email: string; + roles: string[]; + iat: number; + exp: number; +} + +@Injectable() +export class JwtStrategy extends PassportStrategy(Strategy) { + constructor(private readonly configService: ConfigService) { + super({ + jwtFromRequest: ExtractJwt.fromAuthHeaderAsBearerToken(), + ignoreExpiration: false, + secretOrKey: configService.get('JWT_SECRET'), + }); + } + + async validate(payload: JwtPayload) { + return { + userId: BigInt(payload.sub), + email: payload.email, + roles: payload.roles, + }; + } +} diff --git a/backend/services/reporting-service/test/app.e2e-spec.ts b/backend/services/reporting-service/test/app.e2e-spec.ts new file mode 100644 index 00000000..5ecdfff8 --- /dev/null +++ b/backend/services/reporting-service/test/app.e2e-spec.ts @@ -0,0 +1,174 @@ +import { Test, TestingModule } from '@nestjs/testing'; +import { INestApplication, ValidationPipe } from '@nestjs/common'; +import * as request from 'supertest'; +import { AppModule } from '../src/app.module'; +import { ReportPeriod } from '../src/domain/value-objects'; + +describe('Reporting Service (e2e)', () => { + let app: INestApplication; + + beforeAll(async () => { + const moduleFixture: TestingModule = await Test.createTestingModule({ + imports: [AppModule], + }).compile(); + + app = moduleFixture.createNestApplication(); + app.setGlobalPrefix('api/v1'); + app.useGlobalPipes( + new ValidationPipe({ + whitelist: true, + forbidNonWhitelisted: true, + transform: true, + transformOptions: { + enableImplicitConversion: true, + }, + }), + ); + // TransformInterceptor and GlobalExceptionFilter are already registered in AppModule + // Don't add them again to avoid double-wrapping responses + await app.init(); + }); + + afterAll(async () => { + await app.close(); + }); + + describe('Health Check', () => { + it('/api/v1/health (GET) should return ok status', () => { + return request(app.getHttpServer()) + .get('/api/v1/health') + .expect(200) + .expect((res) => { + // Response is wrapped by TransformInterceptor: { success: true, data: { ... }, timestamp: ... } + expect(res.body.success).toBe(true); + expect(res.body.data).toBeDefined(); + expect(res.body.data.status).toBe('ok'); + expect(res.body.data.service).toBe('reporting-service'); + }); + }); + + it('/api/v1/health/ready (GET) should return ready status', () => { + return request(app.getHttpServer()) + .get('/api/v1/health/ready') + .expect(200) + .expect((res) => { + expect(res.body.success).toBe(true); + expect(res.body.data).toBeDefined(); + expect(res.body.data.status).toBe('ready'); + }); + }); + }); + + describe('Reports API', () => { + describe('GET /api/v1/reports/definitions', () => { + it('should return list of report definitions', () => { + return request(app.getHttpServer()) + .get('/api/v1/reports/definitions') + .expect(200) + .expect((res) => { + expect(res.body.success).toBe(true); + expect(res.body.data).toBeDefined(); + expect(Array.isArray(res.body.data)).toBe(true); + }); + }); + }); + + describe('GET /api/v1/reports/definitions/:code', () => { + it('should return 404 for non-existent report code', () => { + return request(app.getHttpServer()) + .get('/api/v1/reports/definitions/NON_EXISTENT_CODE') + .expect(404) + .expect((res) => { + expect(res.body.statusCode).toBe(404); + }); + }); + }); + + describe('POST /api/v1/reports/generate', () => { + it('should validate request body', () => { + return request(app.getHttpServer()) + .post('/api/v1/reports/generate') + .send({}) + .expect(400); + }); + + it('should reject invalid report period', () => { + return request(app.getHttpServer()) + .post('/api/v1/reports/generate') + .send({ + reportCode: 'RPT_LEADERBOARD', + reportPeriod: 'INVALID_PERIOD', + startDate: '2024-01-01', + endDate: '2024-01-31', + }) + .expect(400); + }); + + it('should return 404 for non-existent report code', () => { + return request(app.getHttpServer()) + .post('/api/v1/reports/generate') + .send({ + reportCode: 'NON_EXISTENT', + reportPeriod: ReportPeriod.DAILY, + startDate: '2024-01-01', + endDate: '2024-01-01', + }) + .expect(404); + }); + }); + + describe('GET /api/v1/reports/snapshots', () => { + it('should return empty array when no report code provided', () => { + return request(app.getHttpServer()) + .get('/api/v1/reports/snapshots') + .expect(200) + .expect((res) => { + expect(res.body.success).toBe(true); + expect(res.body.data).toBeDefined(); + expect(Array.isArray(res.body.data)).toBe(true); + }); + }); + + it('should filter snapshots by report code', () => { + return request(app.getHttpServer()) + .get('/api/v1/reports/snapshots') + .query({ reportCode: 'RPT_LEADERBOARD' }) + .expect(200) + .expect((res) => { + expect(res.body.success).toBe(true); + expect(res.body.data).toBeDefined(); + expect(Array.isArray(res.body.data)).toBe(true); + }); + }); + }); + + describe('GET /api/v1/reports/snapshots/:code/latest', () => { + it('should return 404 when no snapshots exist', () => { + return request(app.getHttpServer()) + .get('/api/v1/reports/snapshots/RPT_LEADERBOARD/latest') + .expect(404); + }); + }); + }); + + describe('Export API', () => { + describe('POST /api/v1/export', () => { + it('should validate request body', () => { + return request(app.getHttpServer()) + .post('/api/v1/export') + .send({}) + .expect(400); + }); + + it('should reject invalid format', () => { + return request(app.getHttpServer()) + .post('/api/v1/export') + .send({ + snapshotId: '1', + format: 'INVALID_FORMAT', + }) + .expect(400); + }); + }); + }); +}); diff --git a/backend/services/reporting-service/test/integration/report-definition.repository.integration.spec.ts b/backend/services/reporting-service/test/integration/report-definition.repository.integration.spec.ts new file mode 100644 index 00000000..d391851e --- /dev/null +++ b/backend/services/reporting-service/test/integration/report-definition.repository.integration.spec.ts @@ -0,0 +1,171 @@ +import { Test, TestingModule } from '@nestjs/testing'; +import { ConfigModule } from '@nestjs/config'; +import { PrismaService } from '../../src/infrastructure/persistence/prisma/prisma.service'; +import { ReportDefinitionRepository } from '../../src/infrastructure/persistence/repositories/report-definition.repository.impl'; +import { ReportDefinition } from '../../src/domain/aggregates/report-definition'; +import { ReportType, OutputFormat, ReportSchedule } from '../../src/domain/value-objects'; + +describe('ReportDefinitionRepository Integration', () => { + let module: TestingModule; + let repository: ReportDefinitionRepository; + let prisma: PrismaService; + + beforeAll(async () => { + module = await Test.createTestingModule({ + imports: [ + ConfigModule.forRoot({ + isGlobal: true, + envFilePath: ['.env.test', '.env'], + }), + ], + providers: [PrismaService, ReportDefinitionRepository], + }).compile(); + + repository = module.get(ReportDefinitionRepository); + prisma = module.get(PrismaService); + }); + + afterAll(async () => { + await prisma.$disconnect(); + await module.close(); + }); + + beforeEach(async () => { + // Clean up test data + await prisma.reportDefinition.deleteMany({ + where: { reportCode: { startsWith: 'TEST_' } }, + }); + }); + + describe('save', () => { + it('should create a new report definition', async () => { + const definition = ReportDefinition.create({ + reportType: ReportType.LEADERBOARD_REPORT, + reportName: 'Test Report', + reportCode: 'TEST_REPO_001', + description: 'Integration test report', + outputFormats: [OutputFormat.EXCEL, OutputFormat.CSV], + }); + + const saved = await repository.save(definition); + + expect(saved.id).toBeDefined(); + expect(saved.reportCode).toBe('TEST_REPO_001'); + expect(saved.reportName).toBe('Test Report'); + expect(saved.outputFormats).toContain(OutputFormat.EXCEL); + }); + + it('should update an existing report definition', async () => { + // Create first + const definition = ReportDefinition.create({ + reportType: ReportType.PLANTING_REPORT, + reportName: 'Original Name', + reportCode: 'TEST_REPO_002', + }); + + const saved = await repository.save(definition); + + // Update + saved.updateName('Updated Name'); + saved.updateDescription('Updated description'); + const updated = await repository.save(saved); + + expect(updated.reportName).toBe('Updated Name'); + expect(updated.description).toBe('Updated description'); + }); + }); + + describe('findByCode', () => { + it('should find definition by code', async () => { + const definition = ReportDefinition.create({ + reportType: ReportType.COMMUNITY_REPORT, + reportName: 'Find Test', + reportCode: 'TEST_REPO_003', + }); + + await repository.save(definition); + + const found = await repository.findByCode('TEST_REPO_003'); + + expect(found).not.toBeNull(); + expect(found?.reportName).toBe('Find Test'); + }); + + it('should return null for non-existent code', async () => { + const found = await repository.findByCode('NON_EXISTENT_CODE'); + expect(found).toBeNull(); + }); + }); + + describe('findActive', () => { + it('should find all active definitions', async () => { + // Create active definition + const active = ReportDefinition.create({ + reportType: ReportType.LEADERBOARD_REPORT, + reportName: 'Active Report', + reportCode: 'TEST_REPO_004', + }); + await repository.save(active); + + // Create inactive definition + const inactive = ReportDefinition.create({ + reportType: ReportType.LEADERBOARD_REPORT, + reportName: 'Inactive Report', + reportCode: 'TEST_REPO_005', + }); + inactive.deactivate(); + await repository.save(inactive); + + const activeDefinitions = await repository.findActive(); + const testDefinitions = activeDefinitions.filter(d => d.reportCode.startsWith('TEST_')); + + expect(testDefinitions.length).toBe(1); + expect(testDefinitions[0].reportCode).toBe('TEST_REPO_004'); + }); + }); + + describe('findScheduled', () => { + it('should find scheduled definitions', async () => { + // Create scheduled definition + const scheduled = ReportDefinition.create({ + reportType: ReportType.PLANTING_REPORT, + reportName: 'Scheduled Report', + reportCode: 'TEST_REPO_006', + schedule: ReportSchedule.daily(1, 0), + }); + await repository.save(scheduled); + + // Create non-scheduled definition + const nonScheduled = ReportDefinition.create({ + reportType: ReportType.PLANTING_REPORT, + reportName: 'Non-Scheduled Report', + reportCode: 'TEST_REPO_007', + }); + await repository.save(nonScheduled); + + const scheduledDefinitions = await repository.findScheduled(); + const testDefinitions = scheduledDefinitions.filter(d => d.reportCode.startsWith('TEST_')); + + expect(testDefinitions.length).toBe(1); + expect(testDefinitions[0].reportCode).toBe('TEST_REPO_006'); + }); + }); + + describe('delete', () => { + it('should delete a definition', async () => { + const definition = ReportDefinition.create({ + reportType: ReportType.LEADERBOARD_REPORT, + reportName: 'To Delete', + reportCode: 'TEST_REPO_008', + }); + + const saved = await repository.save(definition); + expect(saved.id).toBeDefined(); + + await repository.delete(saved.id!); + + const found = await repository.findByCode('TEST_REPO_008'); + expect(found).toBeNull(); + }); + }); +}); diff --git a/backend/services/reporting-service/test/integration/report-snapshot.repository.integration.spec.ts b/backend/services/reporting-service/test/integration/report-snapshot.repository.integration.spec.ts new file mode 100644 index 00000000..73a48316 --- /dev/null +++ b/backend/services/reporting-service/test/integration/report-snapshot.repository.integration.spec.ts @@ -0,0 +1,172 @@ +import { Test, TestingModule } from '@nestjs/testing'; +import { ConfigModule } from '@nestjs/config'; +import { PrismaService } from '../../src/infrastructure/persistence/prisma/prisma.service'; +import { ReportSnapshotRepository } from '../../src/infrastructure/persistence/repositories/report-snapshot.repository.impl'; +import { ReportSnapshot } from '../../src/domain/aggregates/report-snapshot'; +import { + ReportType, + ReportPeriod, + SnapshotData, + DataSource, +} from '../../src/domain/value-objects'; + +describe('ReportSnapshotRepository Integration', () => { + let module: TestingModule; + let repository: ReportSnapshotRepository; + let prisma: PrismaService; + + beforeAll(async () => { + module = await Test.createTestingModule({ + imports: [ + ConfigModule.forRoot({ + isGlobal: true, + envFilePath: ['.env.test', '.env'], + }), + ], + providers: [PrismaService, ReportSnapshotRepository], + }).compile(); + + repository = module.get(ReportSnapshotRepository); + prisma = module.get(PrismaService); + }); + + afterAll(async () => { + await prisma.$disconnect(); + await module.close(); + }); + + beforeEach(async () => { + // Clean up test data + await prisma.reportSnapshot.deleteMany({ + where: { reportCode: { startsWith: 'TEST_' } }, + }); + }); + + describe('save', () => { + it('should create a new report snapshot', async () => { + const snapshotData = SnapshotData.create({ + rows: [ + { rank: 1, userId: 1, score: 100 }, + { rank: 2, userId: 2, score: 90 }, + ], + summary: { totalEntries: 2 }, + }); + + const dataSource = DataSource.create(['test-service']); + + const snapshot = ReportSnapshot.create({ + reportType: ReportType.LEADERBOARD_REPORT, + reportCode: 'TEST_SNAP_001', + reportPeriod: ReportPeriod.DAILY, + periodKey: '2024-01-15', + snapshotData, + dataSource, + periodStartAt: new Date('2024-01-15'), + periodEndAt: new Date('2024-01-15'), + }); + + const saved = await repository.save(snapshot); + + expect(saved.id).toBeDefined(); + expect(saved.reportCode).toBe('TEST_SNAP_001'); + expect(saved.rowCount).toBe(2); + }); + }); + + describe('findByCodeAndPeriodKey', () => { + it('should find snapshot by code and period key', async () => { + const snapshotData = SnapshotData.create({ + rows: [{ data: 'test' }], + }); + + const snapshot = ReportSnapshot.create({ + reportType: ReportType.PLANTING_REPORT, + reportCode: 'TEST_SNAP_002', + reportPeriod: ReportPeriod.MONTHLY, + periodKey: '2024-01', + snapshotData, + dataSource: DataSource.create(['test']), + periodStartAt: new Date('2024-01-01'), + periodEndAt: new Date('2024-01-31'), + }); + + await repository.save(snapshot); + + const found = await repository.findByCodeAndPeriodKey('TEST_SNAP_002', '2024-01'); + + expect(found).not.toBeNull(); + expect(found?.reportPeriod).toBe(ReportPeriod.MONTHLY); + }); + }); + + describe('findLatestByCode', () => { + it('should find the latest snapshot by code', async () => { + // Create older snapshot + const older = ReportSnapshot.create({ + reportType: ReportType.PLANTING_REPORT, + reportCode: 'TEST_SNAP_003', + reportPeriod: ReportPeriod.DAILY, + periodKey: '2024-01-14', + snapshotData: SnapshotData.create({ rows: [] }), + dataSource: DataSource.create(['test']), + periodStartAt: new Date('2024-01-14'), + periodEndAt: new Date('2024-01-14'), + }); + await repository.save(older); + + // Create newer snapshot + const newer = ReportSnapshot.create({ + reportType: ReportType.PLANTING_REPORT, + reportCode: 'TEST_SNAP_003', + reportPeriod: ReportPeriod.DAILY, + periodKey: '2024-01-15', + snapshotData: SnapshotData.create({ rows: [{ latest: true }] }), + dataSource: DataSource.create(['test']), + periodStartAt: new Date('2024-01-15'), + periodEndAt: new Date('2024-01-15'), + }); + await repository.save(newer); + + const latest = await repository.findLatestByCode('TEST_SNAP_003'); + + expect(latest).not.toBeNull(); + expect(latest?.periodKey).toBe('2024-01-15'); + }); + }); + + describe('findByPeriod', () => { + it('should find snapshots by period type', async () => { + // Create daily snapshot + const daily = ReportSnapshot.create({ + reportType: ReportType.PLANTING_REPORT, + reportCode: 'TEST_SNAP_004', + reportPeriod: ReportPeriod.DAILY, + periodKey: '2024-01-15', + snapshotData: SnapshotData.create({ rows: [] }), + dataSource: DataSource.create(['test']), + periodStartAt: new Date('2024-01-15'), + periodEndAt: new Date('2024-01-15'), + }); + await repository.save(daily); + + // Create monthly snapshot + const monthly = ReportSnapshot.create({ + reportType: ReportType.PLANTING_REPORT, + reportCode: 'TEST_SNAP_005', + reportPeriod: ReportPeriod.MONTHLY, + periodKey: '2024-01', + snapshotData: SnapshotData.create({ rows: [] }), + dataSource: DataSource.create(['test']), + periodStartAt: new Date('2024-01-01'), + periodEndAt: new Date('2024-01-31'), + }); + await repository.save(monthly); + + const dailySnapshots = await repository.findByPeriod(ReportPeriod.DAILY); + const testDailySnapshots = dailySnapshots.filter(s => s.reportCode.startsWith('TEST_')); + + expect(testDailySnapshots.length).toBe(1); + expect(testDailySnapshots[0].reportCode).toBe('TEST_SNAP_004'); + }); + }); +}); diff --git a/backend/services/reporting-service/test/jest-e2e.json b/backend/services/reporting-service/test/jest-e2e.json new file mode 100644 index 00000000..a6d62daa --- /dev/null +++ b/backend/services/reporting-service/test/jest-e2e.json @@ -0,0 +1,20 @@ +{ + "moduleFileExtensions": ["js", "json", "ts"], + "rootDir": ".", + "testEnvironment": "node", + "testRegex": ".e2e-spec.ts$", + "testTimeout": 30000, + "setupFilesAfterEnv": ["/setup-e2e.ts"], + "transform": { + "^.+\\.(t|j)s$": "ts-jest" + }, + "moduleNameMapper": { + "^@/(.*)$": "/../src/$1", + "^@domain/(.*)$": "/../src/domain/$1", + "^@application/(.*)$": "/../src/application/$1", + "^@infrastructure/(.*)$": "/../src/infrastructure/$1", + "^@api/(.*)$": "/../src/api/$1", + "^@shared/(.*)$": "/../src/shared/$1", + "^@config/(.*)$": "/../src/config/$1" + } +} diff --git a/backend/services/reporting-service/test/setup-e2e.ts b/backend/services/reporting-service/test/setup-e2e.ts new file mode 100644 index 00000000..955bdbba --- /dev/null +++ b/backend/services/reporting-service/test/setup-e2e.ts @@ -0,0 +1,5 @@ +import * as dotenv from 'dotenv'; +import * as path from 'path'; + +// Load test environment variables +dotenv.config({ path: path.resolve(__dirname, '../.env.test') }); diff --git a/backend/services/reporting-service/tsconfig.build.json b/backend/services/reporting-service/tsconfig.build.json new file mode 100644 index 00000000..64f86c6b --- /dev/null +++ b/backend/services/reporting-service/tsconfig.build.json @@ -0,0 +1,4 @@ +{ + "extends": "./tsconfig.json", + "exclude": ["node_modules", "test", "dist", "**/*spec.ts"] +} diff --git a/backend/services/reporting-service/tsconfig.json b/backend/services/reporting-service/tsconfig.json index e69de29b..d0bb99d3 100644 --- a/backend/services/reporting-service/tsconfig.json +++ b/backend/services/reporting-service/tsconfig.json @@ -0,0 +1,30 @@ +{ + "compilerOptions": { + "module": "commonjs", + "declaration": true, + "removeComments": true, + "emitDecoratorMetadata": true, + "experimentalDecorators": true, + "allowSyntheticDefaultImports": true, + "target": "ES2021", + "sourceMap": true, + "outDir": "./dist", + "baseUrl": "./", + "incremental": true, + "skipLibCheck": true, + "strictNullChecks": true, + "noImplicitAny": true, + "strictBindCallApply": true, + "forceConsistentCasingInFileNames": true, + "noFallthroughCasesInSwitch": true, + "paths": { + "@/*": ["src/*"], + "@domain/*": ["src/domain/*"], + "@application/*": ["src/application/*"], + "@infrastructure/*": ["src/infrastructure/*"], + "@api/*": ["src/api/*"], + "@shared/*": ["src/shared/*"], + "@config/*": ["src/config/*"] + } + } +}