diff --git a/backend/services/mining-admin-service/prisma/schema.prisma b/backend/services/mining-admin-service/prisma/schema.prisma index 7f5b7559..b2fea37c 100644 --- a/backend/services/mining-admin-service/prisma/schema.prisma +++ b/backend/services/mining-admin-service/prisma/schema.prisma @@ -359,3 +359,382 @@ model ProcessedEvent { @@index([processedAt]) @@map("processed_events") } + +// ============================================================================= +// CDC 同步表 - 区域数据 (from mining-wallet-service) +// ============================================================================= + +model SyncedProvince { + id String @id @default(uuid()) + originalId String @unique @map("original_id") + code String @unique + name String + status String @default("ACTIVE") + syncedAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + cities SyncedCity[] + + @@index([code]) + @@map("synced_provinces") +} + +model SyncedCity { + id String @id @default(uuid()) + originalId String @unique @map("original_id") + provinceId String @map("province_id") + code String @unique + name String + status String @default("ACTIVE") + syncedAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + province SyncedProvince @relation(fields: [provinceId], references: [originalId]) + userMappings SyncedUserRegionMapping[] + + @@index([provinceId]) + @@index([code]) + @@map("synced_cities") +} + +model SyncedUserRegionMapping { + id String @id @default(uuid()) + accountSequence String @unique @map("account_sequence") + cityId String @map("city_id") + assignedAt DateTime @map("assigned_at") + assignedBy String? @map("assigned_by") + syncedAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + city SyncedCity @relation(fields: [cityId], references: [originalId]) + + @@index([cityId]) + @@map("synced_user_region_mappings") +} + +// ============================================================================= +// CDC 同步表 - 钱包系统账户 (from mining-wallet-service) +// ============================================================================= + +model SyncedWalletSystemAccount { + id String @id @default(uuid()) + originalId String @unique @map("original_id") + accountType String @map("account_type") // HEADQUARTERS, OPERATION, PROVINCE, CITY, FEE, HOT_WALLET, COLD_WALLET + name String + code String @unique + + // 关联区域 + provinceId String? @map("province_id") + cityId String? @map("city_id") + + // 余额信息 + shareBalance Decimal @default(0) @map("share_balance") @db.Decimal(30, 8) + usdtBalance Decimal @default(0) @map("usdt_balance") @db.Decimal(30, 8) + greenPointBalance Decimal @default(0) @map("green_point_balance") @db.Decimal(30, 8) + frozenShare Decimal @default(0) @map("frozen_share") @db.Decimal(30, 8) + frozenUsdt Decimal @default(0) @map("frozen_usdt") @db.Decimal(30, 8) + + // 累计统计 + totalInflow Decimal @default(0) @map("total_inflow") @db.Decimal(30, 8) + totalOutflow Decimal @default(0) @map("total_outflow") @db.Decimal(30, 8) + + // 链上地址 + blockchainAddress String? @map("blockchain_address") + + isActive Boolean @default(true) @map("is_active") + syncedAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + @@index([accountType]) + @@index([provinceId]) + @@index([cityId]) + @@map("synced_wallet_system_accounts") +} + +// ============================================================================= +// CDC 同步表 - 池账户 (from mining-wallet-service) +// ============================================================================= + +model SyncedWalletPoolAccount { + id String @id @default(uuid()) + originalId String @unique @map("original_id") + poolType String @unique @map("pool_type") // SHARE_POOL, BLACK_HOLE_POOL, CIRCULATION_POOL + name String + + // 余额信息 + balance Decimal @default(0) @db.Decimal(30, 8) + totalInflow Decimal @default(0) @map("total_inflow") @db.Decimal(30, 8) + totalOutflow Decimal @default(0) @map("total_outflow") @db.Decimal(30, 8) + + // 黑洞池特有字段 + targetBurn Decimal? @map("target_burn") @db.Decimal(30, 8) + remainingBurn Decimal? @map("remaining_burn") @db.Decimal(30, 8) + + isActive Boolean @default(true) @map("is_active") + syncedAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + @@index([poolType]) + @@map("synced_wallet_pool_accounts") +} + +// ============================================================================= +// CDC 同步表 - 用户钱包 (from mining-wallet-service) +// ============================================================================= + +model SyncedUserWallet { + id String @id @default(uuid()) + originalId String @unique @map("original_id") + accountSequence String @map("account_sequence") + walletType String @map("wallet_type") // CONTRIBUTION, TOKEN_STORAGE, GREEN_POINTS + + // 余额信息 + balance Decimal @default(0) @db.Decimal(30, 8) + frozenBalance Decimal @default(0) @map("frozen_balance") @db.Decimal(30, 8) + + // 累计统计 + totalInflow Decimal @default(0) @map("total_inflow") @db.Decimal(30, 8) + totalOutflow Decimal @default(0) @map("total_outflow") @db.Decimal(30, 8) + + isActive Boolean @default(true) @map("is_active") + syncedAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + @@unique([accountSequence, walletType]) + @@index([accountSequence]) + @@index([walletType]) + @@map("synced_user_wallets") +} + +// ============================================================================= +// CDC 同步表 - 提现请求 (from mining-wallet-service) +// ============================================================================= + +model SyncedWithdrawRequest { + id String @id @default(uuid()) + originalId String @unique @map("original_id") + requestNo String @unique @map("request_no") + accountSequence String @map("account_sequence") + + // 提现信息 + assetType String @map("asset_type") + amount Decimal @db.Decimal(30, 8) + fee Decimal @default(0) @db.Decimal(30, 8) + netAmount Decimal @map("net_amount") @db.Decimal(30, 8) + + // 目标地址 + toAddress String @map("to_address") + + // 状态 + status String // PENDING, PROCESSING, CONFIRMING, COMPLETED, FAILED, CANCELLED + + // 链上信息 + txHash String? @map("tx_hash") + blockNumber BigInt? @map("block_number") + confirmations Int @default(0) + + // 错误信息 + errorMessage String? @map("error_message") + + // 审核信息 + approvedBy String? @map("approved_by") + approvedAt DateTime? @map("approved_at") + + createdAt DateTime @map("created_at") + completedAt DateTime? @map("completed_at") + syncedAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + @@index([accountSequence]) + @@index([status]) + @@index([txHash]) + @@index([createdAt(sort: Desc)]) + @@map("synced_withdraw_requests") +} + +// ============================================================================= +// CDC 同步表 - 充值记录 (from mining-wallet-service) +// ============================================================================= + +model SyncedDepositRecord { + id String @id @default(uuid()) + originalId String @unique @map("original_id") + txHash String @unique @map("tx_hash") + + // 来源信息 + fromAddress String @map("from_address") + toAddress String @map("to_address") + + // 充值信息 + assetType String @map("asset_type") + amount Decimal @db.Decimal(30, 8) + + // 链上信息 + blockNumber BigInt @map("block_number") + confirmations Int @default(0) + + // 匹配的用户 + matchedAccountSeq String? @map("matched_account_seq") + isProcessed Boolean @default(false) @map("is_processed") + processedAt DateTime? @map("processed_at") + + createdAt DateTime @map("created_at") + syncedAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + @@index([fromAddress]) + @@index([toAddress]) + @@index([matchedAccountSeq]) + @@index([isProcessed]) + @@map("synced_deposit_records") +} + +// ============================================================================= +// CDC 同步表 - DEX Swap 记录 (from mining-wallet-service) +// ============================================================================= + +model SyncedDexSwapRecord { + id String @id @default(uuid()) + originalId String @unique @map("original_id") + swapNo String @unique @map("swap_no") + accountSequence String @map("account_sequence") + + // Swap 信息 + fromAsset String @map("from_asset") + toAsset String @map("to_asset") + fromAmount Decimal @map("from_amount") @db.Decimal(30, 8) + toAmount Decimal @map("to_amount") @db.Decimal(30, 8) + exchangeRate Decimal @map("exchange_rate") @db.Decimal(30, 18) + + // 滑点/手续费 + slippage Decimal @default(0) @db.Decimal(10, 4) + fee Decimal @default(0) @db.Decimal(30, 8) + + // 状态 + status String // PENDING, PROCESSING, CONFIRMING, COMPLETED, FAILED, CANCELLED + + // 链上信息 + txHash String? @map("tx_hash") + blockNumber BigInt? @map("block_number") + + errorMessage String? @map("error_message") + + createdAt DateTime @map("created_at") + completedAt DateTime? @map("completed_at") + syncedAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + @@index([accountSequence]) + @@index([status]) + @@index([txHash]) + @@map("synced_dex_swap_records") +} + +// ============================================================================= +// CDC 同步表 - 区块链地址绑定 (from mining-wallet-service) +// ============================================================================= + +model SyncedBlockchainAddressBinding { + id String @id @default(uuid()) + originalId String @unique @map("original_id") + accountSequence String @unique @map("account_sequence") + + // KAVA 地址 + kavaAddress String @unique @map("kava_address") + + // 验证信息 + isVerified Boolean @default(false) @map("is_verified") + verifiedAt DateTime? @map("verified_at") + verificationTxHash String? @map("verification_tx_hash") + + createdAt DateTime @map("created_at") + syncedAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + @@index([kavaAddress]) + @@map("synced_blockchain_address_bindings") +} + +// ============================================================================= +// CDC 同步表 - 黑洞合约 (from mining-wallet-service) +// ============================================================================= + +model SyncedBlackHoleContract { + id String @id @default(uuid()) + originalId String @unique @map("original_id") + contractAddress String @unique @map("contract_address") + name String + + // 累计销毁 + totalBurned Decimal @default(0) @map("total_burned") @db.Decimal(30, 8) + targetBurn Decimal @map("target_burn") @db.Decimal(30, 8) + remainingBurn Decimal @map("remaining_burn") @db.Decimal(30, 8) + + isActive Boolean @default(true) @map("is_active") + syncedAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + @@map("synced_black_hole_contracts") +} + +// ============================================================================= +// CDC 同步表 - 销毁到黑洞记录 (from mining-wallet-service) +// ============================================================================= + +model SyncedBurnToBlackHoleRecord { + id String @id @default(uuid()) + originalId String @unique @map("original_id") + blackHoleId String @map("black_hole_id") + + // 销毁信息 + amount Decimal @db.Decimal(30, 8) + + // 来源 + sourceType String @map("source_type") // USER, SYSTEM_ACCOUNT, POOL, BLOCKCHAIN, EXTERNAL + sourceAccountSeq String? @map("source_account_seq") + sourceUserId String? @map("source_user_id") + sourcePoolType String? @map("source_pool_type") + + // 链上信息 + txHash String? @map("tx_hash") + blockNumber BigInt? @map("block_number") + + // 备注 + memo String? @db.Text + + createdAt DateTime @map("created_at") + syncedAt DateTime @default(now()) + + @@index([blackHoleId]) + @@index([sourceAccountSeq]) + @@index([txHash]) + @@index([createdAt(sort: Desc)]) + @@map("synced_burn_to_black_hole_records") +} + +// ============================================================================= +// CDC 同步表 - 手续费配置 (from mining-wallet-service) +// ============================================================================= + +model SyncedFeeConfig { + id String @id @default(uuid()) + originalId String @unique @map("original_id") + feeType String @unique @map("fee_type") // WITHDRAW, TRADE, SWAP, TRANSFER + + // 费率配置 + feeRate Decimal @map("fee_rate") @db.Decimal(10, 6) + minFee Decimal @map("min_fee") @db.Decimal(30, 8) + maxFee Decimal? @map("max_fee") @db.Decimal(30, 8) + + // 分配比例 + headquartersRate Decimal @map("headquarters_rate") @db.Decimal(10, 6) + operationRate Decimal @map("operation_rate") @db.Decimal(10, 6) + provinceRate Decimal @map("province_rate") @db.Decimal(10, 6) + cityRate Decimal @map("city_rate") @db.Decimal(10, 6) + + isActive Boolean @default(true) @map("is_active") + syncedAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + @@map("synced_fee_configs") +} diff --git a/backend/services/mining-admin-service/src/infrastructure/kafka/cdc-sync.service.ts b/backend/services/mining-admin-service/src/infrastructure/kafka/cdc-sync.service.ts index fb5f802a..fa24fef1 100644 --- a/backend/services/mining-admin-service/src/infrastructure/kafka/cdc-sync.service.ts +++ b/backend/services/mining-admin-service/src/infrastructure/kafka/cdc-sync.service.ts @@ -6,6 +6,7 @@ import { CdcEvent, ServiceEvent, } from './cdc-consumer.service'; +import { WalletSyncHandlers } from './wallet-sync.handlers'; /** * CDC 同步服务 @@ -19,6 +20,7 @@ export class CdcSyncService implements OnModuleInit { private readonly configService: ConfigService, private readonly prisma: PrismaService, private readonly cdcConsumer: CdcConsumerService, + private readonly walletHandlers: WalletSyncHandlers, ) {} async onModuleInit() { @@ -107,6 +109,138 @@ export class CdcSyncService implements OnModuleInit { this.handleCirculationPoolUpdated.bind(this), ); + + // =========================================================================== + // 从 mining-wallet-service 同步钱包数据 + // =========================================================================== + const walletTopic = this.configService.get( + 'CDC_TOPIC_WALLET', + 'mining-admin.wallet.accounts', + ); + this.cdcConsumer.addTopic(walletTopic); + + // 区域数据 + this.cdcConsumer.registerServiceHandler( + 'ProvinceCreated', + this.walletHandlers.handleProvinceCreated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'ProvinceUpdated', + this.walletHandlers.handleProvinceUpdated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'CityCreated', + this.walletHandlers.handleCityCreated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'CityUpdated', + this.walletHandlers.handleCityUpdated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'UserRegionMappingCreated', + this.walletHandlers.handleUserRegionMappingCreated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'UserRegionMappingUpdated', + this.walletHandlers.handleUserRegionMappingUpdated.bind(this.walletHandlers), + ); + + // 系统账户 + this.cdcConsumer.registerServiceHandler( + 'WalletSystemAccountCreated', + this.walletHandlers.handleWalletSystemAccountCreated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'WalletSystemAccountUpdated', + this.walletHandlers.handleWalletSystemAccountUpdated.bind(this.walletHandlers), + ); + + // 池账户 + this.cdcConsumer.registerServiceHandler( + 'WalletPoolAccountCreated', + this.walletHandlers.handleWalletPoolAccountCreated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'WalletPoolAccountUpdated', + this.walletHandlers.handleWalletPoolAccountUpdated.bind(this.walletHandlers), + ); + + // 用户钱包 + this.cdcConsumer.registerServiceHandler( + 'UserWalletCreated', + this.walletHandlers.handleUserWalletCreated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'UserWalletUpdated', + this.walletHandlers.handleUserWalletUpdated.bind(this.walletHandlers), + ); + + // 提现请求 + this.cdcConsumer.registerServiceHandler( + 'WithdrawRequestCreated', + this.walletHandlers.handleWithdrawRequestCreated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'WithdrawRequestUpdated', + this.walletHandlers.handleWithdrawRequestUpdated.bind(this.walletHandlers), + ); + + // 充值记录 + this.cdcConsumer.registerServiceHandler( + 'DepositRecordCreated', + this.walletHandlers.handleDepositRecordCreated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'DepositRecordUpdated', + this.walletHandlers.handleDepositRecordUpdated.bind(this.walletHandlers), + ); + + // DEX Swap + this.cdcConsumer.registerServiceHandler( + 'DexSwapRecordCreated', + this.walletHandlers.handleDexSwapRecordCreated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'DexSwapRecordUpdated', + this.walletHandlers.handleDexSwapRecordUpdated.bind(this.walletHandlers), + ); + + // 地址绑定 + this.cdcConsumer.registerServiceHandler( + 'BlockchainAddressBindingCreated', + this.walletHandlers.handleBlockchainAddressBindingCreated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'BlockchainAddressBindingUpdated', + this.walletHandlers.handleBlockchainAddressBindingUpdated.bind(this.walletHandlers), + ); + + // 黑洞合约 + this.cdcConsumer.registerServiceHandler( + 'BlackHoleContractCreated', + this.walletHandlers.handleBlackHoleContractCreated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'BlackHoleContractUpdated', + this.walletHandlers.handleBlackHoleContractUpdated.bind(this.walletHandlers), + ); + + // 销毁记录 + this.cdcConsumer.registerServiceHandler( + 'BurnToBlackHoleRecordCreated', + this.walletHandlers.handleBurnToBlackHoleRecordCreated.bind(this.walletHandlers), + ); + + // 费率配置 + this.cdcConsumer.registerServiceHandler( + 'FeeConfigCreated', + this.walletHandlers.handleFeeConfigCreated.bind(this.walletHandlers), + ); + this.cdcConsumer.registerServiceHandler( + 'FeeConfigUpdated', + this.walletHandlers.handleFeeConfigUpdated.bind(this.walletHandlers), + ); + this.logger.log('CDC sync handlers registered'); } diff --git a/backend/services/mining-admin-service/src/infrastructure/kafka/kafka.module.ts b/backend/services/mining-admin-service/src/infrastructure/kafka/kafka.module.ts index 0e5f04b4..cc5a2b03 100644 --- a/backend/services/mining-admin-service/src/infrastructure/kafka/kafka.module.ts +++ b/backend/services/mining-admin-service/src/infrastructure/kafka/kafka.module.ts @@ -2,10 +2,11 @@ import { Module } from '@nestjs/common'; import { ConfigModule, ConfigService } from '@nestjs/config'; import { CdcConsumerService } from './cdc-consumer.service'; import { CdcSyncService } from './cdc-sync.service'; +import { WalletSyncHandlers } from './wallet-sync.handlers'; @Module({ imports: [ConfigModule], - providers: [CdcConsumerService, CdcSyncService], - exports: [CdcConsumerService, CdcSyncService], + providers: [CdcConsumerService, CdcSyncService, WalletSyncHandlers], + exports: [CdcConsumerService, CdcSyncService, WalletSyncHandlers], }) export class KafkaModule {} diff --git a/backend/services/mining-admin-service/src/infrastructure/kafka/wallet-sync.handlers.ts b/backend/services/mining-admin-service/src/infrastructure/kafka/wallet-sync.handlers.ts new file mode 100644 index 00000000..3ccdb6ff --- /dev/null +++ b/backend/services/mining-admin-service/src/infrastructure/kafka/wallet-sync.handlers.ts @@ -0,0 +1,750 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { PrismaService } from '../persistence/prisma/prisma.service'; +import { ServiceEvent } from './cdc-consumer.service'; + +/** + * mining-wallet-service CDC 事件处理器 + */ +@Injectable() +export class WalletSyncHandlers { + private readonly logger = new Logger(WalletSyncHandlers.name); + + constructor(private readonly prisma: PrismaService) {} + + // =========================================================================== + // 区域数据处理 + // =========================================================================== + + async handleProvinceCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedProvince.upsert({ + where: { originalId: payload.id }, + create: { + originalId: payload.id, + code: payload.code, + name: payload.name, + status: payload.status || 'ACTIVE', + }, + update: { + code: payload.code, + name: payload.name, + status: payload.status || 'ACTIVE', + }, + }); + + this.logger.debug(`Synced province: ${payload.code}`); + } catch (error) { + this.logger.error(`Failed to sync province: ${payload.code}`, error); + } + } + + async handleProvinceUpdated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedProvince.updateMany({ + where: { originalId: payload.id }, + data: { + code: payload.code, + name: payload.name, + status: payload.status, + }, + }); + + this.logger.debug(`Updated province: ${payload.code}`); + } catch (error) { + this.logger.error(`Failed to update province: ${payload.code}`, error); + } + } + + async handleCityCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedCity.upsert({ + where: { originalId: payload.id }, + create: { + originalId: payload.id, + provinceId: payload.provinceId, + code: payload.code, + name: payload.name, + status: payload.status || 'ACTIVE', + }, + update: { + provinceId: payload.provinceId, + code: payload.code, + name: payload.name, + status: payload.status || 'ACTIVE', + }, + }); + + this.logger.debug(`Synced city: ${payload.code}`); + } catch (error) { + this.logger.error(`Failed to sync city: ${payload.code}`, error); + } + } + + async handleCityUpdated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedCity.updateMany({ + where: { originalId: payload.id }, + data: { + provinceId: payload.provinceId, + code: payload.code, + name: payload.name, + status: payload.status, + }, + }); + + this.logger.debug(`Updated city: ${payload.code}`); + } catch (error) { + this.logger.error(`Failed to update city: ${payload.code}`, error); + } + } + + async handleUserRegionMappingCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedUserRegionMapping.upsert({ + where: { accountSequence: payload.accountSequence }, + create: { + accountSequence: payload.accountSequence, + cityId: payload.cityId, + assignedAt: new Date(payload.assignedAt), + assignedBy: payload.assignedBy, + }, + update: { + cityId: payload.cityId, + assignedAt: new Date(payload.assignedAt), + assignedBy: payload.assignedBy, + }, + }); + + this.logger.debug(`Synced user region mapping: ${payload.accountSequence}`); + } catch (error) { + this.logger.error(`Failed to sync user region mapping: ${payload.accountSequence}`, error); + } + } + + async handleUserRegionMappingUpdated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedUserRegionMapping.updateMany({ + where: { accountSequence: payload.accountSequence }, + data: { + cityId: payload.cityId, + assignedAt: new Date(payload.assignedAt), + assignedBy: payload.assignedBy, + }, + }); + + this.logger.debug(`Updated user region mapping: ${payload.accountSequence}`); + } catch (error) { + this.logger.error(`Failed to update user region mapping: ${payload.accountSequence}`, error); + } + } + + // =========================================================================== + // 系统账户处理 + // =========================================================================== + + async handleWalletSystemAccountCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedWalletSystemAccount.upsert({ + where: { originalId: payload.id }, + create: { + originalId: payload.id, + accountType: payload.accountType, + name: payload.name, + code: payload.code, + provinceId: payload.provinceId, + cityId: payload.cityId, + shareBalance: payload.shareBalance || 0, + usdtBalance: payload.usdtBalance || 0, + greenPointBalance: payload.greenPointBalance || 0, + frozenShare: payload.frozenShare || 0, + frozenUsdt: payload.frozenUsdt || 0, + totalInflow: payload.totalInflow || 0, + totalOutflow: payload.totalOutflow || 0, + blockchainAddress: payload.blockchainAddress, + isActive: payload.isActive ?? true, + }, + update: { + accountType: payload.accountType, + name: payload.name, + code: payload.code, + provinceId: payload.provinceId, + cityId: payload.cityId, + shareBalance: payload.shareBalance, + usdtBalance: payload.usdtBalance, + greenPointBalance: payload.greenPointBalance, + frozenShare: payload.frozenShare, + frozenUsdt: payload.frozenUsdt, + totalInflow: payload.totalInflow, + totalOutflow: payload.totalOutflow, + blockchainAddress: payload.blockchainAddress, + isActive: payload.isActive, + }, + }); + + this.logger.debug(`Synced wallet system account: ${payload.code}`); + } catch (error) { + this.logger.error(`Failed to sync wallet system account: ${payload.code}`, error); + } + } + + async handleWalletSystemAccountUpdated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedWalletSystemAccount.updateMany({ + where: { originalId: payload.id }, + data: { + name: payload.name, + shareBalance: payload.shareBalance, + usdtBalance: payload.usdtBalance, + greenPointBalance: payload.greenPointBalance, + frozenShare: payload.frozenShare, + frozenUsdt: payload.frozenUsdt, + totalInflow: payload.totalInflow, + totalOutflow: payload.totalOutflow, + blockchainAddress: payload.blockchainAddress, + isActive: payload.isActive, + }, + }); + + this.logger.debug(`Updated wallet system account: ${payload.code}`); + } catch (error) { + this.logger.error(`Failed to update wallet system account: ${payload.code}`, error); + } + } + + // =========================================================================== + // 池账户处理 + // =========================================================================== + + async handleWalletPoolAccountCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedWalletPoolAccount.upsert({ + where: { originalId: payload.id }, + create: { + originalId: payload.id, + poolType: payload.poolType, + name: payload.name, + balance: payload.balance || 0, + totalInflow: payload.totalInflow || 0, + totalOutflow: payload.totalOutflow || 0, + targetBurn: payload.targetBurn, + remainingBurn: payload.remainingBurn, + isActive: payload.isActive ?? true, + }, + update: { + name: payload.name, + balance: payload.balance, + totalInflow: payload.totalInflow, + totalOutflow: payload.totalOutflow, + targetBurn: payload.targetBurn, + remainingBurn: payload.remainingBurn, + isActive: payload.isActive, + }, + }); + + this.logger.debug(`Synced wallet pool account: ${payload.poolType}`); + } catch (error) { + this.logger.error(`Failed to sync wallet pool account: ${payload.poolType}`, error); + } + } + + async handleWalletPoolAccountUpdated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedWalletPoolAccount.updateMany({ + where: { originalId: payload.id }, + data: { + name: payload.name, + balance: payload.balance, + totalInflow: payload.totalInflow, + totalOutflow: payload.totalOutflow, + targetBurn: payload.targetBurn, + remainingBurn: payload.remainingBurn, + isActive: payload.isActive, + }, + }); + + this.logger.debug(`Updated wallet pool account: ${payload.poolType}`); + } catch (error) { + this.logger.error(`Failed to update wallet pool account: ${payload.poolType}`, error); + } + } + + // =========================================================================== + // 用户钱包处理 + // =========================================================================== + + async handleUserWalletCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedUserWallet.upsert({ + where: { originalId: payload.id }, + create: { + originalId: payload.id, + accountSequence: payload.accountSequence, + walletType: payload.walletType, + balance: payload.balance || 0, + frozenBalance: payload.frozenBalance || 0, + totalInflow: payload.totalInflow || 0, + totalOutflow: payload.totalOutflow || 0, + isActive: payload.isActive ?? true, + }, + update: { + balance: payload.balance, + frozenBalance: payload.frozenBalance, + totalInflow: payload.totalInflow, + totalOutflow: payload.totalOutflow, + isActive: payload.isActive, + }, + }); + + this.logger.debug(`Synced user wallet: ${payload.accountSequence}/${payload.walletType}`); + } catch (error) { + this.logger.error(`Failed to sync user wallet: ${payload.accountSequence}/${payload.walletType}`, error); + } + } + + async handleUserWalletUpdated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedUserWallet.updateMany({ + where: { originalId: payload.id }, + data: { + balance: payload.balance, + frozenBalance: payload.frozenBalance, + totalInflow: payload.totalInflow, + totalOutflow: payload.totalOutflow, + isActive: payload.isActive, + }, + }); + + this.logger.debug(`Updated user wallet: ${payload.accountSequence}/${payload.walletType}`); + } catch (error) { + this.logger.error(`Failed to update user wallet: ${payload.accountSequence}/${payload.walletType}`, error); + } + } + + // =========================================================================== + // 提现请求处理 + // =========================================================================== + + async handleWithdrawRequestCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedWithdrawRequest.upsert({ + where: { originalId: payload.id }, + create: { + originalId: payload.id, + requestNo: payload.requestNo, + accountSequence: payload.accountSequence, + assetType: payload.assetType, + amount: payload.amount, + fee: payload.fee || 0, + netAmount: payload.netAmount, + toAddress: payload.toAddress, + status: payload.status, + txHash: payload.txHash, + blockNumber: payload.blockNumber, + confirmations: payload.confirmations || 0, + errorMessage: payload.errorMessage, + approvedBy: payload.approvedBy, + approvedAt: payload.approvedAt ? new Date(payload.approvedAt) : null, + createdAt: new Date(payload.createdAt), + completedAt: payload.completedAt ? new Date(payload.completedAt) : null, + }, + update: { + status: payload.status, + txHash: payload.txHash, + blockNumber: payload.blockNumber, + confirmations: payload.confirmations, + errorMessage: payload.errorMessage, + approvedBy: payload.approvedBy, + approvedAt: payload.approvedAt ? new Date(payload.approvedAt) : null, + completedAt: payload.completedAt ? new Date(payload.completedAt) : null, + }, + }); + + this.logger.debug(`Synced withdraw request: ${payload.requestNo}`); + } catch (error) { + this.logger.error(`Failed to sync withdraw request: ${payload.requestNo}`, error); + } + } + + async handleWithdrawRequestUpdated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedWithdrawRequest.updateMany({ + where: { originalId: payload.id }, + data: { + status: payload.status, + txHash: payload.txHash, + blockNumber: payload.blockNumber, + confirmations: payload.confirmations, + errorMessage: payload.errorMessage, + approvedBy: payload.approvedBy, + approvedAt: payload.approvedAt ? new Date(payload.approvedAt) : null, + completedAt: payload.completedAt ? new Date(payload.completedAt) : null, + }, + }); + + this.logger.debug(`Updated withdraw request: ${payload.requestNo}`); + } catch (error) { + this.logger.error(`Failed to update withdraw request: ${payload.requestNo}`, error); + } + } + + // =========================================================================== + // 充值记录处理 + // =========================================================================== + + async handleDepositRecordCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedDepositRecord.upsert({ + where: { originalId: payload.id }, + create: { + originalId: payload.id, + txHash: payload.txHash, + fromAddress: payload.fromAddress, + toAddress: payload.toAddress, + assetType: payload.assetType, + amount: payload.amount, + blockNumber: payload.blockNumber, + confirmations: payload.confirmations || 0, + matchedAccountSeq: payload.matchedAccountSeq, + isProcessed: payload.isProcessed || false, + processedAt: payload.processedAt ? new Date(payload.processedAt) : null, + createdAt: new Date(payload.createdAt), + }, + update: { + confirmations: payload.confirmations, + matchedAccountSeq: payload.matchedAccountSeq, + isProcessed: payload.isProcessed, + processedAt: payload.processedAt ? new Date(payload.processedAt) : null, + }, + }); + + this.logger.debug(`Synced deposit record: ${payload.txHash}`); + } catch (error) { + this.logger.error(`Failed to sync deposit record: ${payload.txHash}`, error); + } + } + + async handleDepositRecordUpdated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedDepositRecord.updateMany({ + where: { originalId: payload.id }, + data: { + confirmations: payload.confirmations, + matchedAccountSeq: payload.matchedAccountSeq, + isProcessed: payload.isProcessed, + processedAt: payload.processedAt ? new Date(payload.processedAt) : null, + }, + }); + + this.logger.debug(`Updated deposit record: ${payload.txHash}`); + } catch (error) { + this.logger.error(`Failed to update deposit record: ${payload.txHash}`, error); + } + } + + // =========================================================================== + // DEX Swap 处理 + // =========================================================================== + + async handleDexSwapRecordCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedDexSwapRecord.upsert({ + where: { originalId: payload.id }, + create: { + originalId: payload.id, + swapNo: payload.swapNo, + accountSequence: payload.accountSequence, + fromAsset: payload.fromAsset, + toAsset: payload.toAsset, + fromAmount: payload.fromAmount, + toAmount: payload.toAmount, + exchangeRate: payload.exchangeRate, + slippage: payload.slippage || 0, + fee: payload.fee || 0, + status: payload.status, + txHash: payload.txHash, + blockNumber: payload.blockNumber, + errorMessage: payload.errorMessage, + createdAt: new Date(payload.createdAt), + completedAt: payload.completedAt ? new Date(payload.completedAt) : null, + }, + update: { + toAmount: payload.toAmount, + exchangeRate: payload.exchangeRate, + status: payload.status, + txHash: payload.txHash, + blockNumber: payload.blockNumber, + errorMessage: payload.errorMessage, + completedAt: payload.completedAt ? new Date(payload.completedAt) : null, + }, + }); + + this.logger.debug(`Synced dex swap record: ${payload.swapNo}`); + } catch (error) { + this.logger.error(`Failed to sync dex swap record: ${payload.swapNo}`, error); + } + } + + async handleDexSwapRecordUpdated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedDexSwapRecord.updateMany({ + where: { originalId: payload.id }, + data: { + toAmount: payload.toAmount, + exchangeRate: payload.exchangeRate, + status: payload.status, + txHash: payload.txHash, + blockNumber: payload.blockNumber, + errorMessage: payload.errorMessage, + completedAt: payload.completedAt ? new Date(payload.completedAt) : null, + }, + }); + + this.logger.debug(`Updated dex swap record: ${payload.swapNo}`); + } catch (error) { + this.logger.error(`Failed to update dex swap record: ${payload.swapNo}`, error); + } + } + + // =========================================================================== + // 地址绑定处理 + // =========================================================================== + + async handleBlockchainAddressBindingCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedBlockchainAddressBinding.upsert({ + where: { originalId: payload.id }, + create: { + originalId: payload.id, + accountSequence: payload.accountSequence, + kavaAddress: payload.kavaAddress, + isVerified: payload.isVerified || false, + verifiedAt: payload.verifiedAt ? new Date(payload.verifiedAt) : null, + verificationTxHash: payload.verificationTxHash, + createdAt: new Date(payload.createdAt), + }, + update: { + kavaAddress: payload.kavaAddress, + isVerified: payload.isVerified, + verifiedAt: payload.verifiedAt ? new Date(payload.verifiedAt) : null, + verificationTxHash: payload.verificationTxHash, + }, + }); + + this.logger.debug(`Synced blockchain address binding: ${payload.accountSequence}`); + } catch (error) { + this.logger.error(`Failed to sync blockchain address binding: ${payload.accountSequence}`, error); + } + } + + async handleBlockchainAddressBindingUpdated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedBlockchainAddressBinding.updateMany({ + where: { originalId: payload.id }, + data: { + kavaAddress: payload.kavaAddress, + isVerified: payload.isVerified, + verifiedAt: payload.verifiedAt ? new Date(payload.verifiedAt) : null, + verificationTxHash: payload.verificationTxHash, + }, + }); + + this.logger.debug(`Updated blockchain address binding: ${payload.accountSequence}`); + } catch (error) { + this.logger.error(`Failed to update blockchain address binding: ${payload.accountSequence}`, error); + } + } + + // =========================================================================== + // 黑洞合约处理 + // =========================================================================== + + async handleBlackHoleContractCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedBlackHoleContract.upsert({ + where: { originalId: payload.id }, + create: { + originalId: payload.id, + contractAddress: payload.contractAddress, + name: payload.name, + totalBurned: payload.totalBurned || 0, + targetBurn: payload.targetBurn, + remainingBurn: payload.remainingBurn, + isActive: payload.isActive ?? true, + }, + update: { + name: payload.name, + totalBurned: payload.totalBurned, + targetBurn: payload.targetBurn, + remainingBurn: payload.remainingBurn, + isActive: payload.isActive, + }, + }); + + this.logger.debug(`Synced black hole contract: ${payload.contractAddress}`); + } catch (error) { + this.logger.error(`Failed to sync black hole contract: ${payload.contractAddress}`, error); + } + } + + async handleBlackHoleContractUpdated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedBlackHoleContract.updateMany({ + where: { originalId: payload.id }, + data: { + name: payload.name, + totalBurned: payload.totalBurned, + targetBurn: payload.targetBurn, + remainingBurn: payload.remainingBurn, + isActive: payload.isActive, + }, + }); + + this.logger.debug(`Updated black hole contract: ${payload.contractAddress}`); + } catch (error) { + this.logger.error(`Failed to update black hole contract: ${payload.contractAddress}`, error); + } + } + + // =========================================================================== + // 销毁记录处理 + // =========================================================================== + + async handleBurnToBlackHoleRecordCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedBurnToBlackHoleRecord.upsert({ + where: { originalId: payload.id }, + create: { + originalId: payload.id, + blackHoleId: payload.blackHoleId, + amount: payload.amount, + sourceType: payload.sourceType, + sourceAccountSeq: payload.sourceAccountSeq, + sourceUserId: payload.sourceUserId, + sourcePoolType: payload.sourcePoolType, + txHash: payload.txHash, + blockNumber: payload.blockNumber, + memo: payload.memo, + createdAt: new Date(payload.createdAt), + }, + update: { + txHash: payload.txHash, + blockNumber: payload.blockNumber, + }, + }); + + this.logger.debug(`Synced burn to black hole record: ${payload.id}`); + } catch (error) { + this.logger.error(`Failed to sync burn to black hole record: ${payload.id}`, error); + } + } + + // =========================================================================== + // 费率配置处理 + // =========================================================================== + + async handleFeeConfigCreated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedFeeConfig.upsert({ + where: { originalId: payload.id }, + create: { + originalId: payload.id, + feeType: payload.feeType, + feeRate: payload.feeRate, + minFee: payload.minFee, + maxFee: payload.maxFee, + headquartersRate: payload.headquartersRate, + operationRate: payload.operationRate, + provinceRate: payload.provinceRate, + cityRate: payload.cityRate, + isActive: payload.isActive ?? true, + }, + update: { + feeRate: payload.feeRate, + minFee: payload.minFee, + maxFee: payload.maxFee, + headquartersRate: payload.headquartersRate, + operationRate: payload.operationRate, + provinceRate: payload.provinceRate, + cityRate: payload.cityRate, + isActive: payload.isActive, + }, + }); + + this.logger.debug(`Synced fee config: ${payload.feeType}`); + } catch (error) { + this.logger.error(`Failed to sync fee config: ${payload.feeType}`, error); + } + } + + async handleFeeConfigUpdated(event: ServiceEvent): Promise { + const { payload } = event; + + try { + await this.prisma.syncedFeeConfig.updateMany({ + where: { originalId: payload.id }, + data: { + feeRate: payload.feeRate, + minFee: payload.minFee, + maxFee: payload.maxFee, + headquartersRate: payload.headquartersRate, + operationRate: payload.operationRate, + provinceRate: payload.provinceRate, + cityRate: payload.cityRate, + isActive: payload.isActive, + }, + }); + + this.logger.debug(`Updated fee config: ${payload.feeType}`); + } catch (error) { + this.logger.error(`Failed to update fee config: ${payload.feeType}`, error); + } + } +} diff --git a/backend/services/mining-service/prisma/schema.prisma b/backend/services/mining-service/prisma/schema.prisma index e8bd1c42..59e39bda 100644 --- a/backend/services/mining-service/prisma/schema.prisma +++ b/backend/services/mining-service/prisma/schema.prisma @@ -11,33 +11,33 @@ datasource db { // 挖矿全局配置 model MiningConfig { - id String @id @default(uuid()) - totalShares Decimal @db.Decimal(30, 8) // 总积分股数量 (100.02B) - distributionPool Decimal @db.Decimal(30, 8) // 分配池 (200M) - remainingDistribution Decimal @db.Decimal(30, 8) // 剩余可分配 - halvingPeriodYears Int @default(2) // 减半周期(年) - currentEra Int @default(1) // 当前纪元 - eraStartDate DateTime // 当前纪元开始日期 - minuteDistribution Decimal @db.Decimal(30, 18) // 每分钟分配量 - isActive Boolean @default(false) // 是否已激活挖矿 - activatedAt DateTime? // 激活时间 - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt + id String @id @default(uuid()) + totalShares Decimal @db.Decimal(30, 8) // 总积分股数量 (100.02B) + distributionPool Decimal @db.Decimal(30, 8) // 分配池 (200M) + remainingDistribution Decimal @db.Decimal(30, 8) // 剩余可分配 + halvingPeriodYears Int @default(2) // 减半周期(年) + currentEra Int @default(1) // 当前纪元 + eraStartDate DateTime // 当前纪元开始日期 + minuteDistribution Decimal @db.Decimal(30, 18) // 每分钟分配量 + isActive Boolean @default(false) // 是否已激活挖矿 + activatedAt DateTime? // 激活时间 + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt @@map("mining_configs") } // 减半纪元记录 model MiningEra { - id String @id @default(uuid()) - eraNumber Int @unique - startDate DateTime - endDate DateTime? - initialDistribution Decimal @db.Decimal(30, 8) // 纪元初始可分配量 - totalDistributed Decimal @db.Decimal(30, 8) @default(0) // 已分配量 - minuteDistribution Decimal @db.Decimal(30, 18) // 每分钟分配量 - isActive Boolean @default(true) - createdAt DateTime @default(now()) + id String @id @default(uuid()) + eraNumber Int @unique + startDate DateTime + endDate DateTime? + initialDistribution Decimal @db.Decimal(30, 8) // 纪元初始可分配量 + totalDistributed Decimal @default(0) @db.Decimal(30, 8) // 已分配量 + minuteDistribution Decimal @db.Decimal(30, 18) // 每分钟分配量 + isActive Boolean @default(true) + createdAt DateTime @default(now()) @@map("mining_eras") } @@ -46,18 +46,18 @@ model MiningEra { // 用户挖矿账户 model MiningAccount { - id String @id @default(uuid()) - accountSequence String @unique - totalMined Decimal @db.Decimal(30, 8) @default(0) // 总挖到的积分股 - availableBalance Decimal @db.Decimal(30, 8) @default(0) // 可用余额 - frozenBalance Decimal @db.Decimal(30, 8) @default(0) // 冻结余额 - totalContribution Decimal @db.Decimal(30, 8) @default(0) // 当前算力(从 contribution-service 同步) - lastSyncedAt DateTime? // 最后同步算力时间 - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt + id String @id @default(uuid()) + accountSequence String @unique + totalMined Decimal @default(0) @db.Decimal(30, 8) // 总挖到的积分股 + availableBalance Decimal @default(0) @db.Decimal(30, 8) // 可用余额 + frozenBalance Decimal @default(0) @db.Decimal(30, 8) // 冻结余额 + totalContribution Decimal @default(0) @db.Decimal(30, 8) // 当前算力(从 contribution-service 同步) + lastSyncedAt DateTime? // 最后同步算力时间 + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt - records MiningRecord[] - transactions MiningTransaction[] + records MiningRecord[] + transactions MiningTransaction[] @@index([totalContribution(sort: Desc)]) @@map("mining_accounts") @@ -69,12 +69,12 @@ model MiningRecord { accountSequence String miningMinute DateTime // 挖矿分钟(精确到分钟) contributionRatio Decimal @db.Decimal(30, 18) // 当时的算力占比 - totalContribution Decimal @db.Decimal(30, 8) // 当时的总算力 + totalContribution Decimal @db.Decimal(30, 8) // 当时的总算力 minuteDistribution Decimal @db.Decimal(30, 18) // 当分钟总分配量 minedAmount Decimal @db.Decimal(30, 18) // 挖到的数量 createdAt DateTime @default(now()) - account MiningAccount @relation(fields: [accountSequence], references: [accountSequence]) + account MiningAccount @relation(fields: [accountSequence], references: [accountSequence]) @@unique([accountSequence, miningMinute]) @@index([miningMinute]) @@ -83,21 +83,31 @@ model MiningRecord { // 挖矿交易流水 model MiningTransaction { - id String @id @default(uuid()) + id String @id @default(uuid()) accountSequence String - type String // MINE, FREEZE, UNFREEZE, TRANSFER_OUT, TRANSFER_IN, BURN - amount Decimal @db.Decimal(30, 8) - balanceBefore Decimal @db.Decimal(30, 8) - balanceAfter Decimal @db.Decimal(30, 8) - referenceId String? // 关联ID(如交易ID、划转ID) - referenceType String? // 关联类型 - description String? - createdAt DateTime @default(now()) + type String // MINE, FREEZE, UNFREEZE, TRANSFER_OUT, TRANSFER_IN, BURN + amount Decimal @db.Decimal(30, 8) + balanceBefore Decimal @db.Decimal(30, 8) + balanceAfter Decimal @db.Decimal(30, 8) + referenceId String? // 关联ID(如交易ID、划转ID) + referenceType String? // 关联类型 - account MiningAccount @relation(fields: [accountSequence], references: [accountSequence]) + // 交易对手方信息 + counterpartyType String? @map("counterparty_type") // USER, POOL, SYSTEM + counterpartyAccountSeq String? @map("counterparty_account_seq") // 对手方账户序列号 + counterpartyUserId String? @map("counterparty_user_id") // 对手方用户ID + + // 详细备注(包含完整交易信息,格式: "划转到用户[U123456]") + memo String? @db.Text + description String? // 保留兼容旧字段 + createdAt DateTime @default(now()) + + account MiningAccount @relation(fields: [accountSequence], references: [accountSequence]) @@index([accountSequence, createdAt(sort: Desc)]) @@index([type]) + @@index([counterpartyAccountSeq]) + @@index([counterpartyUserId]) @@map("mining_transactions") } @@ -105,13 +115,13 @@ model MiningTransaction { // 每分钟挖矿统计 model MinuteMiningStat { - id String @id @default(uuid()) - minute DateTime @unique - totalContribution Decimal @db.Decimal(30, 8) // 参与挖矿的总算力 - totalDistributed Decimal @db.Decimal(30, 18) // 该分钟分配的总量 - participantCount Int // 参与者数量 - burnAmount Decimal @db.Decimal(30, 8) @default(0) // 该分钟销毁量 - createdAt DateTime @default(now()) + id String @id @default(uuid()) + minute DateTime @unique + totalContribution Decimal @db.Decimal(30, 8) // 参与挖矿的总算力 + totalDistributed Decimal @db.Decimal(30, 18) // 该分钟分配的总量 + participantCount Int // 参与者数量 + burnAmount Decimal @default(0) @db.Decimal(30, 8) // 该分钟销毁量 + createdAt DateTime @default(now()) @@index([minute(sort: Desc)]) @@map("minute_mining_stats") @@ -135,15 +145,15 @@ model DailyMiningStat { // 黑洞账户 model BlackHole { - id String @id @default(uuid()) - totalBurned Decimal @db.Decimal(30, 8) @default(0) // 已销毁总量 - targetBurn Decimal @db.Decimal(30, 8) // 目标销毁量 (10B) - remainingBurn Decimal @db.Decimal(30, 8) // 剩余待销毁 - lastBurnMinute DateTime? - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt + id String @id @default(uuid()) + totalBurned Decimal @default(0) @db.Decimal(30, 8) // 已销毁总量 + targetBurn Decimal @db.Decimal(30, 8) // 目标销毁量 (10B) + remainingBurn Decimal @db.Decimal(30, 8) // 剩余待销毁 + lastBurnMinute DateTime? + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt - records BurnRecord[] + records BurnRecord[] @@map("black_holes") } @@ -155,12 +165,21 @@ model BurnRecord { burnMinute DateTime burnAmount Decimal @db.Decimal(30, 18) remainingTarget Decimal @db.Decimal(30, 8) // 销毁后剩余目标 - createdAt DateTime @default(now()) - blackHole BlackHole @relation(fields: [blackHoleId], references: [id]) + // 来源信息(从哪个池/用户销毁) + sourceType String? @map("source_type") // CIRCULATION_POOL, USER, SYSTEM + sourceAccountSeq String? @map("source_account_seq") // 来源账户序列号 + sourceUserId String? @map("source_user_id") // 来源用户ID + + // 详细备注 + memo String? @db.Text + createdAt DateTime @default(now()) + + blackHole BlackHole @relation(fields: [blackHoleId], references: [id]) @@unique([blackHoleId, burnMinute]) @@index([burnMinute]) + @@index([sourceAccountSeq]) @@map("burn_records") } @@ -168,19 +187,111 @@ model BurnRecord { // 价格快照(每分钟) model PriceSnapshot { - id String @id @default(uuid()) - snapshotTime DateTime @unique - price Decimal @db.Decimal(30, 18) // 当时价格 - sharePool Decimal @db.Decimal(30, 8) // 股池 - blackHoleAmount Decimal @db.Decimal(30, 8) // 黑洞数量 - circulationPool Decimal @db.Decimal(30, 8) // 流通池 - effectiveDenominator Decimal @db.Decimal(30, 8) // 有效分母 - createdAt DateTime @default(now()) + id String @id @default(uuid()) + snapshotTime DateTime @unique + price Decimal @db.Decimal(30, 18) // 当时价格 + sharePool Decimal @db.Decimal(30, 8) // 股池 + blackHoleAmount Decimal @db.Decimal(30, 8) // 黑洞数量 + circulationPool Decimal @db.Decimal(30, 8) // 流通池 + effectiveDenominator Decimal @db.Decimal(30, 8) // 有效分母 + createdAt DateTime @default(now()) @@index([snapshotTime(sort: Desc)]) @@map("price_snapshots") } +// ==================== 池账户系统 ==================== + +// 池账户类型枚举 +enum PoolAccountType { + SHARE_POOL // 积分股池 - 总股池 + BLACK_HOLE_POOL // 黑洞积分股池 - 销毁池 + CIRCULATION_POOL // 流通积分股池 - 流通池 +} + +// 池账户交易类型枚举 +enum PoolTransactionType { + // 积分股池操作 + MINING_DISTRIBUTE // 挖矿分配(股池 -> 用户) + FEE_COLLECT // 手续费收取(用户 -> 股池) + INITIAL_INJECT // 初始注入 + + // 黑洞池操作 + BURN // 销毁(流通池 -> 黑洞) + + // 流通池操作 + USER_TRANSFER_IN // 用户划入(用户 -> 流通池) + USER_TRANSFER_OUT // 用户划出(流通池 -> 用户) + TRADE_BUY // 交易买入 + TRADE_SELL // 交易卖出 + + // 通用操作 + POOL_TRANSFER // 池间划转 + ADJUSTMENT // 系统调整 +} + +// 池账户(管理三大池:积分股池、黑洞积分股池、流通积分股池) +model PoolAccount { + id String @id @default(uuid()) + poolType PoolAccountType @unique @map("pool_type") // 池类型 + name String // 池名称 + balance Decimal @default(0) @db.Decimal(30, 8) // 当前余额 + totalInflow Decimal @default(0) @db.Decimal(30, 8) // 累计流入 + totalOutflow Decimal @default(0) @db.Decimal(30, 8) // 累计流出 + isActive Boolean @default(true) @map("is_active") + description String? + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + transactions PoolTransaction[] + + @@index([poolType]) + @@map("pool_accounts") +} + +// 池账户交易明细(包含交易对手方信息) +model PoolTransaction { + id String @id @default(uuid()) + poolAccountId String @map("pool_account_id") + poolType PoolAccountType @map("pool_type") + transactionType PoolTransactionType @map("transaction_type") + + // 金额信息 + amount Decimal @db.Decimal(30, 8) + balanceBefore Decimal @map("balance_before") @db.Decimal(30, 8) + balanceAfter Decimal @map("balance_after") @db.Decimal(30, 8) + + // 交易对手方信息(关键:用户ID和账户序列号) + counterpartyType String? @map("counterparty_type") // USER, POOL, SYSTEM, EXTERNAL + counterpartyAccountSeq String? @map("counterparty_account_seq") // 对手方账户序列号 + counterpartyUserId String? @map("counterparty_user_id") // 对手方用户ID + counterpartyPoolType PoolAccountType? @map("counterparty_pool_type") // 如果对手方是池账户 + + // 关联信息 + referenceId String? @map("reference_id") // 关联业务ID(如订单ID、划转ID) + referenceType String? @map("reference_type") // 关联类型 + txHash String? @map("tx_hash") // 链上交易哈希(如有) + + // 详细备注(包含完整交易信息) + // 格式示例: "挖矿分配给用户[U123456], 算力占比0.5%, 分钟2024-01-10 10:30" + memo String? @db.Text + + // 扩展数据(JSON格式存储更多业务细节) + metadata Json? + + createdAt DateTime @default(now()) @map("created_at") + + poolAccount PoolAccount @relation(fields: [poolAccountId], references: [id]) + + @@index([poolAccountId, createdAt(sort: Desc)]) + @@index([poolType, transactionType]) + @@index([counterpartyAccountSeq]) + @@index([counterpartyUserId]) + @@index([referenceId]) + @@index([createdAt(sort: Desc)]) + @@map("pool_transactions") +} + // ==================== Outbox ==================== enum OutboxStatus { diff --git a/backend/services/mining-wallet-service/.env.example b/backend/services/mining-wallet-service/.env.example new file mode 100644 index 00000000..a4b2c919 --- /dev/null +++ b/backend/services/mining-wallet-service/.env.example @@ -0,0 +1,34 @@ +# ============================================================================= +# Mining Wallet Service - Environment Configuration +# ============================================================================= + +# Server +PORT=3025 +NODE_ENV=development + +# Database +DATABASE_URL="postgresql://user:password@localhost:5432/mining_wallet_db?schema=public" + +# Redis +REDIS_HOST=localhost +REDIS_PORT=6379 +REDIS_PASSWORD= +REDIS_DB=15 + +# Kafka +KAFKA_BROKERS=localhost:9092 + +# JWT +JWT_SECRET=your-jwt-secret-key + +# CORS +CORS_ORIGIN=* + +# KAVA Blockchain +KAVA_RPC_URL=https://evm.kava.io +KAVA_CHAIN_ID=2222 +KAVA_HOT_WALLET_PRIVATE_KEY= +KAVA_BLACK_HOLE_ADDRESS=0x000000000000000000000000000000000000dEaD + +# Swagger +SWAGGER_ENABLED=true diff --git a/backend/services/mining-wallet-service/Dockerfile b/backend/services/mining-wallet-service/Dockerfile new file mode 100644 index 00000000..63c8c39c --- /dev/null +++ b/backend/services/mining-wallet-service/Dockerfile @@ -0,0 +1,48 @@ +# ============================================================================= +# Mining Wallet Service - Dockerfile +# ============================================================================= + +# Stage 1: Build +FROM node:20-alpine AS builder +WORKDIR /app + +RUN apk add --no-cache python3 make g++ + +COPY package.json package-lock.json ./ +RUN npm ci + +COPY prisma ./prisma/ +RUN npx prisma generate + +COPY . . +RUN npm run build + +# Stage 2: Production +FROM node:20-alpine AS runner +WORKDIR /app + +ENV NODE_ENV=production +ENV TZ=Asia/Shanghai + +RUN apk add --no-cache curl tzdata + +RUN addgroup --system --gid 1001 nodejs && \ + adduser --system --uid 1001 nestjs + +COPY package.json package-lock.json ./ +RUN npm ci --only=production && npm cache clean --force + +COPY prisma ./prisma/ +RUN npx prisma generate + +COPY --from=builder /app/dist ./dist + +RUN chown -R nestjs:nodejs /app +USER nestjs + +EXPOSE 3025 + +HEALTHCHECK --interval=30s --timeout=10s --start-period=10s --retries=3 \ + CMD curl -f http://localhost:3025/health || exit 1 + +CMD ["node", "dist/main.js"] diff --git a/backend/services/mining-wallet-service/nest-cli.json b/backend/services/mining-wallet-service/nest-cli.json new file mode 100644 index 00000000..f9aa683b --- /dev/null +++ b/backend/services/mining-wallet-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/mining-wallet-service/package-lock.json b/backend/services/mining-wallet-service/package-lock.json new file mode 100644 index 00000000..4712d799 --- /dev/null +++ b/backend/services/mining-wallet-service/package-lock.json @@ -0,0 +1,10383 @@ +{ + "name": "mining-wallet-service", + "version": "1.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "mining-wallet-service", + "version": "1.0.0", + "dependencies": { + "@nestjs/common": "^10.3.0", + "@nestjs/config": "^3.1.1", + "@nestjs/core": "^10.3.0", + "@nestjs/microservices": "^10.3.0", + "@nestjs/platform-express": "^10.3.0", + "@nestjs/schedule": "^4.0.0", + "@nestjs/swagger": "^7.1.17", + "@prisma/client": "^5.7.1", + "class-transformer": "^0.5.1", + "class-validator": "^0.14.0", + "decimal.js": "^10.4.3", + "ethers": "^6.9.0", + "ioredis": "^5.3.2", + "jsonwebtoken": "^9.0.2", + "kafkajs": "^2.2.4", + "reflect-metadata": "^0.1.14", + "rxjs": "^7.8.1", + "swagger-ui-express": "^5.0.0" + }, + "devDependencies": { + "@nestjs/cli": "^10.2.1", + "@nestjs/schematics": "^10.0.3", + "@nestjs/testing": "^10.3.0", + "@types/express": "^4.17.21", + "@types/jest": "^29.5.11", + "@types/jsonwebtoken": "^9.0.5", + "@types/node": "^20.10.5", + "@typescript-eslint/eslint-plugin": "^6.15.0", + "@typescript-eslint/parser": "^6.15.0", + "eslint": "^8.56.0", + "eslint-config-prettier": "^9.1.0", + "eslint-plugin-prettier": "^5.1.0", + "jest": "^29.7.0", + "prettier": "^3.1.1", + "prisma": "^5.7.1", + "ts-jest": "^29.1.1", + "ts-loader": "^9.5.1", + "ts-node": "^10.9.2", + "tsconfig-paths": "^4.2.0", + "typescript": "^5.3.3" + } + }, + "node_modules/@adraffy/ens-normalize": { + "version": "1.10.1", + "resolved": "https://registry.npmjs.org/@adraffy/ens-normalize/-/ens-normalize-1.10.1.tgz", + "integrity": "sha512-96Z2IP3mYmF1Xg2cDm8f1gWGf/HUVedQ3FMifV4kG/PQ4yEP51xDtRAEfhVNt5f/uzpNkZHwWQuUcu6D6K+Ekw==", + "license": "MIT" + }, + "node_modules/@angular-devkit/core": { + "version": "17.3.11", + "resolved": "https://registry.npmjs.org/@angular-devkit/core/-/core-17.3.11.tgz", + "integrity": "sha512-vTNDYNsLIWpYk2I969LMQFH29GTsLzxNk/0cLw5q56ARF0v5sIWfHYwGTS88jdDqIpuuettcSczbxeA7EuAmqQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ajv": "8.12.0", + "ajv-formats": "2.1.1", + "jsonc-parser": "3.2.1", + "picomatch": "4.0.1", + "rxjs": "7.8.1", + "source-map": "0.7.4" + }, + "engines": { + "node": "^18.13.0 || >=20.9.0", + "npm": "^6.11.0 || ^7.5.6 || >=8.0.0", + "yarn": ">= 1.13.0" + }, + "peerDependencies": { + "chokidar": "^3.5.2" + }, + "peerDependenciesMeta": { + "chokidar": { + "optional": true + } + } + }, + "node_modules/@angular-devkit/core/node_modules/rxjs": { + "version": "7.8.1", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-7.8.1.tgz", + "integrity": "sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.1.0" + } + }, + "node_modules/@angular-devkit/schematics": { + "version": "17.3.11", + "resolved": "https://registry.npmjs.org/@angular-devkit/schematics/-/schematics-17.3.11.tgz", + "integrity": "sha512-I5wviiIqiFwar9Pdk30Lujk8FczEEc18i22A5c6Z9lbmhPQdTroDnEQdsfXjy404wPe8H62s0I15o4pmMGfTYQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@angular-devkit/core": "17.3.11", + "jsonc-parser": "3.2.1", + "magic-string": "0.30.8", + "ora": "5.4.1", + "rxjs": "7.8.1" + }, + "engines": { + "node": "^18.13.0 || >=20.9.0", + "npm": "^6.11.0 || ^7.5.6 || >=8.0.0", + "yarn": ">= 1.13.0" + } + }, + "node_modules/@angular-devkit/schematics-cli": { + "version": "17.3.11", + "resolved": "https://registry.npmjs.org/@angular-devkit/schematics-cli/-/schematics-cli-17.3.11.tgz", + "integrity": "sha512-kcOMqp+PHAKkqRad7Zd7PbpqJ0LqLaNZdY1+k66lLWmkEBozgq8v4ASn/puPWf9Bo0HpCiK+EzLf0VHE8Z/y6Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@angular-devkit/core": "17.3.11", + "@angular-devkit/schematics": "17.3.11", + "ansi-colors": "4.1.3", + "inquirer": "9.2.15", + "symbol-observable": "4.0.0", + "yargs-parser": "21.1.1" + }, + "bin": { + "schematics": "bin/schematics.js" + }, + "engines": { + "node": "^18.13.0 || >=20.9.0", + "npm": "^6.11.0 || ^7.5.6 || >=8.0.0", + "yarn": ">= 1.13.0" + } + }, + "node_modules/@angular-devkit/schematics-cli/node_modules/chalk": { + "version": "5.6.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.6.2.tgz", + "integrity": "sha512-7NzBL0rN6fMUW+f7A6Io4h40qQlG+xGmtMxfbnH/K7TAtt8JQWVQK+6g0UXKMeVJoyV5EkkNsErQ8pVD3bLHbA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.17.0 || ^14.13 || >=16.0.0" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/@angular-devkit/schematics-cli/node_modules/cli-width": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-4.1.0.tgz", + "integrity": "sha512-ouuZd4/dm2Sw5Gmqy6bGyNNNe1qt9RpmxveLSO7KcgsTnU7RXfsw+/bukWGo1abgBiMAic068rclZsO4IWmmxQ==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">= 12" + } + }, + "node_modules/@angular-devkit/schematics-cli/node_modules/inquirer": { + "version": "9.2.15", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-9.2.15.tgz", + "integrity": "sha512-vI2w4zl/mDluHt9YEQ/543VTCwPKWiHzKtm9dM2V0NdFcqEexDAjUHzO1oA60HRNaVifGXXM1tRRNluLVHa0Kg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@ljharb/through": "^2.3.12", + "ansi-escapes": "^4.3.2", + "chalk": "^5.3.0", + "cli-cursor": "^3.1.0", + "cli-width": "^4.1.0", + "external-editor": "^3.1.0", + "figures": "^3.2.0", + "lodash": "^4.17.21", + "mute-stream": "1.0.0", + "ora": "^5.4.1", + "run-async": "^3.0.0", + "rxjs": "^7.8.1", + "string-width": "^4.2.3", + "strip-ansi": "^6.0.1", + "wrap-ansi": "^6.2.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/@angular-devkit/schematics-cli/node_modules/mute-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-1.0.0.tgz", + "integrity": "sha512-avsJQhyd+680gKXyG/sQc0nXaC6rBkPOfyHYcFb9+hdkqQkR9bdnkJ0AMZhke0oesPqIO+mFFJ+IdBc7mst4IA==", + "dev": true, + "license": "ISC", + "engines": { + "node": "^14.17.0 || ^16.13.0 || >=18.0.0" + } + }, + "node_modules/@angular-devkit/schematics-cli/node_modules/run-async": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-3.0.0.tgz", + "integrity": "sha512-540WwVDOMxA6dN6We19EcT9sc3hkXPw5mzRNGM3FkdN/vtE9NFvj5lFAPNwUDmJjXidm3v7TC1cTE7t17Ulm1Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/@angular-devkit/schematics/node_modules/rxjs": { + "version": "7.8.1", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-7.8.1.tgz", + "integrity": "sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.1.0" + } + }, + "node_modules/@babel/code-frame": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.27.1.tgz", + "integrity": "sha512-cjQ7ZlQ0Mv3b47hABuTevyTuYN4i+loJKGeV9flcCgIK37cCXRh+L1bd3iBHlynerhQ7BhCkn2BPbQUL+rGqFg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-validator-identifier": "^7.27.1", + "js-tokens": "^4.0.0", + "picocolors": "^1.1.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.28.5.tgz", + "integrity": "sha512-6uFXyCayocRbqhZOB+6XcuZbkMNimwfVGFji8CTZnCzOHVGvDqzvitu1re2AU5LROliz7eQPhB8CpAMvnx9EjA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.28.5.tgz", + "integrity": "sha512-e7jT4DxYvIDLk1ZHmU/m/mB19rex9sv0c2ftBtjSBv+kVM/902eh0fINUzD7UwLLNR+jU585GxUJ8/EBfAM5fw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@babel/generator": "^7.28.5", + "@babel/helper-compilation-targets": "^7.27.2", + "@babel/helper-module-transforms": "^7.28.3", + "@babel/helpers": "^7.28.4", + "@babel/parser": "^7.28.5", + "@babel/template": "^7.27.2", + "@babel/traverse": "^7.28.5", + "@babel/types": "^7.28.5", + "@jridgewell/remapping": "^2.3.5", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/core/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/generator": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.28.5.tgz", + "integrity": "sha512-3EwLFhZ38J4VyIP6WNtt2kUdW9dokXA9Cr4IVIFHuCpZ3H8/YFOl5JjZHisrn1fATPBmKKqXzDFvh9fUwHz6CQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.28.5", + "@babel/types": "^7.28.5", + "@jridgewell/gen-mapping": "^0.3.12", + "@jridgewell/trace-mapping": "^0.3.28", + "jsesc": "^3.0.2" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.27.2", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.27.2.tgz", + "integrity": "sha512-2+1thGUUWWjLTYTHZWK1n8Yga0ijBz1XAhUXcKy81rd5g6yh7hGqMp45v7cadSbEHc9G3OTv45SyneRN3ps4DQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/compat-data": "^7.27.2", + "@babel/helper-validator-option": "^7.27.1", + "browserslist": "^4.24.0", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/helper-globals": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz", + "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.27.1.tgz", + "integrity": "sha512-0gSFWUPNXNopqtIPQvlD5WgXYI5GY2kP2cCvoT8kczjbfcfuIljTbcWrulD1CIPIX2gt1wghbDy08yE1p+/r3w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/traverse": "^7.27.1", + "@babel/types": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.28.3", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.3.tgz", + "integrity": "sha512-gytXUbs8k2sXS9PnQptz5o0QnpLL51SwASIORY6XaBKF88nsOT0Zw9szLqlSGQDP/4TljBAD5y98p2U1fqkdsw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.27.1", + "@babel/helper-validator-identifier": "^7.27.1", + "@babel/traverse": "^7.28.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-plugin-utils": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.27.1.tgz", + "integrity": "sha512-1gn1Up5YXka3YYAHGKpbideQ5Yjf1tDa9qYcgysz+cNCXukyLl6DjPXhD3VRwSb8c0J9tA4b2+rHEZtc6R0tlw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz", + "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz", + "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz", + "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.28.4", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.28.4.tgz", + "integrity": "sha512-HFN59MmQXGHVyYadKLVumYsA9dBFun/ldYxipEjzA4196jpLZd8UjEEBLkbEkvfYreDqJhZxYAWFPtrfhNpj4w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.27.2", + "@babel/types": "^7.28.4" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.28.5.tgz", + "integrity": "sha512-KKBU1VGYR7ORr3At5HAtUQ+TV3SzRCXmA/8OdDZiLDBIZxVyzXuztPjfLd3BV1PRAQGCMWWSHYhL0F8d5uHBDQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.5" + }, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/plugin-syntax-async-generators": { + "version": "7.8.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", + "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-bigint": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-bigint/-/plugin-syntax-bigint-7.8.3.tgz", + "integrity": "sha512-wnTnFlG+YxQm3vDxpGE57Pj0srRU4sHE/mDkt1qv2YJJSeUAec2ma4WLUnUPeKjyrfntVwe/N6dCXpU+zL3Npg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-class-properties": { + "version": "7.12.13", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.13.tgz", + "integrity": "sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.12.13" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-class-static-block": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-static-block/-/plugin-syntax-class-static-block-7.14.5.tgz", + "integrity": "sha512-b+YyPmr6ldyNnM6sqYeMWE+bgJcJpO6yS4QD7ymxgH34GBPNDM/THBh8iunyvKIZztiwLH4CJZ0RxTk9emgpjw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-attributes": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-attributes/-/plugin-syntax-import-attributes-7.27.1.tgz", + "integrity": "sha512-oFT0FrKHgF53f4vOsZGi2Hh3I35PfSmVs4IBFLFj4dnafP+hIWDLg3VyKmUHfLoLHlyxY4C7DGtmHuJgn+IGww==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-meta": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-meta/-/plugin-syntax-import-meta-7.10.4.tgz", + "integrity": "sha512-Yqfm+XDx0+Prh3VSeEQCPU81yC+JWZ2pDPFSS4ZdpfZhp4MkFMaDC1UqseovEKwSUpnIL7+vK+Clp7bfh0iD7g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-json-strings": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz", + "integrity": "sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-jsx": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.27.1.tgz", + "integrity": "sha512-y8YTNIeKoyhGd9O0Jiyzyyqk8gdjnumGTQPsz0xOZOQ2RmkVJeZ1vmmfIvFEKqucBG6axJGBZDE/7iI5suUI/w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-logical-assignment-operators": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz", + "integrity": "sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-nullish-coalescing-operator": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz", + "integrity": "sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-numeric-separator": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz", + "integrity": "sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-object-rest-spread": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz", + "integrity": "sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-catch-binding": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz", + "integrity": "sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-chaining": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz", + "integrity": "sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-private-property-in-object": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-private-property-in-object/-/plugin-syntax-private-property-in-object-7.14.5.tgz", + "integrity": "sha512-0wVnp9dxJ72ZUJDV27ZfbSj6iHLoytYZmh3rFcxNnvsJF3ktkzLDZPy/mA17HGsaQT3/DQsWYX1f1QGWkCoVUg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-top-level-await": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.14.5.tgz", + "integrity": "sha512-hx++upLv5U1rgYfwe1xBQUhRmU41NEvpUvrp8jkrSCdvGSnM5/qdRMtylJ6PG5OFkBaHkbTAKTnd3/YyESRHFw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-typescript": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-typescript/-/plugin-syntax-typescript-7.27.1.tgz", + "integrity": "sha512-xfYCBMxveHrRMnAWl1ZlPXOZjzkN82THFvLhQhFXFt81Z5HnN+EtUkZhv/zcKpmT3fzmWZB0ywiBrbC3vogbwQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/template": { + "version": "7.27.2", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.27.2.tgz", + "integrity": "sha512-LPDZ85aEJyYSd18/DkjNh4/y1ntkE5KwUHWTiqgRxruuZL2F1yuHligVHLvcHY2vMHXttKFpJn6LwfI7cw7ODw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@babel/parser": "^7.27.2", + "@babel/types": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.28.5.tgz", + "integrity": "sha512-TCCj4t55U90khlYkVV/0TfkJkAkUg3jZFA3Neb7unZT8CPok7iiRfaX0F+WnqWqt7OxhOn0uBKXCw4lbL8W0aQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@babel/generator": "^7.28.5", + "@babel/helper-globals": "^7.28.0", + "@babel/parser": "^7.28.5", + "@babel/template": "^7.27.2", + "@babel/types": "^7.28.5", + "debug": "^4.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.28.5.tgz", + "integrity": "sha512-qQ5m48eI/MFLQ5PxQj4PFaprjyCTLI37ElWMmNs0K8Lk3dVeOdNpB3ks8jc7yM5CDmVC73eMVk/trk3fgmrUpA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.27.1", + "@babel/helper-validator-identifier": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@bcoe/v8-coverage": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz", + "integrity": "sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@borewit/text-codec": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/@borewit/text-codec/-/text-codec-0.2.1.tgz", + "integrity": "sha512-k7vvKPbf7J2fZ5klGRD9AeKfUvojuZIQ3BT5u7Jfv+puwXkUBUT5PVyMDfJZpy30CBDXGMgw7fguK/lpOMBvgw==", + "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.1", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.1.tgz", + "integrity": "sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "eslint-visitor-keys": "^3.4.3" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" + } + }, + "node_modules/@eslint-community/regexpp": { + "version": "4.12.2", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", + "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/eslintrc": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.4.tgz", + "integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^9.6.0", + "globals": "^13.19.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.0", + "minimatch": "^3.1.2", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint/eslintrc/node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/@eslint/eslintrc/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/@eslint/eslintrc/node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true, + "license": "MIT" + }, + "node_modules/@eslint/eslintrc/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/@eslint/js": { + "version": "8.57.1", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.57.1.tgz", + "integrity": "sha512-d9zaMRSTIKDLhctzH12MtXvJKSSUhaHcjV+2Z+GK+EEY7XKpP5yR4x+N3TAcHTcu963nIr+TMcCb4DBCYX1z6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + } + }, + "node_modules/@humanwhocodes/config-array": { + "version": "0.13.0", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.13.0.tgz", + "integrity": "sha512-DZLEEqFWQFiyK6h5YIeynKx7JlvCYWL0cImfSRXZ9l4Sg2efkFGTuFf6vzXjK1cq6IYkU+Eg/JizXw+TD2vRNw==", + "deprecated": "Use @eslint/config-array instead", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@humanwhocodes/object-schema": "^2.0.3", + "debug": "^4.3.1", + "minimatch": "^3.0.5" + }, + "engines": { + "node": ">=10.10.0" + } + }, + "node_modules/@humanwhocodes/config-array/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/@humanwhocodes/config-array/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=12.22" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@humanwhocodes/object-schema": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-2.0.3.tgz", + "integrity": "sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==", + "deprecated": "Use @eslint/object-schema instead", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/@ioredis/commands": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/@ioredis/commands/-/commands-1.5.0.tgz", + "integrity": "sha512-eUgLqrMf8nJkZxT24JvVRrQya1vZkQh8BBeYNwGDqa5I0VUi8ACx7uFvAaLxintokpTenkK6DASvo/bvNbBGow==", + "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/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/@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/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/@nestjs/cli/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==", + "dev": true, + "license": "MIT", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "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.22", + "resolved": "https://registry.npmjs.org/@nestjs/common/-/common-10.4.22.tgz", + "integrity": "sha512-fxJ4v85nDHaqT1PmfNCQ37b/jcv2OojtXTaK1P2uAXhzLf9qq6WNUOFvxBrV4fhQek1EQoT1o9oj5xAZmv3NRw==", + "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.22", + "resolved": "https://registry.npmjs.org/@nestjs/core/-/core-10.4.22.tgz", + "integrity": "sha512-6IX9+VwjiKtCjx+mXVPncpkQ5ZjKfmssOZPFexmT+6T9H9wZ3svpYACAo7+9e7Nr9DZSoRZw3pffkJP7Z0UjaA==", + "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/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.22", + "resolved": "https://registry.npmjs.org/@nestjs/microservices/-/microservices-10.4.22.tgz", + "integrity": "sha512-9Oxc0jQuppGLaQv5yaB2tVS2rAZzZ9NqDS1A4UlDLiYwJB7M6e89G6tmyOQjGjPwgoXPxQS4Vg2voSiKiED2gw==", + "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/platform-express": { + "version": "10.4.22", + "resolved": "https://registry.npmjs.org/@nestjs/platform-express/-/platform-express-10.4.22.tgz", + "integrity": "sha512-ySSq7Py/DFozzZdNDH67m/vHoeVdphDniWBnl6q5QVoXldDdrZIHLXLRMPayTDh5A95nt7jjJzmD4qpTbNQ6tA==", + "license": "MIT", + "peer": true, + "dependencies": { + "body-parser": "1.20.4", + "cors": "2.8.5", + "express": "4.22.1", + "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/platform-express/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/@nestjs/platform-express/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/@nestjs/platform-express/node_modules/cookie-signature": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.7.tgz", + "integrity": "sha512-NXdYc3dLr47pBkpUCHtKSwIOQXLVn8dZEuywboCOJY/osA0wFSLlSawr3KN8qXJEyX66FcONTH8EIlVuK0yyFA==", + "license": "MIT" + }, + "node_modules/@nestjs/platform-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/@nestjs/platform-express/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/@nestjs/platform-express/node_modules/express": { + "version": "4.22.1", + "resolved": "https://registry.npmjs.org/express/-/express-4.22.1.tgz", + "integrity": "sha512-F2X8g9P1X7uCPZMA3MVf9wcTqlyNp7IhH5qPCI0izhaOIYXaW9L535tGA3qmjRzpH+bZczqq7hVKxTR4NWnu+g==", + "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.14.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/@nestjs/platform-express/node_modules/finalhandler": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.2.tgz", + "integrity": "sha512-aA4RyPcd3badbdABGDuTXCMTtOneUCAYH/gxoYRTZlIJdF0YPWuGqiAsIrhNnnqdXGswYk6dGujem4w80UJFhg==", + "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.2", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/@nestjs/platform-express/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/@nestjs/platform-express/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/@nestjs/platform-express/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/@nestjs/platform-express/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/@nestjs/platform-express/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/@nestjs/platform-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/@nestjs/platform-express/node_modules/send": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/send/-/send-0.19.2.tgz", + "integrity": "sha512-VMbMxbDeehAxpOtWJXlcUS5E8iXh6QmN+BkRX1GARS3wRaXEEgzCcB10gTQazO42tpNIya8xIyNx8fll1OFPrg==", + "license": "MIT", + "dependencies": { + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "1.2.0", + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "~0.5.2", + "http-errors": "~2.0.1", + "mime": "1.6.0", + "ms": "2.1.3", + "on-finished": "~2.4.1", + "range-parser": "~1.2.1", + "statuses": "~2.0.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/@nestjs/platform-express/node_modules/serve-static": { + "version": "1.16.3", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.3.tgz", + "integrity": "sha512-x0RTqQel6g5SY7Lg6ZreMmsOzncHFU7nhnRWkKgWuMTu5NN0DR5oruckMqRvacAN9d5w6ARnRBXl9xhDCgfMeA==", + "license": "MIT", + "dependencies": { + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "~0.19.1" + }, + "engines": { + "node": ">= 0.8.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/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.22", + "resolved": "https://registry.npmjs.org/@nestjs/testing/-/testing-10.4.22.tgz", + "integrity": "sha512-HO9aPus3bAedAC+jKVAA8jTdaj4fs5M9fing4giHrcYV2txe9CvC1l1WAjwQ9RDhEHdugjY4y+FZA/U/YqPZrA==", + "dev": true, + "license": "MIT", + "dependencies": { + "tslib": "2.8.1" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/nest" + }, + "peerDependencies": { + "@nestjs/common": "^10.0.0", + "@nestjs/core": "^10.0.0", + "@nestjs/microservices": "^10.0.0", + "@nestjs/platform-express": "^10.0.0" + }, + "peerDependenciesMeta": { + "@nestjs/microservices": { + "optional": true + }, + "@nestjs/platform-express": { + "optional": true + } + } + }, + "node_modules/@noble/curves": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@noble/curves/-/curves-1.2.0.tgz", + "integrity": "sha512-oYclrNgRaM9SsBUBVbb8M6DTV7ZHRTKugureoYEncY5c65HOmRzvSiTE3y5CYaPYJA/GVkrhXEoF0M3Ya9PMnw==", + "license": "MIT", + "dependencies": { + "@noble/hashes": "1.3.2" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + } + }, + "node_modules/@noble/hashes": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/@noble/hashes/-/hashes-1.3.2.tgz", + "integrity": "sha512-MVC8EAQp7MvEcm30KWENFjgR+Mkmf+D189XJTkFIlwohU5hcBbn1ZkKq7KVTi2Hme3PMGF390DaL52beVrIihQ==", + "license": "MIT", + "engines": { + "node": ">= 16" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + } + }, + "node_modules/@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/@pkgjs/parseargs": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz", + "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==", + "dev": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=14" + } + }, + "node_modules/@pkgr/core": { + "version": "0.2.9", + "resolved": "https://registry.npmjs.org/@pkgr/core/-/core-0.2.9.tgz", + "integrity": "sha512-QNqXyfVS2wm9hweSYD2O7F0G06uurj9kZ96TRQE5Y9hU7+tgdZwIkbAKc5Ocy1HxEY2kuDQa6cQ1WRs/O5LFKA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.20.0 || ^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/pkgr" + } + }, + "node_modules/@prisma/client": { + "version": "5.22.0", + "resolved": "https://registry.npmjs.org/@prisma/client/-/client-5.22.0.tgz", + "integrity": "sha512-M0SVXfyHnQREBKxCgyo7sffrKttwE6R8PMq330MIUF0pTwjUhLbW84pFDlf06B27XyCR++VtjugEnIHdr07SVA==", + "hasInstallScript": true, + "license": "Apache-2.0", + "engines": { + "node": ">=16.13" + }, + "peerDependencies": { + "prisma": "*" + }, + "peerDependenciesMeta": { + "prisma": { + "optional": true + } + } + }, + "node_modules/@prisma/debug": { + "version": "5.22.0", + "resolved": "https://registry.npmjs.org/@prisma/debug/-/debug-5.22.0.tgz", + "integrity": "sha512-AUt44v3YJeggO2ZU5BkXI7M4hu9BF2zzH2iF2V5pyXT/lRTyWiElZ7It+bRH1EshoMRxHgpYg4VB6rCM+mG5jQ==", + "devOptional": true, + "license": "Apache-2.0" + }, + "node_modules/@prisma/engines": { + "version": "5.22.0", + "resolved": "https://registry.npmjs.org/@prisma/engines/-/engines-5.22.0.tgz", + "integrity": "sha512-UNjfslWhAt06kVL3CjkuYpHAWSO6L4kDCVPegV6itt7nD1kSJavd3vhgAEhjglLJJKEdJ7oIqDJ+yHk6qO8gPA==", + "devOptional": true, + "hasInstallScript": true, + "license": "Apache-2.0", + "dependencies": { + "@prisma/debug": "5.22.0", + "@prisma/engines-version": "5.22.0-44.605197351a3c8bdd595af2d2a9bc3025bca48ea2", + "@prisma/fetch-engine": "5.22.0", + "@prisma/get-platform": "5.22.0" + } + }, + "node_modules/@prisma/engines-version": { + "version": "5.22.0-44.605197351a3c8bdd595af2d2a9bc3025bca48ea2", + "resolved": "https://registry.npmjs.org/@prisma/engines-version/-/engines-version-5.22.0-44.605197351a3c8bdd595af2d2a9bc3025bca48ea2.tgz", + "integrity": "sha512-2PTmxFR2yHW/eB3uqWtcgRcgAbG1rwG9ZriSvQw+nnb7c4uCr3RAcGMb6/zfE88SKlC1Nj2ziUvc96Z379mHgQ==", + "devOptional": true, + "license": "Apache-2.0" + }, + "node_modules/@prisma/fetch-engine": { + "version": "5.22.0", + "resolved": "https://registry.npmjs.org/@prisma/fetch-engine/-/fetch-engine-5.22.0.tgz", + "integrity": "sha512-bkrD/Mc2fSvkQBV5EpoFcZ87AvOgDxbG99488a5cexp5Ccny+UM6MAe/UFkUC0wLYD9+9befNOqGiIJhhq+HbA==", + "devOptional": true, + "license": "Apache-2.0", + "dependencies": { + "@prisma/debug": "5.22.0", + "@prisma/engines-version": "5.22.0-44.605197351a3c8bdd595af2d2a9bc3025bca48ea2", + "@prisma/get-platform": "5.22.0" + } + }, + "node_modules/@prisma/get-platform": { + "version": "5.22.0", + "resolved": "https://registry.npmjs.org/@prisma/get-platform/-/get-platform-5.22.0.tgz", + "integrity": "sha512-pHhpQdr1UPFpt+zFfnPazhulaZYCUqeIcPpJViYoq9R+D/yw4fjE+CtnsnKzPYm0ddUbeXUzjGVGIRVgPDCk4Q==", + "devOptional": true, + "license": "Apache-2.0", + "dependencies": { + "@prisma/debug": "5.22.0" + } + }, + "node_modules/@sinclair/typebox": { + "version": "0.27.8", + "resolved": "https://registry.npmjs.org/@sinclair/typebox/-/typebox-0.27.8.tgz", + "integrity": "sha512-+Fj43pSMwJs4KRrH/938Uf+uAELIgVBmQzg/q1YG10djyfA3TnrU8N8XzqCh/okZdszqBQTZf96idMfE5lnwTA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@sinonjs/commons": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-3.0.1.tgz", + "integrity": "sha512-K3mCHKQ9sVh8o1C9cxkwxaOmXoAMlDxC1mYyHrjqOWEcBjYr76t96zL2zlj5dUGZ3HSw240X1qgH3Mjf1yJWpQ==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "type-detect": "4.0.8" + } + }, + "node_modules/@sinonjs/fake-timers": { + "version": "10.3.0", + "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-10.3.0.tgz", + "integrity": "sha512-V4BG07kuYSUkTCSBHG8G8TNhM+F19jXFWnQtzj+we8DrkpSBCee9Z3Ms8yiGer/dlmhe35/Xdgyo3/0rQKg7YA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@sinonjs/commons": "^3.0.0" + } + }, + "node_modules/@tokenizer/inflate": { + "version": "0.2.7", + "resolved": "https://registry.npmjs.org/@tokenizer/inflate/-/inflate-0.2.7.tgz", + "integrity": "sha512-MADQgmZT1eKjp06jpI2yozxaU9uVs4GzzgSL+uEq7bVcJ9V1ZXQkeGNql1fsSI0gMy1vhvNTNbUqrx+pZfJVmg==", + "license": "MIT", + "dependencies": { + "debug": "^4.4.0", + "fflate": "^0.8.2", + "token-types": "^6.0.0" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/Borewit" + } + }, + "node_modules/@tokenizer/token": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/@tokenizer/token/-/token-0.3.0.tgz", + "integrity": "sha512-OvjF+z51L3ov0OyAU0duzsYuvO01PH7x4t6DJx+guahgTnBHkhJdG7soQeTSFLWN3efnHyibZ4Z8l2EuWwJN3A==", + "license": "MIT" + }, + "node_modules/@tsconfig/node10": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.12.tgz", + "integrity": "sha512-UCYBaeFvM11aU2y3YPZ//O5Rhj+xKyzy7mvcIoAjASbigy8mHMryP5cK7dgjlz2hWxh1g5pLw084E0a/wlUSFQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@tsconfig/node12": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/@tsconfig/node12/-/node12-1.0.11.tgz", + "integrity": "sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==", + "dev": true, + "license": "MIT" + }, + "node_modules/@tsconfig/node14": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@tsconfig/node14/-/node14-1.0.3.tgz", + "integrity": "sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==", + "dev": true, + "license": "MIT" + }, + "node_modules/@tsconfig/node16": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/@tsconfig/node16/-/node16-1.0.4.tgz", + "integrity": "sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/babel__core": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz", + "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.20.7", + "@babel/types": "^7.20.7", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "node_modules/@types/babel__generator": { + "version": "7.27.0", + "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.27.0.tgz", + "integrity": "sha512-ufFd2Xi92OAVPYsy+P4n7/U7e68fex0+Ee8gSG9KX7eo084CWiQ4sdxktvdl0bOPupXtVJPY19zk6EwWqUQ8lg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__template": { + "version": "7.4.4", + "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz", + "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__traverse": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.28.0.tgz", + "integrity": "sha512-8PvcXf70gTDZBgt9ptxJ8elBeBjcLOAcOtoO/mPJjtji1+CdGbHgm77om1GrsPxsiE+uXIpNSK64UYaIwQXd4Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.2" + } + }, + "node_modules/@types/body-parser": { + "version": "1.19.6", + "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.6.tgz", + "integrity": "sha512-HLFeCYgz89uk22N5Qg3dvGvsv46B8GLvKKo1zKG4NybA8U2DiEO3w9lqGg29t/tfLRJpJ6iQxnVw4OnB7MoM9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/connect": "*", + "@types/node": "*" + } + }, + "node_modules/@types/connect": { + "version": "3.4.38", + "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.38.tgz", + "integrity": "sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/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.8", + "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.19.8.tgz", + "integrity": "sha512-02S5fmqeoKzVZCHPZid4b8JH2eM5HzQLZWN2FohQEy/0eXTq8VXZfSN6Pcr3F6N9R/vNrj7cpgbhjie6m/1tCA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*", + "@types/qs": "*", + "@types/range-parser": "*", + "@types/send": "*" + } + }, + "node_modules/@types/graceful-fs": { + "version": "4.1.9", + "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.9.tgz", + "integrity": "sha512-olP3sd1qOEe5dXTSaFvQG+02VdRXcdytWLAZsAq1PecU8uqQAhkrnbli7DagjtXKW/Bl7YJbUsa8MPcuc8LHEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/http-errors": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.5.tgz", + "integrity": "sha512-r8Tayk8HJnX0FztbZN7oVqGccWgw98T/0neJphO91KkmOzug1KkofZURD4UaD5uH8AqcFLfdPErnBod0u71/qg==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/istanbul-lib-coverage": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.6.tgz", + "integrity": "sha512-2QF/t/auWm0lsy8XtKVPG19v3sSOQlJe/YHZgfjb/KBBHOGSV+J2q/S671rcq9uTBrLAXmZpqJiaQbMT+zNU1w==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/istanbul-lib-report": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-report/-/istanbul-lib-report-3.0.3.tgz", + "integrity": "sha512-NQn7AHQnk/RSLOxrBbGyJM/aVQ+pjj5HCgasFxc0K/KhoATfQ/47AyUl15I2yBUpihjmas+a+VJBOqecrFH+uA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/istanbul-lib-coverage": "*" + } + }, + "node_modules/@types/istanbul-reports": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.4.tgz", + "integrity": "sha512-pk2B1NWalF9toCRu6gjBzR69syFjP4Od8WRAX+0mmf9lAjCRicLOWc+ZrxZHx/0XRjotgkF9t6iaMJ+aXcOdZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/istanbul-lib-report": "*" + } + }, + "node_modules/@types/jest": { + "version": "29.5.14", + "resolved": "https://registry.npmjs.org/@types/jest/-/jest-29.5.14.tgz", + "integrity": "sha512-ZN+4sdnLUbo8EVvVc2ao0GFW6oVrQRPn4K2lglySj7APvSrgzxHiNNK99us4WDMi57xxA2yggblIAMNhXOotLQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "expect": "^29.0.0", + "pretty-format": "^29.0.0" + } + }, + "node_modules/@types/json-schema": { + "version": "7.0.15", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/jsonwebtoken": { + "version": "9.0.10", + "resolved": "https://registry.npmjs.org/@types/jsonwebtoken/-/jsonwebtoken-9.0.10.tgz", + "integrity": "sha512-asx5hIG9Qmf/1oStypjanR7iKTv0gXQ1Ov/jfrX6kS/EO0OFni8orbmGCn0672NHR3kXHwpAwR+B368ZGN/2rA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/ms": "*", + "@types/node": "*" + } + }, + "node_modules/@types/luxon": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/@types/luxon/-/luxon-3.4.2.tgz", + "integrity": "sha512-TifLZlFudklWlMBfhubvgqTXRzLDI5pCbGa4P8a3wPyUQSW+1xQ5eDsreP9DWHX3tjq1ke96uYG/nwundroWcA==", + "license": "MIT" + }, + "node_modules/@types/mime": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.5.tgz", + "integrity": "sha512-/pyBZWSLD2n0dcHE3hq8s8ZvcETHtEuF+3E7XVt0Ig2nvsVQXdghHVcEkIWjy9A0wKfTn97a/PSDYohKIlnP/w==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/ms": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@types/ms/-/ms-2.1.0.tgz", + "integrity": "sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/node": { + "version": "20.19.28", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.28.tgz", + "integrity": "sha512-VyKBr25BuFDzBFCK5sUM6ZXiWfqgCTwTAOK8qzGV/m9FCirXYDlmczJ+d5dXBAQALGCdRRdbteKYfJ84NGEusw==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "undici-types": "~6.21.0" + } + }, + "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/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": "2.0.0", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-2.0.0.tgz", + "integrity": "sha512-5cvg6CtKwfgdmVqY1WIiXKc3Q1bkRqGLi+2W/6ao+6Y7gu/RCwRuAhGEzh5B4KlszSuTLgZYuqFqo5bImjNKng==", + "license": "MIT", + "dependencies": { + "mime-types": "^3.0.0", + "negotiator": "^1.0.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/acorn": { + "version": "8.15.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", + "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", + "dev": true, + "license": "MIT", + "peer": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-import-phases": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/acorn-import-phases/-/acorn-import-phases-1.0.4.tgz", + "integrity": "sha512-wKmbr/DDiIXzEOiWrTTUcDm24kQ2vGfZQvM2fwg2vXqR5uW6aapr7ObPtj1th32b9u90/Pf4AItvdTh42fBmVQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10.13.0" + }, + "peerDependencies": { + "acorn": "^8.14.0" + } + }, + "node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/acorn-walk": { + "version": "8.3.4", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.3.4.tgz", + "integrity": "sha512-ueEepnujpqee2o5aIYnvHU6C0A42MNdsIDeqy5BydrkuC5R1ZuUFnm27EeFJGoEHJQgn3uleRvmTXaJgfXbt4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "acorn": "^8.11.0" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/aes-js": { + "version": "4.0.0-beta.5", + "resolved": "https://registry.npmjs.org/aes-js/-/aes-js-4.0.0-beta.5.tgz", + "integrity": "sha512-G965FqalsNyrPqgEGON7nIx1e/OVENSgiEIzyC63haUMuvNnwIgIjMs52hlTCKhkBny7A2ORNlfY9Zu+jmGk1Q==", + "license": "MIT" + }, + "node_modules/ajv": { + "version": "8.12.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.12.0.tgz", + "integrity": "sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "fast-deep-equal": "^3.1.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/ajv-formats": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ajv-formats/-/ajv-formats-2.1.1.tgz", + "integrity": "sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ajv": "^8.0.0" + }, + "peerDependencies": { + "ajv": "^8.0.0" + }, + "peerDependenciesMeta": { + "ajv": { + "optional": true + } + } + }, + "node_modules/ajv-keywords": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-5.1.0.tgz", + "integrity": "sha512-YCS/JNFAUyr5vAuhk1DWm1CBxRHW9LbJ2ozWeemrIqpbsqKjHVxYPyi5GC0rjZIT5JxJ3virVTS8wk4i/Z+krw==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-deep-equal": "^3.1.3" + }, + "peerDependencies": { + "ajv": "^8.8.2" + } + }, + "node_modules/ansi-colors": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.3.tgz", + "integrity": "sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "type-fest": "^0.21.3" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/ansi-escapes/node_modules/type-fest": { + "version": "0.21.3", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.21.3.tgz", + "integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==", + "dev": true, + "license": "(MIT OR CC0-1.0)", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "license": "MIT", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/anymatch": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", + "dev": true, + "license": "ISC", + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/anymatch/node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/append-field": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/append-field/-/append-field-1.0.0.tgz", + "integrity": "sha512-klpgFSWLW1ZEs8svjfb7g4qWY0YS5imI82dTg+QahUvJ8YqAY0P10Uk8tTyh9ZGuYEZEMaeJYCF5BFuX552hsw==", + "license": "MIT" + }, + "node_modules/arg": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", + "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", + "dev": true, + "license": "MIT" + }, + "node_modules/argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "license": "Python-2.0" + }, + "node_modules/array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==", + "license": "MIT" + }, + "node_modules/array-timsort": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/array-timsort/-/array-timsort-1.0.3.tgz", + "integrity": "sha512-/+3GRL7dDAGEfM6TseQk/U+mi18TU2Ms9I3UlLdUMhz2hbvGNTKdj9xniwXfUqgYhHxRx0+8UnKkvlNwVU+cWQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/array-union": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", + "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-29.7.0.tgz", + "integrity": "sha512-BrvGY3xZSwEcCzKvKsCi2GgHqDqsYkOP4/by5xCgIwGXQxIEh+8ew3gmrE1y7XRR6LHZIj6yLYnUi/mm2KXKBg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/transform": "^29.7.0", + "@types/babel__core": "^7.1.14", + "babel-plugin-istanbul": "^6.1.1", + "babel-preset-jest": "^29.6.3", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.8.0" + } + }, + "node_modules/babel-plugin-istanbul": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-6.1.1.tgz", + "integrity": "sha512-Y1IQok9821cC9onCx5otgFfRm7Lm+I+wwxOx738M/WLPZ9Q42m4IG5W0FNX8WLL2gYMZo3JkuXIH2DOpWM+qwA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/helper-plugin-utils": "^7.0.0", + "@istanbuljs/load-nyc-config": "^1.0.0", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-instrument": "^5.0.4", + "test-exclude": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-plugin-istanbul/node_modules/istanbul-lib-instrument": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-5.2.1.tgz", + "integrity": "sha512-pzqtp31nLv/XFOzXGuvhCb8qhjmTVo5vjVk19XE4CRlSWz0KoeJ3bw9XsA7nOp9YBf4qHjwBxkDzKcME/J29Yg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/core": "^7.12.3", + "@babel/parser": "^7.14.7", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-plugin-istanbul/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/babel-plugin-jest-hoist": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-29.6.3.tgz", + "integrity": "sha512-ESAc/RJvGTFEzRwOTT4+lNDk/GNHMkKbNzsvT0qKRfDyyYTskxB5rnU2njIDYVxXCBHHEI1c0YwHob3WaYujOg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.3.3", + "@babel/types": "^7.3.3", + "@types/babel__core": "^7.1.14", + "@types/babel__traverse": "^7.0.6" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/babel-preset-current-node-syntax": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.2.0.tgz", + "integrity": "sha512-E/VlAEzRrsLEb2+dv8yp3bo4scof3l9nR4lrld+Iy5NyVqgVYUJnDAmunkhPMisRI32Qc4iRiz425d8vM++2fg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/plugin-syntax-async-generators": "^7.8.4", + "@babel/plugin-syntax-bigint": "^7.8.3", + "@babel/plugin-syntax-class-properties": "^7.12.13", + "@babel/plugin-syntax-class-static-block": "^7.14.5", + "@babel/plugin-syntax-import-attributes": "^7.24.7", + "@babel/plugin-syntax-import-meta": "^7.10.4", + "@babel/plugin-syntax-json-strings": "^7.8.3", + "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", + "@babel/plugin-syntax-numeric-separator": "^7.10.4", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", + "@babel/plugin-syntax-optional-chaining": "^7.8.3", + "@babel/plugin-syntax-private-property-in-object": "^7.14.5", + "@babel/plugin-syntax-top-level-await": "^7.14.5" + }, + "peerDependencies": { + "@babel/core": "^7.0.0 || ^8.0.0-0" + } + }, + "node_modules/babel-preset-jest": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-29.6.3.tgz", + "integrity": "sha512-0B3bhxR6snWXJZtR/RliHTDPRgn1sNHOR0yVtq/IiQFyuOVjFS+wuio/R4gSNkyYmKmJB4wGZv2NZanmKmTnNA==", + "dev": true, + "license": "MIT", + "dependencies": { + "babel-plugin-jest-hoist": "^29.6.3", + "babel-preset-current-node-syntax": "^1.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/base64-js": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/baseline-browser-mapping": { + "version": "2.9.14", + "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.9.14.tgz", + "integrity": "sha512-B0xUquLkiGLgHhpPBqvl7GWegWBUNuujQ6kXd/r1U38ElPT6Ok8KZ8e+FpUGEc2ZoRQUzq/aUnaKFc/svWUGSg==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "baseline-browser-mapping": "dist/cli.js" + } + }, + "node_modules/binary-extensions": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", + "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/bl": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", + "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "buffer": "^5.5.0", + "inherits": "^2.0.4", + "readable-stream": "^3.4.0" + } + }, + "node_modules/body-parser": { + "version": "1.20.4", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.4.tgz", + "integrity": "sha512-ZTgYYLMOXY9qKU/57FAo8F+HA2dGX7bqGc71txDRC1rS4frdFI5R7NhluHxH6M0YItAP0sHB4uqAOcYKxO6uGA==", + "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.1", + "iconv-lite": "~0.4.24", + "on-finished": "~2.4.1", + "qs": "~6.14.0", + "raw-body": "~2.5.3", + "type-is": "~1.6.18", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, + "node_modules/body-parser/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "license": "MIT", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/body-parser/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", + "license": "MIT" + }, + "node_modules/brace-expansion": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz", + "integrity": "sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/braces": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", + "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", + "dev": true, + "license": "MIT", + "dependencies": { + "fill-range": "^7.1.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/browserslist": { + "version": "4.28.1", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.1.tgz", + "integrity": "sha512-ZC5Bd0LgJXgwGqUknZY/vkUQ04r8NXnJZ3yYi4vDmSiZmC/pdSN0NbNRPxZpbtO4uAfDUAFffO8IZoM3Gj8IkA==", + "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.9.0", + "caniuse-lite": "^1.0.30001759", + "electron-to-chromium": "^1.5.263", + "node-releases": "^2.0.27", + "update-browserslist-db": "^1.2.0" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/bs-logger": { + "version": "0.2.6", + "resolved": "https://registry.npmjs.org/bs-logger/-/bs-logger-0.2.6.tgz", + "integrity": "sha512-pd8DCoxmbgc7hyPKOvxtqNcjYoOsABPQdcCUjGp3d42VR2CX1ORhk2A87oqqu5R1kk+76nsxZupkmyd+MVtCog==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-json-stable-stringify": "2.x" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/bser": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/bser/-/bser-2.1.1.tgz", + "integrity": "sha512-gQxTNE/GAfIIrmHLUE3oJyp5FO6HRBfhjnw4/wMmA63ZGDJnWBmgY/lyQBpnDUkGmAhbSe39tx2d/iTOAfglwQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "node-int64": "^0.4.0" + } + }, + "node_modules/buffer": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", + "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT", + "dependencies": { + "base64-js": "^1.3.1", + "ieee754": "^1.1.13" + } + }, + "node_modules/buffer-equal-constant-time": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", + "integrity": "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==", + "license": "BSD-3-Clause" + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", + "license": "MIT" + }, + "node_modules/busboy": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/busboy/-/busboy-1.6.0.tgz", + "integrity": "sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==", + "dependencies": { + "streamsearch": "^1.1.0" + }, + "engines": { + "node": ">=10.16.0" + } + }, + "node_modules/bytes": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", + "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/call-bind": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.8.tgz", + "integrity": "sha512-oKlSFMcMwpUg2ednkhQ454wfWiU/ul3CkJe/PEHcTKuiX6RpbehUiFMXu13HalGZxfUwCQzZG747YXBn1im9ww==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.0", + "es-define-property": "^1.0.0", + "get-intrinsic": "^1.2.4", + "set-function-length": "^1.2.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/call-bind-apply-helpers": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", + "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/call-bound": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", + "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "get-intrinsic": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/camelcase": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001764", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001764.tgz", + "integrity": "sha512-9JGuzl2M+vPL+pz70gtMF9sHdMFbY9FJaQBi186cHKH3pSzDvzoUJUPV6fqiKIMyXbud9ZLg4F3Yza1vJ1+93g==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "CC-BY-4.0" + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/char-regex": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz", + "integrity": "sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + } + }, + "node_modules/chardet": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz", + "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", + "dev": true, + "license": "MIT" + }, + "node_modules/chokidar": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", + "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", + "dev": true, + "license": "MIT", + "dependencies": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/chrome-trace-event": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.4.tgz", + "integrity": "sha512-rNjApaLzuwaOTjCiT8lSDdGN1APCiqkChLMJxJPWLunPAt5fy8xgU9/jNOchV84wfIxrA0lRQB7oCT8jrn/wrQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.0" + } + }, + "node_modules/ci-info": { + "version": "3.9.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.9.0.tgz", + "integrity": "sha512-NIxF55hv4nSqQswkAeiOi1r83xy8JldOFDTWiug55KBu9Jnblncd2U6ViHmYgHf01TPZS77NJBhBMKdWj9HQMQ==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/sibiraj-s" + } + ], + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/cjs-module-lexer": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.4.3.tgz", + "integrity": "sha512-9z8TZaGM1pfswYeXrUpzPrkx8UnWYdhJclsiYMm6x/w5+nN+8Tf/LnAgfLGQCm59qAOxU8WwHEq2vNwF6i4j+Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/class-transformer": { + "version": "0.5.1", + "resolved": "https://registry.npmjs.org/class-transformer/-/class-transformer-0.5.1.tgz", + "integrity": "sha512-SQa1Ws6hUbfC98vKGxZH3KFY0Y1lm5Zm0SY8XX9zbK7FJCyVEac3ATW0RIpwzW+oOfmHE5PMPufDG9hCfoEOMw==", + "license": "MIT", + "peer": true + }, + "node_modules/class-validator": { + "version": "0.14.3", + "resolved": "https://registry.npmjs.org/class-validator/-/class-validator-0.14.3.tgz", + "integrity": "sha512-rXXekcjofVN1LTOSw+u4u9WXVEUvNBVjORW154q/IdmYWy1nMbOU9aNtZB0t8m+FJQ9q91jlr2f9CwwUFdFMRA==", + "license": "MIT", + "peer": true, + "dependencies": { + "@types/validator": "^13.15.3", + "libphonenumber-js": "^1.11.1", + "validator": "^13.15.20" + } + }, + "node_modules/cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "dev": true, + "license": "MIT", + "dependencies": { + "restore-cursor": "^3.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cli-spinners": { + "version": "2.9.2", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.9.2.tgz", + "integrity": "sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/cli-table3": { + "version": "0.6.5", + "resolved": "https://registry.npmjs.org/cli-table3/-/cli-table3-0.6.5.tgz", + "integrity": "sha512-+W/5efTR7y5HRD7gACw9yQjqMVvEMLBHmboM/kPWam+H+Hmyrgjh6YncVKK122YZkXrLudzTuAukUw9FnMf7IQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "string-width": "^4.2.0" + }, + "engines": { + "node": "10.* || >= 12.*" + }, + "optionalDependencies": { + "@colors/colors": "1.5.0" + } + }, + "node_modules/cli-width": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-3.0.0.tgz", + "integrity": "sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">= 10" + } + }, + "node_modules/cliui": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", + "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", + "dev": true, + "license": "ISC", + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.1", + "wrap-ansi": "^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/cliui/node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/clone": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz", + "integrity": "sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8" + } + }, + "node_modules/cluster-key-slot": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/cluster-key-slot/-/cluster-key-slot-1.1.2.tgz", + "integrity": "sha512-RMr0FhtfXemyinomL4hrWcYJxmX6deFdCxpJzhDttxgO1+bcCnkk+9drydLVDmAMG7NE6aN/fl4F7ucU/90gAA==", + "license": "Apache-2.0", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/co": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", + "integrity": "sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==", + "dev": true, + "license": "MIT", + "engines": { + "iojs": ">= 1.0.0", + "node": ">= 0.12.0" + } + }, + "node_modules/collect-v8-coverage": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.3.tgz", + "integrity": "sha512-1L5aqIkwPfiodaMgQunkF1zRhNqifHBmtbbbxcr6yVxxBnliw4TDOW6NxpO8DJLgJ16OT+Y4ztZqP6p/FtXnAw==", + "dev": true, + "license": "MIT" + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "license": "MIT", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "license": "MIT" + }, + "node_modules/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/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true, + "license": "MIT" + }, + "node_modules/concat-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-2.0.0.tgz", + "integrity": "sha512-MWufYdFw53ccGjCA+Ol7XJYpAlW6/prSMzuPOTRnJGcGzuhLn4Scrz7qf6o8bROZ514ltazcIFJZevcfbo0x7A==", + "engines": [ + "node >= 6.0" + ], + "license": "MIT", + "dependencies": { + "buffer-from": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^3.0.2", + "typedarray": "^0.0.6" + } + }, + "node_modules/consola": { + "version": "2.15.3", + "resolved": "https://registry.npmjs.org/consola/-/consola-2.15.3.tgz", + "integrity": "sha512-9vAdYbHj6x2fLKC4+oPH0kFzY/orMZyG2Aj+kNylHxKGJ/Ed4dpNyAQYwJOdqO4zdM7XpVHmyejQDcQHrnuXbw==", + "license": "MIT" + }, + "node_modules/content-disposition": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-1.0.1.tgz", + "integrity": "sha512-oIXISMynqSqm241k6kcQ5UwttDILMK4BiurCfGEREw6+X9jkkpEe5T9FZaApyLGGOnFuyMWZpdolTXMtvEJ08Q==", + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "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.2", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.2.tgz", + "integrity": "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie-signature": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.2.2.tgz", + "integrity": "sha512-D76uU73ulSXrD1UXF4KE2TMxVVwhsnCgfAyTg9k8P6KGZjlXKrOLe4dJQKI3Bxi5wjesZoFXJWElNWBjPZMbhg==", + "license": "MIT", + "engines": { + "node": ">=6.6.0" + } + }, + "node_modules/core-util-is": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz", + "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/cors": { + "version": "2.8.5", + "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", + "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", + "license": "MIT", + "dependencies": { + "object-assign": "^4", + "vary": "^1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/cosmiconfig": { + "version": "8.3.6", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-8.3.6.tgz", + "integrity": "sha512-kcZ6+W5QzcJ3P1Mt+83OUv/oHFqZHIx8DuxG6eZ5RGMERoLqp4BuGjhHLYGK+Kf5XVkQvqBSmAy/nGWN3qDgEA==", + "dev": true, + "license": "MIT", + "dependencies": { + "import-fresh": "^3.3.0", + "js-yaml": "^4.1.0", + "parse-json": "^5.2.0", + "path-type": "^4.0.0" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/d-fischer" + }, + "peerDependencies": { + "typescript": ">=4.9.5" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/create-jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/create-jest/-/create-jest-29.7.0.tgz", + "integrity": "sha512-Adz2bdH0Vq3F53KEMJOoftQFutWCukm6J24wbPWRO4k1kMY7gS7ds/uoJkNuV8wDCtWWnuwGcJwpWcih+zEW1Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "jest-config": "^29.7.0", + "jest-util": "^29.7.0", + "prompts": "^2.0.1" + }, + "bin": { + "create-jest": "bin/create-jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/create-require": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz", + "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/cron": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/cron/-/cron-3.2.1.tgz", + "integrity": "sha512-w2n5l49GMmmkBFEsH9FIDhjZ1n1QgTMOCMGuQtOXs5veNiosZmso6bQGuqOJSYAXXrG84WQFVneNk+Yt0Ua9iw==", + "license": "MIT", + "dependencies": { + "@types/luxon": "~3.4.0", + "luxon": "~3.5.0" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/decimal.js": { + "version": "10.6.0", + "resolved": "https://registry.npmjs.org/decimal.js/-/decimal.js-10.6.0.tgz", + "integrity": "sha512-YpgQiITW3JXGntzdUmyUR1V812Hn8T1YVXhCu+wO3OpS4eU9l4YdD3qjyiKdV6mvV29zapkMeD390UVEf2lkUg==", + "license": "MIT" + }, + "node_modules/dedent": { + "version": "1.7.1", + "resolved": "https://registry.npmjs.org/dedent/-/dedent-1.7.1.tgz", + "integrity": "sha512-9JmrhGZpOlEgOLdQgSm0zxFaYoQon408V1v49aqTWuXENVlnCuY9JBZcXZiCsZQWDjTm5Qf/nIvAy77mXDAjEg==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "babel-plugin-macros": "^3.1.0" + }, + "peerDependenciesMeta": { + "babel-plugin-macros": { + "optional": true + } + } + }, + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/deepmerge": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.3.1.tgz", + "integrity": "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/defaults": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.4.tgz", + "integrity": "sha512-eFuaLoy/Rxalv2kr+lqMlUnrDWV+3j4pljOIJgLIhI058IQfWJ7vXhyEIHu+HtC738klGALYxOKDO0bQP3tg8A==", + "dev": true, + "license": "MIT", + "dependencies": { + "clone": "^1.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/define-data-property": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", + "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "gopd": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/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/diff": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", + "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.3.1" + } + }, + "node_modules/diff-sequences": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-29.6.3.tgz", + "integrity": "sha512-EjePK1srD3P08o2j4f0ExnylqRs5B9tJjcp9t1krH2qRi8CCdsYfwe9JgSLurFBWwq4uOlipzfk5fHNvwFKr8Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/dir-glob": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", + "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-type": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/dotenv": { + "version": "16.4.5", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.5.tgz", + "integrity": "sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==", + "license": "BSD-2-Clause", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://dotenvx.com" + } + }, + "node_modules/dotenv-expand": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/dotenv-expand/-/dotenv-expand-10.0.0.tgz", + "integrity": "sha512-GopVGCpVS1UKH75VKHGuQFqS1Gusej0z4FyQkPdwjil2gNIv+LNsqBlboOzpJFZKVT95GkCyWJbBSdFEFUWI2A==", + "license": "BSD-2-Clause", + "engines": { + "node": ">=12" + } + }, + "node_modules/dunder-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", + "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.1", + "es-errors": "^1.3.0", + "gopd": "^1.2.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/eastasianwidth": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", + "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==", + "dev": true, + "license": "MIT" + }, + "node_modules/ecdsa-sig-formatter": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", + "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", + "license": "Apache-2.0", + "dependencies": { + "safe-buffer": "^5.0.1" + } + }, + "node_modules/ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==", + "license": "MIT" + }, + "node_modules/electron-to-chromium": { + "version": "1.5.267", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.267.tgz", + "integrity": "sha512-0Drusm6MVRXSOJpGbaSVgcQsuB4hEkMpHXaVstcPmhu5LIedxs1xNK/nIxmQIU/RPC0+1/o0AVZfBTkTNJOdUw==", + "dev": true, + "license": "ISC" + }, + "node_modules/emittery": { + "version": "0.13.1", + "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.13.1.tgz", + "integrity": "sha512-DeWwawk6r5yR9jFgnDKYt4sLS0LmHJJi3ZOnb5/JdbYwj3nW+FxQnHIjhBKz8YLC7oRNPVM9NQ47I3CVx34eqQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sindresorhus/emittery?sponsor=1" + } + }, + "node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true, + "license": "MIT" + }, + "node_modules/encodeurl": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", + "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/enhanced-resolve": { + "version": "5.18.4", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.18.4.tgz", + "integrity": "sha512-LgQMM4WXU3QI+SYgEc2liRgznaD5ojbmY3sb8LxyguVkIg5FxdpTkvk72te2R38/TGKxH634oLxXRGY6d7AP+Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "graceful-fs": "^4.2.4", + "tapable": "^2.2.0" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/error-ex": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.4.tgz", + "integrity": "sha512-sqQamAnR14VgCr1A618A3sGrygcpK+HEbenA/HiEAkkUwcZIIB/tgWqHFxWgOyDh4nB4JCRimh79dR5Ywc9MDQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-arrayish": "^0.2.1" + } + }, + "node_modules/es-define-property": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", + "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-errors": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", + "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-module-lexer": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-2.0.0.tgz", + "integrity": "sha512-5POEcUuZybH7IdmGsD8wlf0AI55wMecM9rVBTI/qEAy2c1kTOm3DjFYjrBdI2K3BaJjJYfYFeRtM0t9ssnRuxw==", + "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/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.7.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.7.0.tgz", + "integrity": "sha512-Ap6G0WQwcU/LHsvLwON1fAQX9Zp0A2Y6Y/cJBl9r/JbW90Zyg4/zbG6zzKa2OTALELarYHmKu0GhpM5EO+7T0g==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/ethers": { + "version": "6.16.0", + "resolved": "https://registry.npmjs.org/ethers/-/ethers-6.16.0.tgz", + "integrity": "sha512-U1wulmetNymijEhpSEQ7Ct/P/Jw9/e7R1j5XIbPRydgV2DjLVMsULDlNksq3RQnFgKoLlZf88ijYtWEXcPa07A==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/ethers-io/" + }, + { + "type": "individual", + "url": "https://www.buymeacoffee.com/ricmoo" + } + ], + "license": "MIT", + "dependencies": { + "@adraffy/ens-normalize": "1.10.1", + "@noble/curves": "1.2.0", + "@noble/hashes": "1.3.2", + "@types/node": "22.7.5", + "aes-js": "4.0.0-beta.5", + "tslib": "2.7.0", + "ws": "8.17.1" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/ethers/node_modules/@types/node": { + "version": "22.7.5", + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.7.5.tgz", + "integrity": "sha512-jML7s2NAzMWc//QSJ1a3prpk78cOPchGvXJsC3C6R6PSMoooztvRVQEz89gmBTBY1SPMaqo5teB4uNHPdetShQ==", + "license": "MIT", + "dependencies": { + "undici-types": "~6.19.2" + } + }, + "node_modules/ethers/node_modules/tslib": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.7.0.tgz", + "integrity": "sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==", + "license": "0BSD" + }, + "node_modules/ethers/node_modules/undici-types": { + "version": "6.19.8", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.19.8.tgz", + "integrity": "sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==", + "license": "MIT" + }, + "node_modules/events": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/events/-/events-3.3.0.tgz", + "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8.x" + } + }, + "node_modules/execa": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz", + "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", + "dev": true, + "license": "MIT", + "dependencies": { + "cross-spawn": "^7.0.3", + "get-stream": "^6.0.0", + "human-signals": "^2.1.0", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.1", + "onetime": "^5.1.2", + "signal-exit": "^3.0.3", + "strip-final-newline": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sindresorhus/execa?sponsor=1" + } + }, + "node_modules/execa/node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/exit": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", + "integrity": "sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/expect": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/expect/-/expect-29.7.0.tgz", + "integrity": "sha512-2Zks0hf1VLFYI1kbh0I5jP3KHHyCHpkfyHBzsSXRFgl/Bg9mWYfMW8oD+PdMPlEwy5HNsR9JutYy6pMeOh61nw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/expect-utils": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/express": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/express/-/express-5.2.1.tgz", + "integrity": "sha512-hIS4idWWai69NezIdRt2xFVofaF4j+6INOpJlVOLDO8zXGpUVEVzIYk12UUi2JzjEzWL3IOAxcTubgz9Po0yXw==", + "license": "MIT", + "dependencies": { + "accepts": "^2.0.0", + "body-parser": "^2.2.1", + "content-disposition": "^1.0.0", + "content-type": "^1.0.5", + "cookie": "^0.7.1", + "cookie-signature": "^1.2.1", + "debug": "^4.4.0", + "depd": "^2.0.0", + "encodeurl": "^2.0.0", + "escape-html": "^1.0.3", + "etag": "^1.8.1", + "finalhandler": "^2.1.0", + "fresh": "^2.0.0", + "http-errors": "^2.0.0", + "merge-descriptors": "^2.0.0", + "mime-types": "^3.0.0", + "on-finished": "^2.4.1", + "once": "^1.4.0", + "parseurl": "^1.3.3", + "proxy-addr": "^2.0.7", + "qs": "^6.14.0", + "range-parser": "^1.2.1", + "router": "^2.2.0", + "send": "^1.1.0", + "serve-static": "^2.2.0", + "statuses": "^2.0.1", + "type-is": "^2.0.1", + "vary": "^1.1.2" + }, + "engines": { + "node": ">= 18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/express/node_modules/body-parser": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-2.2.2.tgz", + "integrity": "sha512-oP5VkATKlNwcgvxi0vM0p/D3n2C3EReYVX+DNYs5TjZFn/oQt2j+4sVJtSMr18pdRr8wjTcBl6LoV+FUwzPmNA==", + "license": "MIT", + "dependencies": { + "bytes": "^3.1.2", + "content-type": "^1.0.5", + "debug": "^4.4.3", + "http-errors": "^2.0.0", + "iconv-lite": "^0.7.0", + "on-finished": "^2.4.1", + "qs": "^6.14.1", + "raw-body": "^3.0.1", + "type-is": "^2.0.1" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/express/node_modules/iconv-lite": { + "version": "0.7.2", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.7.2.tgz", + "integrity": "sha512-im9DjEDQ55s9fL4EYzOAv0yMqmMBSZp6G0VvFyTMPKWxiSBHUj9NW/qqLmXUwXrrM7AvqSlTCfvqRb0cM8yYqw==", + "license": "MIT", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/express/node_modules/media-typer": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-1.1.0.tgz", + "integrity": "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/express/node_modules/raw-body": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-3.0.2.tgz", + "integrity": "sha512-K5zQjDllxWkf7Z5xJdV0/B0WTNqx6vxG70zJE4N0kBs4LovmEYWJzQGxC9bS9RAKu3bgM40lrd5zoLJ12MQ5BA==", + "license": "MIT", + "dependencies": { + "bytes": "~3.1.2", + "http-errors": "~2.0.1", + "iconv-lite": "~0.7.0", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/express/node_modules/type-is": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-2.0.1.tgz", + "integrity": "sha512-OZs6gsjF4vMp32qrCbiVSkrFmXtG/AZhY3t0iAMrMBiAZyV9oALtXO8hsrHbMXF9x6L3grlFuwW2oAz7cav+Gw==", + "license": "MIT", + "dependencies": { + "content-type": "^1.0.5", + "media-typer": "^1.1.0", + "mime-types": "^3.0.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/external-editor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.1.0.tgz", + "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", + "dev": true, + "license": "MIT", + "dependencies": { + "chardet": "^0.7.0", + "iconv-lite": "^0.4.24", + "tmp": "^0.0.33" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/fast-diff": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/fast-diff/-/fast-diff-1.3.0.tgz", + "integrity": "sha512-VxPP4NqbUjj6MaAOafWeUn2cXWLcCtljklUtZf0Ind4XQ+QPtmA0b18zZy0jIQx+ExRVCR/ZQpBmik5lXshNsw==", + "dev": true, + "license": "Apache-2.0" + }, + "node_modules/fast-glob": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz", + "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.2", + "merge2": "^1.3.0", + "micromatch": "^4.0.8" + }, + "engines": { + "node": ">=8.6.0" + } + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fast-safe-stringify": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/fast-safe-stringify/-/fast-safe-stringify-2.1.1.tgz", + "integrity": "sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA==", + "license": "MIT" + }, + "node_modules/fastq": { + "version": "1.20.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.20.1.tgz", + "integrity": "sha512-GGToxJ/w1x32s/D2EKND7kTil4n8OVk/9mycTc4VDza13lOvpUZTGX3mFSCtV9ksdGBVzvsyAVLM6mHFThxXxw==", + "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": "2.1.1", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-2.1.1.tgz", + "integrity": "sha512-S8KoZgRZN+a5rNwqTxlZZePjT/4cnm0ROV70LedRHZ0p8u9fRID0hJUZQpkKLzro8LfmC8sx23bY6tVNxv8pQA==", + "license": "MIT", + "dependencies": { + "debug": "^4.4.0", + "encodeurl": "^2.0.0", + "escape-html": "^1.0.3", + "on-finished": "^2.4.1", + "parseurl": "^1.3.3", + "statuses": "^2.0.1" + }, + "engines": { + "node": ">= 18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "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/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/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": "2.0.0", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-2.0.0.tgz", + "integrity": "sha512-Rx/WycZ60HOaqLKAi6cHRKKI7zxWbJ31MhntmtwMoaTeF7XFH9hhBp8vITaMidfljRQ6eYWCKkaTK+ykVJHP2A==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/fs-extra": { + "version": "10.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-10.1.0.tgz", + "integrity": "sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/fs-monkey": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/fs-monkey/-/fs-monkey-1.1.0.tgz", + "integrity": "sha512-QMUezzXWII9EV5aTFXW1UBVUO77wYPpjqIF8/AviUCThNeSYZykpoTixUeaNNBwmCev0AMDWMAni+f8Hxb1IFw==", + "dev": true, + "license": "Unlicense" + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true, + "license": "ISC" + }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "dev": true, + "license": "ISC", + "engines": { + "node": "6.* || 8.* || >= 10.*" + } + }, + "node_modules/get-intrinsic": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", + "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "es-define-property": "^1.0.1", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.1.1", + "function-bind": "^1.1.2", + "get-proto": "^1.0.1", + "gopd": "^1.2.0", + "has-symbols": "^1.1.0", + "hasown": "^2.0.2", + "math-intrinsics": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-package-type": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", + "integrity": "sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/get-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", + "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", + "license": "MIT", + "dependencies": { + "dunder-proto": "^1.0.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/get-stream": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz", + "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/glob": { + "version": "10.4.5", + "resolved": "https://registry.npmjs.org/glob/-/glob-10.4.5.tgz", + "integrity": "sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==", + "dev": true, + "license": "ISC", + "dependencies": { + "foreground-child": "^3.1.0", + "jackspeak": "^3.1.2", + "minimatch": "^9.0.4", + "minipass": "^7.1.2", + "package-json-from-dist": "^1.0.0", + "path-scurry": "^1.11.1" + }, + "bin": { + "glob": "dist/esm/bin.mjs" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/glob-to-regexp": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.4.1.tgz", + "integrity": "sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw==", + "dev": true, + "license": "BSD-2-Clause" + }, + "node_modules/glob/node_modules/minimatch": { + "version": "9.0.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", + "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/globals": { + "version": "13.24.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", + "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "type-fest": "^0.20.2" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/globby": { + "version": "11.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz", + "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", + "dev": true, + "license": "MIT", + "dependencies": { + "array-union": "^2.1.0", + "dir-glob": "^3.0.1", + "fast-glob": "^3.2.9", + "ignore": "^5.2.0", + "merge2": "^1.4.1", + "slash": "^3.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/gopd": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", + "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true, + "license": "MIT" + }, + "node_modules/handlebars": { + "version": "4.7.8", + "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.8.tgz", + "integrity": "sha512-vafaFqs8MZkRrSX7sFVUdo3ap/eNiLnb4IakshzvP56X5Nr1iGKAIqdX6tMlm6HcNRIkr6AxO5jFEoJzzpT8aQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "minimist": "^1.2.5", + "neo-async": "^2.6.2", + "source-map": "^0.6.1", + "wordwrap": "^1.0.0" + }, + "bin": { + "handlebars": "bin/handlebars" + }, + "engines": { + "node": ">=0.4.7" + }, + "optionalDependencies": { + "uglify-js": "^3.1.4" + } + }, + "node_modules/handlebars/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/has-own-prop": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-own-prop/-/has-own-prop-2.0.0.tgz", + "integrity": "sha512-Pq0h+hvsVm6dDEa8x82GnLSYHOzNDt7f0ddFa3FqcQlgzEiptPqL+XrOJNavjOzSYiYWIrgeVYYgGlLmnxwilQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/has-property-descriptors": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", + "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-define-property": "^1.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-symbols": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", + "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/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.1", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.1.tgz", + "integrity": "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ==", + "license": "MIT", + "dependencies": { + "depd": "~2.0.0", + "inherits": "~2.0.4", + "setprototypeof": "~1.2.0", + "statuses": "~2.0.2", + "toidentifier": "~1.0.1" + }, + "engines": { + "node": ">= 0.8" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/human-signals": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz", + "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=10.17.0" + } + }, + "node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "license": "MIT", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ieee754": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", + "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "BSD-3-Clause" + }, + "node_modules/ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, + "node_modules/import-fresh": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", + "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/import-local": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.2.0.tgz", + "integrity": "sha512-2SPlun1JUPWoM6t3F0dw0FkCF/jWY8kttcY4f599GLTSjh2OCuuhdTkJQsEcZzBqbXZGKMK2OqW1oZsjtf/gQA==", + "dev": true, + "license": "MIT", + "dependencies": { + "pkg-dir": "^4.2.0", + "resolve-cwd": "^3.0.0" + }, + "bin": { + "import-local-fixture": "fixtures/cli.js" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "deprecated": "This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.", + "dev": true, + "license": "ISC", + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "license": "ISC" + }, + "node_modules/inquirer": { + "version": "8.2.6", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-8.2.6.tgz", + "integrity": "sha512-M1WuAmb7pn9zdFRtQYk26ZBoY043Sse0wVDdk4Bppr+JOXyQYybdtvK+l9wUibhtjdjvtoiNy8tk+EgsYIUqKg==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-escapes": "^4.2.1", + "chalk": "^4.1.1", + "cli-cursor": "^3.1.0", + "cli-width": "^3.0.0", + "external-editor": "^3.0.3", + "figures": "^3.0.0", + "lodash": "^4.17.21", + "mute-stream": "0.0.8", + "ora": "^5.4.1", + "run-async": "^2.4.0", + "rxjs": "^7.5.5", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0", + "through": "^2.3.6", + "wrap-ansi": "^6.0.1" + }, + "engines": { + "node": ">=12.0.0" + } + }, + "node_modules/ioredis": { + "version": "5.9.1", + "resolved": "https://registry.npmjs.org/ioredis/-/ioredis-5.9.1.tgz", + "integrity": "sha512-BXNqFQ66oOsR82g9ajFFsR8ZKrjVvYCLyeML9IvSMAsP56XH2VXBdZjmI11p65nXXJxTEt1hie3J2QeFJVgrtQ==", + "license": "MIT", + "peer": true, + "dependencies": { + "@ioredis/commands": "1.5.0", + "cluster-key-slot": "^1.1.0", + "debug": "^4.3.4", + "denque": "^2.1.0", + "lodash.defaults": "^4.2.0", + "lodash.isarguments": "^3.1.0", + "redis-errors": "^1.2.0", + "redis-parser": "^3.0.0", + "standard-as-callback": "^2.1.0" + }, + "engines": { + "node": ">=12.22.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/ioredis" + } + }, + "node_modules/ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", + "license": "MIT", + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", + "dev": true, + "license": "MIT" + }, + "node_modules/is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dev": true, + "license": "MIT", + "dependencies": { + "binary-extensions": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-core-module": { + "version": "2.16.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", + "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", + "dev": true, + "license": "MIT", + "dependencies": { + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-generator-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-generator-fn/-/is-generator-fn-2.1.0.tgz", + "integrity": "sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-interactive": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-interactive/-/is-interactive-1.0.0.tgz", + "integrity": "sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-path-inside": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", + "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-promise": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-4.0.0.tgz", + "integrity": "sha512-hvpoI6korhJMnej285dSg6nu1+e6uxs7zG3BYAm5byqDsgJNWwxzM6z6iZiAgQR4TJ30JmBTOwqZUw3WlyH3AQ==", + "license": "MIT" + }, + "node_modules/is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/is-unicode-supported": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz", + "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true, + "license": "ISC" + }, + "node_modules/istanbul-lib-coverage": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.2.tgz", + "integrity": "sha512-O8dpsF+r0WV/8MNRKfnmrtCWhuKjxrq2w+jpzBL5UZKTi2LeVWnWOmWRxFlesJONmc+wLAGvKQZEOanko0LFTg==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-instrument": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-6.0.3.tgz", + "integrity": "sha512-Vtgk7L/R2JHyyGW07spoFlB8/lpjiOLTjMdms6AFMraYt3BaJauod/NGrfnVG/y4Ix1JEuMRPDPEj2ua+zz1/Q==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/core": "^7.23.9", + "@babel/parser": "^7.23.9", + "@istanbuljs/schema": "^0.1.3", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^7.5.4" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-report": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.1.tgz", + "integrity": "sha512-GCfE1mtsHGOELCU8e/Z7YWzpmybrx/+dSTfLrvY8qRmaY6zXTKWn6WQIjaAFw069icm6GVMNkgu0NzI4iPZUNw==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "istanbul-lib-coverage": "^3.0.0", + "make-dir": "^4.0.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-source-maps": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz", + "integrity": "sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "debug": "^4.1.1", + "istanbul-lib-coverage": "^3.0.0", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-source-maps/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/istanbul-reports": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.2.0.tgz", + "integrity": "sha512-HGYWWS/ehqTV3xN10i23tkPkpH46MLCIMFNCaaKNavAXTF1RkqxawEPtnjnGZ6XKSInBKkiOA5BKS+aZiY3AvA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "html-escaper": "^2.0.0", + "istanbul-lib-report": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/iterare": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/iterare/-/iterare-1.2.1.tgz", + "integrity": "sha512-RKYVTCjAnRthyJes037NX/IiqeidgN1xc3j1RjFfECFp28A1GVwK9nA+i0rJPaHqSZwygLzRnFlzUuHFoWWy+Q==", + "license": "ISC", + "engines": { + "node": ">=6" + } + }, + "node_modules/jackspeak": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-3.4.3.tgz", + "integrity": "sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "@isaacs/cliui": "^8.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + }, + "optionalDependencies": { + "@pkgjs/parseargs": "^0.11.0" + } + }, + "node_modules/jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest/-/jest-29.7.0.tgz", + "integrity": "sha512-NIy3oAFp9shda19hy4HK0HRTWKtPJmGdnvywu01nOqNC2vZg+Z+fvJDxpMQA88eb2I9EcafcdjYgsDthnYTvGw==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@jest/core": "^29.7.0", + "@jest/types": "^29.6.3", + "import-local": "^3.0.2", + "jest-cli": "^29.7.0" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-changed-files": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-29.7.0.tgz", + "integrity": "sha512-fEArFiwf1BpQ+4bXSprcDc3/x4HSzL4al2tozwVpDFpsxALjLYdyiIK4e5Vz66GQJIbXJ82+35PtysofptNX2w==", + "dev": true, + "license": "MIT", + "dependencies": { + "execa": "^5.0.0", + "jest-util": "^29.7.0", + "p-limit": "^3.1.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-circus": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-circus/-/jest-circus-29.7.0.tgz", + "integrity": "sha512-3E1nCMgipcTkCocFwM90XXQab9bS+GMsjdpmPrlelaxwD93Ad8iVEjX/vvHPdLPnFf+L40u+5+iutRdA1N9myw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/expect": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "co": "^4.6.0", + "dedent": "^1.0.0", + "is-generator-fn": "^2.0.0", + "jest-each": "^29.7.0", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "p-limit": "^3.1.0", + "pretty-format": "^29.7.0", + "pure-rand": "^6.0.0", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-cli": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-29.7.0.tgz", + "integrity": "sha512-OVVobw2IubN/GSYsxETi+gOe7Ka59EFMR/twOU3Jb2GnKKeMGJB5SGUUrEz3SFVmJASUdZUzy83sLNNQ2gZslg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/core": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "create-jest": "^29.7.0", + "exit": "^0.1.2", + "import-local": "^3.0.2", + "jest-config": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "yargs": "^17.3.1" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-config": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-29.7.0.tgz", + "integrity": "sha512-uXbpfeQ7R6TZBqI3/TxCU4q4ttk3u0PJeC+E0zbfSoSjq6bJ7buBPxzQPL0ifrkY4DNu4JUdk0ImlBUYi840eQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.11.6", + "@jest/test-sequencer": "^29.7.0", + "@jest/types": "^29.6.3", + "babel-jest": "^29.7.0", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "deepmerge": "^4.2.2", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "jest-circus": "^29.7.0", + "jest-environment-node": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-runner": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "micromatch": "^4.0.4", + "parse-json": "^5.2.0", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@types/node": "*", + "ts-node": ">=9.0.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "ts-node": { + "optional": true + } + } + }, + "node_modules/jest-config/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/jest-config/node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/jest-config/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/jest-diff": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-29.7.0.tgz", + "integrity": "sha512-LMIgiIrhigmPrs03JHpxUh2yISK3vLFPkAodPeo0+BuF7wA2FoQbkEg1u8gBYBThncu7e1oEDUfIXVuTqLRUjw==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.0.0", + "diff-sequences": "^29.6.3", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-docblock": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-29.7.0.tgz", + "integrity": "sha512-q617Auw3A612guyaFgsbFeYpNP5t2aoUNLwBUbc/0kD1R4t9ixDbyFTHd1nok4epoVFpr7PmeWHrhvuV3XaJ4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "detect-newline": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-each": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-29.7.0.tgz", + "integrity": "sha512-gns+Er14+ZrEoC5fhOfYCY1LOHHr0TI+rQUHZS8Ttw2l7gl+80eHc/gFf2Ktkw0+SIACDTeWvpFcv3B04VembQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "jest-get-type": "^29.6.3", + "jest-util": "^29.7.0", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-environment-node": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-29.7.0.tgz", + "integrity": "sha512-DOSwCRqXirTOyheM+4d5YZOrWcdu0LNZ87ewUoywbcb2XR4wKgqiG8vNeYwhjFMbEkfju7wx2GYH0P2gevGvFw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/fake-timers": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-mock": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-haste-map": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-29.7.0.tgz", + "integrity": "sha512-fP8u2pyfqx0K1rGn1R9pyE0/KTn+G7PxktWidOBTqFPLYX0b9ksaMFkhK5vrS3DVun09pckLdlx90QthlW7AmA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/graceful-fs": "^4.1.3", + "@types/node": "*", + "anymatch": "^3.0.3", + "fb-watchman": "^2.0.0", + "graceful-fs": "^4.2.9", + "jest-regex-util": "^29.6.3", + "jest-util": "^29.7.0", + "jest-worker": "^29.7.0", + "micromatch": "^4.0.4", + "walker": "^1.0.8" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "optionalDependencies": { + "fsevents": "^2.3.2" + } + }, + "node_modules/jest-leak-detector": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-29.7.0.tgz", + "integrity": "sha512-kYA8IJcSYtST2BY9I+SMC32nDpBT3J2NvWJx8+JCuCdl/CR1I4EKUJROiP8XtCcxqgTTBGJNdbB1A8XRKbTetw==", + "dev": true, + "license": "MIT", + "dependencies": { + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-matcher-utils": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-29.7.0.tgz", + "integrity": "sha512-sBkD+Xi9DtcChsI3L3u0+N0opgPYnCRPtGcQYrgXmR+hmt/fYfWAL0xRXYU8eWOdfuLgBe0YCW3AFtnRLagq/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.0.0", + "jest-diff": "^29.7.0", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-message-util": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-29.7.0.tgz", + "integrity": "sha512-GBEV4GRADeP+qtB2+6u61stea8mGcOT4mCtrYISZwfu9/ISHFJ/5zOMXYbpBE9RsS5+Gb63DW4FgmnKJ79Kf6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.12.13", + "@jest/types": "^29.6.3", + "@types/stack-utils": "^2.0.0", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "micromatch": "^4.0.4", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-mock": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-29.7.0.tgz", + "integrity": "sha512-ITOMZn+UkYS4ZFh83xYAOzWStloNzJFO2s8DWrE4lhtGD+AorgnbkiKERe4wQVBydIGPx059g6riW5Btp6Llnw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-pnp-resolver": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.3.tgz", + "integrity": "sha512-+3NpwQEnRoIBtx4fyhblQDPgJI0H1IEIkX7ShLUjPGA7TtUTvI1oiKi3SR4oBR0hQhQR80l4WAe5RrXBwWMA8w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + }, + "peerDependencies": { + "jest-resolve": "*" + }, + "peerDependenciesMeta": { + "jest-resolve": { + "optional": true + } + } + }, + "node_modules/jest-regex-util": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-29.6.3.tgz", + "integrity": "sha512-KJJBsRCyyLNWCNBOvZyRDnAIfUiRJ8v+hOBQYGn8gDyF3UegwiP4gwRR3/SDa42g1YbVycTidUF3rKjyLFDWbg==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-resolve": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-29.7.0.tgz", + "integrity": "sha512-IOVhZSrg+UvVAshDSDtHyFCCBUl/Q3AAJv8iZ6ZjnZ74xzvwuzLXid9IIIPgTnY62SJjfuupMKZsZQRsCvxEgA==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-pnp-resolver": "^1.2.2", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "resolve": "^1.20.0", + "resolve.exports": "^2.0.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-resolve-dependencies": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-29.7.0.tgz", + "integrity": "sha512-un0zD/6qxJ+S0et7WxeI3H5XSe9lTBBR7bOHCHXkKR6luG5mwDDlIzVQ0V5cZCuoTgEdcdwzTghYkTWfubi+nA==", + "dev": true, + "license": "MIT", + "dependencies": { + "jest-regex-util": "^29.6.3", + "jest-snapshot": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-runner": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-29.7.0.tgz", + "integrity": "sha512-fsc4N6cPCAahybGBfTRcq5wFR6fpLznMg47sY5aDpsoejOcVYFb07AHuSnR0liMcPTgBsA3ZJL6kFOjPdoNipQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/environment": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "emittery": "^0.13.1", + "graceful-fs": "^4.2.9", + "jest-docblock": "^29.7.0", + "jest-environment-node": "^29.7.0", + "jest-haste-map": "^29.7.0", + "jest-leak-detector": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-resolve": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-util": "^29.7.0", + "jest-watcher": "^29.7.0", + "jest-worker": "^29.7.0", + "p-limit": "^3.1.0", + "source-map-support": "0.5.13" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-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.3", + "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-9.0.3.tgz", + "integrity": "sha512-MT/xP0CrubFRNLNKvxJ2BYfy53Zkm++5bX9dtuPbqAeQpTVe0MQTFhao8+Cp//EmJp244xt6Drw/GVEGCUj40g==", + "license": "MIT", + "dependencies": { + "jws": "^4.0.1", + "lodash.includes": "^4.3.0", + "lodash.isboolean": "^3.0.3", + "lodash.isinteger": "^4.0.4", + "lodash.isnumber": "^3.0.3", + "lodash.isplainobject": "^4.0.6", + "lodash.isstring": "^4.0.1", + "lodash.once": "^4.0.0", + "ms": "^2.1.1", + "semver": "^7.5.4" + }, + "engines": { + "node": ">=12", + "npm": ">=6" + } + }, + "node_modules/jwa": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/jwa/-/jwa-2.0.1.tgz", + "integrity": "sha512-hRF04fqJIP8Abbkq5NKGN0Bbr3JxlQ+qhZufXVr0DvujKy93ZCbXZMHDL4EOtodSbCWxOqR8MS1tXA5hwqCXDg==", + "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": "4.0.1", + "resolved": "https://registry.npmjs.org/jws/-/jws-4.0.1.tgz", + "integrity": "sha512-EKI/M/yqPncGUUh44xz0PxSidXFr/+r0pA70+gIYhjv+et7yxM+s29Y+VGDkovRofQem0fs7Uvf4+YmAdyRduA==", + "license": "MIT", + "dependencies": { + "jwa": "^2.0.1", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/kafkajs": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/kafkajs/-/kafkajs-2.2.4.tgz", + "integrity": "sha512-j/YeapB1vfPT2iOIUn/vxdyKEuhuY2PxMBvf5JWux6iSaukAccrMtXEY/Lb7OvavDhOWME589bpLrEdnVHjfjA==", + "license": "MIT", + "peer": true, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", + "dev": true, + "license": "MIT", + "dependencies": { + "json-buffer": "3.0.1" + } + }, + "node_modules/kleur": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", + "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/leven": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", + "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/libphonenumber-js": { + "version": "1.12.33", + "resolved": "https://registry.npmjs.org/libphonenumber-js/-/libphonenumber-js-1.12.33.tgz", + "integrity": "sha512-r9kw4OA6oDO4dPXkOrXTkArQAafIKAU71hChInV4FxZ69dxCfbwQGDPzqR5/vea94wU705/3AZroEbSoeVWrQw==", + "license": "MIT" + }, + "node_modules/lines-and-columns": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", + "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", + "dev": true, + "license": "MIT" + }, + "node_modules/loader-runner": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/loader-runner/-/loader-runner-4.3.1.tgz", + "integrity": "sha512-IWqP2SCPhyVFTBtRcgMHdzlf9ul25NwaFx4wCEH/KjAXuuHY4yNjvPXsBokp8jCB936PyWRaPKUNh8NvylLp2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.11.5" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", + "license": "MIT" + }, + "node_modules/lodash.defaults": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz", + "integrity": "sha512-qjxPLHd3r5DnsdGacqOMU6pb/avJzdh9tFX2ymgoZE27BmjXrNy/y4LoaiTeAb+O3gL8AfpJGtqfX/ae2leYYQ==", + "license": "MIT" + }, + "node_modules/lodash.includes": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", + "integrity": "sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w==", + "license": "MIT" + }, + "node_modules/lodash.isarguments": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/lodash.isarguments/-/lodash.isarguments-3.1.0.tgz", + "integrity": "sha512-chi4NHZlZqZD18a0imDHnZPrDeBbTtVN7GXMwuGdRH9qotxAjYs3aVLKc7zNOG9eddR5Ksd8rvFEBc9SsggPpg==", + "license": "MIT" + }, + "node_modules/lodash.isboolean": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", + "integrity": "sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg==", + "license": "MIT" + }, + "node_modules/lodash.isinteger": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", + "integrity": "sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA==", + "license": "MIT" + }, + "node_modules/lodash.isnumber": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", + "integrity": "sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw==", + "license": "MIT" + }, + "node_modules/lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==", + "license": "MIT" + }, + "node_modules/lodash.isstring": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", + "integrity": "sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw==", + "license": "MIT" + }, + "node_modules/lodash.memoize": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", + "integrity": "sha512-t7j+NzmgnQzTAYXcsHYLgimltOV1MXHtlOWf6GjL9Kj8GK5FInw5JotxvbOs+IvV1/Dzo04/fCGfLVs7aXb4Ag==", + "dev": true, + "license": "MIT" + }, + "node_modules/lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/lodash.once": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", + "integrity": "sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg==", + "license": "MIT" + }, + "node_modules/log-symbols": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz", + "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.1.0", + "is-unicode-supported": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "license": "ISC", + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/luxon": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/luxon/-/luxon-3.5.0.tgz", + "integrity": "sha512-rh+Zjr6DNfUYR3bPwJEnuwDdqMbxZW7LOQfUN4B54+Cl+0o5zaU9RJ6bcidfDtC1cWCZXQ+nvX8bf6bAji37QQ==", + "license": "MIT", + "engines": { + "node": ">=12" + } + }, + "node_modules/magic-string": { + "version": "0.30.8", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.8.tgz", + "integrity": "sha512-ISQTe55T2ao7XtlAStud6qwYPZjE4GK1S/BeVPus4jrq6JuOnQ00YKQC581RWhR122W7msZV263KzVeLoqidyQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.4.15" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/make-dir": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-4.0.0.tgz", + "integrity": "sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==", + "dev": true, + "license": "MIT", + "dependencies": { + "semver": "^7.5.3" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dev": true, + "license": "ISC" + }, + "node_modules/makeerror": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/makeerror/-/makeerror-1.0.12.tgz", + "integrity": "sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "tmpl": "1.0.5" + } + }, + "node_modules/math-intrinsics": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", + "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/memfs": { + "version": "3.5.3", + "resolved": "https://registry.npmjs.org/memfs/-/memfs-3.5.3.tgz", + "integrity": "sha512-UERzLsxzllchadvbPs5aolHh65ISpKpM+ccLbOJ8/vvpBKmAWf+la7dXFy7Mr0ySHbdHrFv5kGFCUHHe6GFEmw==", + "dev": true, + "license": "Unlicense", + "dependencies": { + "fs-monkey": "^1.0.4" + }, + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/merge-descriptors": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-2.0.0.tgz", + "integrity": "sha512-Snk314V5ayFLhp3fkUREub6WtjBfPdCPY1Ln8/8munuLuiYhsABgBVWsozAG+MWMbVEvcdcpbi9R7ww22l9Q3g==", + "license": "MIT", + "engines": { + "node": ">=18" + }, + "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.54.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.54.0.tgz", + "integrity": "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-3.0.2.tgz", + "integrity": "sha512-Lbgzdk0h4juoQ9fCKXW4by0UJqj+nOOrI9MJ1sSj4nI8aI2eo1qmvQEie4VD1glsS250n15LsWsYtCugiStS5A==", + "license": "MIT", + "dependencies": { + "mime-db": "^1.54.0" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "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": "1.0.0", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-1.0.0.tgz", + "integrity": "sha512-8Ofs/AUQh8MaEcrlq5xOX0CQ9ypTF5dl78mjlMNfOK08fzpgTHQRQPBxcPlEtIw0yRpws+Zo/3r+5WRby7u3Gg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-abort-controller": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/node-abort-controller/-/node-abort-controller-3.1.1.tgz", + "integrity": "sha512-AGK2yQKIjRuqnc6VkX2Xj5d+QW8xZ87pa1UK6yA6ouUyuxfHuMP6umE5QK7UmTeOAymo+Zx1Fxiuw9rVx8taHQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-emoji": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/node-emoji/-/node-emoji-1.11.0.tgz", + "integrity": "sha512-wo2DpQkQp7Sjm2A0cq+sN7EHKO6Sl0ctXeBdFZrL9T9+UywORbufTcTZxom8YqpLQt/FqNMUkOpkZrJVYSKD3A==", + "dev": true, + "license": "MIT", + "dependencies": { + "lodash": "^4.17.21" + } + }, + "node_modules/node-fetch": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz", + "integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==", + "license": "MIT", + "dependencies": { + "whatwg-url": "^5.0.0" + }, + "engines": { + "node": "4.x || >=6.0.0" + }, + "peerDependencies": { + "encoding": "^0.1.0" + }, + "peerDependenciesMeta": { + "encoding": { + "optional": true + } + } + }, + "node_modules/node-int64": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", + "integrity": "sha512-O5lz91xSOeoXP6DulyHfllpq+Eg00MWitZIbtPfoSEvqIHdl5gfcY6hYzDWnj0qD5tz52PI08u9qUvSVeUBeHw==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-releases": { + "version": "2.0.27", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.27.tgz", + "integrity": "sha512-nmh3lCkYZ3grZvqcCH+fjmQ7X+H0OeZgP40OierEaAptX4XofMh5kwNbWh7lBduUzCcV/8kZ+NDLCwm2iorIlA==", + "dev": true, + "license": "MIT" + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/npm-run-path": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz", + "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-inspect": { + "version": "1.13.4", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", + "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/on-finished": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", + "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", + "license": "MIT", + "dependencies": { + "ee-first": "1.1.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "license": "ISC", + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/optionator": { + "version": "0.9.4", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", + "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0", + "word-wrap": "^1.2.5" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/ora": { + "version": "5.4.1", + "resolved": "https://registry.npmjs.org/ora/-/ora-5.4.1.tgz", + "integrity": "sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "bl": "^4.1.0", + "chalk": "^4.1.0", + "cli-cursor": "^3.1.0", + "cli-spinners": "^2.5.0", + "is-interactive": "^1.0.0", + "is-unicode-supported": "^0.1.0", + "log-symbols": "^4.1.0", + "strip-ansi": "^6.0.0", + "wcwidth": "^1.0.1" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/os-tmpdir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-try": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/package-json-from-dist": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz", + "integrity": "sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==", + "dev": true, + "license": "BlueOak-1.0.0" + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "license": "MIT", + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/parse-json": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true, + "license": "MIT" + }, + "node_modules/path-scurry": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.11.1.tgz", + "integrity": "sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "lru-cache": "^10.2.0", + "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0" + }, + "engines": { + "node": ">=16 || 14 >=14.18" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/path-scurry/node_modules/lru-cache": { + "version": "10.4.3", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.4.3.tgz", + "integrity": "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/path-to-regexp": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-3.3.0.tgz", + "integrity": "sha512-qyCH421YQPS2WFDxDjftfc1ZR5WKQzVzqsp4n9M2kQhVOo/ByahFoUNJfl58kOcEGfQ//7weFTDhm+ss8Ecxgw==", + "license": "MIT" + }, + "node_modules/path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/picocolors": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", + "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", + "dev": true, + "license": "ISC" + }, + "node_modules/picomatch": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.1.tgz", + "integrity": "sha512-xUXwsxNjwTQ8K3GnT4pCJm+xq3RUPQbmkYJTP5aFIfNIvbcc/4MUxgBaaRSZJ6yGJZiGSyYlM6MzwTsRk8SYCg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/pirates": { + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.7.tgz", + "integrity": "sha512-TfySrs/5nm8fQJDcBDuUng3VOUKsd7S+zqvbOTiGXHfxX4wK31ard+hoNuvkicM/2YFzlpDgABOevKSsB4G/FA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 6" + } + }, + "node_modules/pkg-dir": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "find-up": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-dir/node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "license": "MIT", + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-dir/node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-dir/node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/pkg-dir/node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pluralize": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-8.0.0.tgz", + "integrity": "sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/prettier": { + "version": "3.7.4", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.7.4.tgz", + "integrity": "sha512-v6UNi1+3hSlVvv8fSaoUbggEM5VErKmmpGA7Pl3HF8V6uKY7rvClBOJlH6yNwQtfTueNkGVpOv/mtWL9L4bgRA==", + "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.1", + "resolved": "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.1.tgz", + "integrity": "sha512-SxToR7P8Y2lWmv/kTzVLC1t/GDI2WGjMwNhLLE9qtH8Q13C+aEmuRlzDst4Up4s0Wc8sF2M+J57iB3cMLqftfg==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-diff": "^1.1.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/prisma": { + "version": "5.22.0", + "resolved": "https://registry.npmjs.org/prisma/-/prisma-5.22.0.tgz", + "integrity": "sha512-vtpjW3XuYCSnMsNVBjLMNkTj6OZbudcPPTPYHqX0CJfpcdWciI1dM8uHETwmDxxiqEwCIE6WvXucWUetJgfu/A==", + "devOptional": true, + "hasInstallScript": true, + "license": "Apache-2.0", + "peer": true, + "dependencies": { + "@prisma/engines": "5.22.0" + }, + "bin": { + "prisma": "build/index.js" + }, + "engines": { + "node": ">=16.13" + }, + "optionalDependencies": { + "fsevents": "2.3.3" + } + }, + "node_modules/prompts": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", + "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "kleur": "^3.0.3", + "sisteransi": "^1.0.5" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "license": "MIT", + "dependencies": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/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.14.1", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.14.1.tgz", + "integrity": "sha512-4EK3+xJl8Ts67nLYNwqw/dsFVnCf+qR7RgXSK9jEEm9unao3njwMDdmsdvoKBKHzxd7tCYz5e5M+SnMjdtXGQQ==", + "license": "BSD-3-Clause", + "dependencies": { + "side-channel": "^1.1.0" + }, + "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.3", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.3.tgz", + "integrity": "sha512-s4VSOf6yN0rvbRZGxs8Om5CWj6seneMwK3oDb4lWDH0UPhWcxwOWw5+qk24bxq87szX1ydrwylIOp2uG1ojUpA==", + "license": "MIT", + "dependencies": { + "bytes": "~3.1.2", + "http-errors": "~2.0.1", + "iconv-lite": "~0.4.24", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, + "node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "license": "MIT", + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, + "license": "MIT", + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, + "node_modules/readdirp/node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/redis-errors": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/redis-errors/-/redis-errors-1.2.0.tgz", + "integrity": "sha512-1qny3OExCf0UvUV/5wpYKf2YwPcOqXzkwKKSmKHiE6ZMQs5heeE/c8eXK+PNllPvmjgAbfnsbpkGZWy8cBpn9w==", + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/redis-parser": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/redis-parser/-/redis-parser-3.0.0.tgz", + "integrity": "sha512-DJnGAeenTdpMEH6uAJRK/uiyEIH9WVsUmoLwzudwGJUwZPp80PDBWPHXSAGNPwNvIXAbe7MSUB1zQFugFml66A==", + "license": "MIT", + "dependencies": { + "redis-errors": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/reflect-metadata": { + "version": "0.1.14", + "resolved": "https://registry.npmjs.org/reflect-metadata/-/reflect-metadata-0.1.14.tgz", + "integrity": "sha512-ZhYeb6nRaXCfhnndflDK8qI6ZQ/YcWZCISRAWICW9XYqMUwjZM9Z0DveWX/ABN01oxSHwVxKQmxeYZSsm0jh5A==", + "license": "Apache-2.0", + "peer": true + }, + "node_modules/repeat-string": { + "version": "1.6.1", + "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", + "integrity": "sha512-PV0dzCYDNfRi1jCDbJzpW7jNNDRuCOG/jI5ctQcGKt/clZD+YcPS3yIlWuTJMmESC8aevCFmWJy5wjAFgNqN6w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10" + } + }, + "node_modules/require-directory": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", + "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/require-from-string": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", + "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve": { + "version": "1.22.11", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.11.tgz", + "integrity": "sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-core-module": "^2.16.1", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-cwd": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", + "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", + "dev": true, + "license": "MIT", + "dependencies": { + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/resolve-cwd/node_modules/resolve-from": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", + "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/resolve.exports": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/resolve.exports/-/resolve.exports-2.0.3.tgz", + "integrity": "sha512-OcXjMsGdhL4XnbShKpAcSqPMzQoYkYyhbEaeSko47MjRP9NfEQMhZkXL1DoFlt9LWQn4YttrdnV6X2OiyzBi+A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + } + }, + "node_modules/restore-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz", + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "dev": true, + "license": "MIT", + "dependencies": { + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/restore-cursor/node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/reusify": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz", + "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==", + "dev": true, + "license": "MIT", + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "deprecated": "Rimraf versions prior to v4 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/rimraf/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/rimraf/node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/rimraf/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/router": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/router/-/router-2.2.0.tgz", + "integrity": "sha512-nLTrUKm2UyiL7rlhapu/Zl45FwNgkZGaCpZbIHajDYgwlJCOzLSk+cIPAnsEqV955GjILJnKbdQC1nVPz+gAYQ==", + "license": "MIT", + "dependencies": { + "debug": "^4.4.0", + "depd": "^2.0.0", + "is-promise": "^4.0.0", + "parseurl": "^1.3.3", + "path-to-regexp": "^8.0.0" + }, + "engines": { + "node": ">= 18" + } + }, + "node_modules/router/node_modules/path-to-regexp": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-8.3.0.tgz", + "integrity": "sha512-7jdwVIRtsP8MYpdXSwOS0YdD0Du+qOoF/AEPIt88PcCFrZCzx41oxku1jD88hZBwbNUIEfpqvuhjFaMAqMTWnA==", + "license": "MIT", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/run-async": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.4.1.tgz", + "integrity": "sha512-tvVnVv01b8c1RrA6Ep7JkStj85Guv/YrMcwqYQnwjsAS2cTmmPGBBjAjpCW7RrSodNSoE2/qg9O4bceNvUuDgQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT", + "dependencies": { + "queue-microtask": "^1.2.2" + } + }, + "node_modules/rxjs": { + "version": "7.8.2", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-7.8.2.tgz", + "integrity": "sha512-dhKf903U/PQZY6boNNtAGdWbG85WAbjT/1xYoZIC7FAY0yWapOBQVsVrDl58W86//e1VpMNBtRV4MaXfdMySFA==", + "license": "Apache-2.0", + "peer": true, + "dependencies": { + "tslib": "^2.1.0" + } + }, + "node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", + "license": "MIT" + }, + "node_modules/schema-utils": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.3.0.tgz", + "integrity": "sha512-pN/yOAvcC+5rQ5nERGuwrjLlYvLTbCibnZ1I7B1LaiAz9BRBlE9GMgE/eqV30P7aJQUf7Ddimy/RsbYO/GrVGg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/json-schema": "^7.0.8", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/schema-utils/node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/schema-utils/node_modules/ajv-keywords": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.5.2.tgz", + "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "ajv": "^6.9.1" + } + }, + "node_modules/schema-utils/node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true, + "license": "MIT" + }, + "node_modules/semver": { + "version": "7.7.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", + "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/send": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/send/-/send-1.2.1.tgz", + "integrity": "sha512-1gnZf7DFcoIcajTjTwjwuDjzuz4PPcY2StKPlsGAQ1+YH20IRVrBaXSWmdjowTJ6u8Rc01PoYOGHXfP1mYcZNQ==", + "license": "MIT", + "dependencies": { + "debug": "^4.4.3", + "encodeurl": "^2.0.0", + "escape-html": "^1.0.3", + "etag": "^1.8.1", + "fresh": "^2.0.0", + "http-errors": "^2.0.1", + "mime-types": "^3.0.2", + "ms": "^2.1.3", + "on-finished": "^2.4.1", + "range-parser": "^1.2.1", + "statuses": "^2.0.2" + }, + "engines": { + "node": ">= 18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "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": "2.2.1", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-2.2.1.tgz", + "integrity": "sha512-xRXBn0pPqQTVQiC8wyQrKs2MOlX24zQ0POGaj0kultvoOCstBQM5yvOhAVSUwOMjQtTvsPWoNCHfPGwaaQJhTw==", + "license": "MIT", + "dependencies": { + "encodeurl": "^2.0.0", + "escape-html": "^1.0.3", + "parseurl": "^1.3.3", + "send": "^1.2.0" + }, + "engines": { + "node": ">= 18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/set-function-length": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", + "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", + "dev": true, + "license": "MIT", + "dependencies": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "gopd": "^1.0.1", + "has-property-descriptors": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/setprototypeof": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", + "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==", + "license": "ISC" + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "license": "MIT", + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/side-channel": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", + "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3", + "side-channel-list": "^1.0.0", + "side-channel-map": "^1.0.1", + "side-channel-weakmap": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-list": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", + "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-map": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", + "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-weakmap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", + "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3", + "side-channel-map": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/signal-exit": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", + "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/sisteransi": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", + "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", + "dev": true, + "license": "MIT" + }, + "node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/source-map": { + "version": "0.7.4", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.4.tgz", + "integrity": "sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">= 8" + } + }, + "node_modules/source-map-support": { + "version": "0.5.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/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.2", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.2.tgz", + "integrity": "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/streamsearch": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-1.1.0.tgz", + "integrity": "sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==", + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/string_decoder": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", + "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", + "license": "MIT", + "dependencies": { + "safe-buffer": "~5.2.0" + } + }, + "node_modules/string-length": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/string-length/-/string-length-4.0.2.tgz", + "integrity": "sha512-+l6rNN5fYHNhZZy41RXsYptCjA2Igmq4EG7kZAYFQI1E1VTXarr6ZPXBg6eq7Y6eK4FEhY6AJlyuFIb/v/S0VQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "char-regex": "^1.0.2", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width-cjs": { + "name": "string-width", + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi-cjs": { + "name": "strip-ansi", + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-bom": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-4.0.0.tgz", + "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-final-newline": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz", + "integrity": "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/strtok3": { + "version": "10.3.4", + "resolved": "https://registry.npmjs.org/strtok3/-/strtok3-10.3.4.tgz", + "integrity": "sha512-KIy5nylvC5le1OdaaoCJ07L+8iQzJHGH6pWDuzS+d07Cu7n1MZ2x26P8ZKIWfbK02+XIL8Mp4RkWeqdUCrDMfg==", + "license": "MIT", + "dependencies": { + "@tokenizer/token": "^0.3.0" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/Borewit" + } + }, + "node_modules/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/swagger-ui-express": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/swagger-ui-express/-/swagger-ui-express-5.0.1.tgz", + "integrity": "sha512-SrNU3RiBGTLLmFU8GIJdOdanJTl4TOmT27tt3bWWHppqYmAZ6IDuEuBvMU6nZq0zLEe6b/1rACXCgLZqO6ZfrA==", + "license": "MIT", + "dependencies": { + "swagger-ui-dist": ">=5.0.0" + }, + "engines": { + "node": ">= v0.10.32" + }, + "peerDependencies": { + "express": ">=4.0.0 || >=5.0.0-beta" + } + }, + "node_modules/symbol-observable": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-4.0.0.tgz", + "integrity": "sha512-b19dMThMV4HVFynSAM1++gBHAbk2Tc/osgLIBZMKsyqh34jb2e8Os7T6ZW/Bt3pJFdBTd2JwAnAAEQV7rSNvcQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10" + } + }, + "node_modules/synckit": { + "version": "0.11.11", + "resolved": "https://registry.npmjs.org/synckit/-/synckit-0.11.11.tgz", + "integrity": "sha512-MeQTA1r0litLUf0Rp/iisCaL8761lKAZHaimlbGK4j0HysC4PLfqygQj9srcs0m2RdtDYnF8UuYyKpbjHYp7Jw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@pkgr/core": "^0.2.9" + }, + "engines": { + "node": "^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/synckit" + } + }, + "node_modules/tapable": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.3.0.tgz", + "integrity": "sha512-g9ljZiwki/LfxmQADO3dEY1CbpmXT5Hm2fJ+QaGKwSXUylMybePR7/67YW7jOrrvjEgL1Fmz5kzyAjWVWLlucg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/terser": { + "version": "5.44.1", + "resolved": "https://registry.npmjs.org/terser/-/terser-5.44.1.tgz", + "integrity": "sha512-t/R3R/n0MSwnnazuPpPNVO60LX0SKL45pyl9YlvxIdkH0Of7D5qM2EVe+yASRIlY5pZ73nclYJfNANGWPwFDZw==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "@jridgewell/source-map": "^0.3.3", + "acorn": "^8.15.0", + "commander": "^2.20.0", + "source-map-support": "~0.5.20" + }, + "bin": { + "terser": "bin/terser" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/terser-webpack-plugin": { + "version": "5.3.16", + "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-5.3.16.tgz", + "integrity": "sha512-h9oBFCWrq78NyWWVcSwZarJkZ01c2AyGrzs1crmHZO3QUg9D61Wu4NPjBy69n7JqylFF5y+CsUZYmYEIZ3mR+Q==", + "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/terser/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/terser/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/test-exclude": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", + "integrity": "sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==", + "dev": true, + "license": "ISC", + "dependencies": { + "@istanbuljs/schema": "^0.1.2", + "glob": "^7.1.4", + "minimatch": "^3.0.4" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/test-exclude/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/test-exclude/node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/test-exclude/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true, + "license": "MIT" + }, + "node_modules/through": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==", + "dev": true, + "license": "MIT" + }, + "node_modules/tmp": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", + "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", + "dev": true, + "license": "MIT", + "dependencies": { + "os-tmpdir": "~1.0.2" + }, + "engines": { + "node": ">=0.6.0" + } + }, + "node_modules/tmpl": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/tmpl/-/tmpl-1.0.5.tgz", + "integrity": "sha512-3f0uOEAQwIqGuWW2MVzYg8fV/QNnc/IpuJNG837rLuczAaLVHslWHZQj4IGiEl5Hs3kkbhwL9Ab7Hrsmuj+Smw==", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/toidentifier": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", + "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", + "license": "MIT", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/token-types": { + "version": "6.1.2", + "resolved": "https://registry.npmjs.org/token-types/-/token-types-6.1.2.tgz", + "integrity": "sha512-dRXchy+C0IgK8WPC6xvCHFRIWYUbqqdEIKPaKo/AcTUNzwLTK6AH7RjdLWsEZcAN/TBdtfUw3PYEgPr5VPr6ww==", + "license": "MIT", + "dependencies": { + "@borewit/text-codec": "^0.2.1", + "@tokenizer/token": "^0.3.0", + "ieee754": "^1.2.1" + }, + "engines": { + "node": ">=14.16" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/Borewit" + } + }, + "node_modules/tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", + "license": "MIT" + }, + "node_modules/tree-kill": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", + "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==", + "dev": true, + "license": "MIT", + "bin": { + "tree-kill": "cli.js" + } + }, + "node_modules/ts-api-utils": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.4.3.tgz", + "integrity": "sha512-i3eMG77UTMD0hZhgRS562pv83RC6ukSAC2GMNWc+9dieh/+jDM5u5YG+NHX6VNDRHQcHwmsTHctP9LhbC3WxVw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=16" + }, + "peerDependencies": { + "typescript": ">=4.2.0" + } + }, + "node_modules/ts-jest": { + "version": "29.4.6", + "resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-29.4.6.tgz", + "integrity": "sha512-fSpWtOO/1AjSNQguk43hb/JCo16oJDnMJf3CdEGNkqsEX3t0KX96xvyX1D7PfLCpVoKu4MfVrqUkFyblYoY4lA==", + "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/type-is/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/type-is/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/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==", + "dev": true, + "license": "MIT" + }, + "node_modules/universalify": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.1.tgz", + "integrity": "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/update-browserslist-db": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.3.tgz", + "integrity": "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==", + "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": "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/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.26", + "resolved": "https://registry.npmjs.org/validator/-/validator-13.15.26.tgz", + "integrity": "sha512-spH26xU080ydGggxRyR1Yhcbgx+j3y5jbNXk/8L+iRvdIEQ4uTRH2Sgf2dokud6Q4oAtsbNvJ1Ft+9xmm6IZcA==", + "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.5.0", + "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-2.5.0.tgz", + "integrity": "sha512-e6vZvY6xboSwLz2GD36c16+O/2Z6fKvIf4pOXptw2rY9MVwE/TXc6RGqxD3I3x0a28lwBY7DE+76uTPSsBrrCA==", + "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.104.1", + "resolved": "https://registry.npmjs.org/webpack/-/webpack-5.104.1.tgz", + "integrity": "sha512-Qphch25abbMNtekmEGJmeRUhLDbe+QfiWTiqpKYkpCOWY64v9eyl+KRRLmqOFA2AvKPpc9DC6+u2n76tQLBoaA==", + "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.28.1", + "chrome-trace-event": "^1.0.2", + "enhanced-resolve": "^5.17.4", + "es-module-lexer": "^2.0.0", + "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.16", + "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/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/webpack/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==", + "dev": true, + "license": "MIT", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/webpack/node_modules/schema-utils": { + "version": "4.3.3", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.3.3.tgz", + "integrity": "sha512-eflK8wEtyOE6+hsaRVPxvUKYCpRgzLqDTb8krvAsRIwOGlHoSgYLgBXoubGgLd2fT41/OUYdb48v4k4WWHQurA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/json-schema": "^7.0.9", + "ajv": "^8.9.0", + "ajv-formats": "^2.1.1", + "ajv-keywords": "^5.1.0" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", + "license": "MIT", + "dependencies": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "license": "ISC", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/word-wrap": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", + "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/wordwrap": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", + "integrity": "sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/wrap-ansi": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", + "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi-cjs": { + "name": "wrap-ansi", + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "license": "ISC" + }, + "node_modules/write-file-atomic": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-4.0.2.tgz", + "integrity": "sha512-7KxauUdBmSdWnmpaGFg+ppNjKF8uNLry8LyzjauQDOVONfFLNKrKvQOxZ/VuTIcS/gge/YNahf5RIIQWTSarlg==", + "dev": true, + "license": "ISC", + "dependencies": { + "imurmurhash": "^0.1.4", + "signal-exit": "^3.0.7" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || >=16.0.0" + } + }, + "node_modules/write-file-atomic/node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/ws": { + "version": "8.17.1", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.17.1.tgz", + "integrity": "sha512-6XQFvXTkbfUOZOKKILFG1PDK2NDQs4azKQl26T0YS5CxqWLgXajbPZ+h4gZekJyRqFU8pvnbAbbs/3TgRPy+GQ==", + "license": "MIT", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": ">=5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + }, + "node_modules/xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", + "license": "MIT", + "engines": { + "node": ">=0.4" + } + }, + "node_modules/y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=10" + } + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true, + "license": "ISC" + }, + "node_modules/yargs": { + "version": "17.7.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "cliui": "^8.0.1", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.3", + "y18n": "^5.0.5", + "yargs-parser": "^21.1.1" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/yargs-parser": { + "version": "21.1.1", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", + "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=12" + } + }, + "node_modules/yn": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", + "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + } + } +} diff --git a/backend/services/mining-wallet-service/package.json b/backend/services/mining-wallet-service/package.json new file mode 100644 index 00000000..c20d51b3 --- /dev/null +++ b/backend/services/mining-wallet-service/package.json @@ -0,0 +1,63 @@ +{ + "name": "mining-wallet-service", + "version": "1.0.0", + "description": "Mining Wallet Service - 100% independent wallet management for mining ecosystem", + "scripts": { + "build": "nest build", + "format": "prettier --write \"src/**/*.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", + "prisma:generate": "prisma generate", + "prisma:migrate": "prisma migrate dev", + "prisma:migrate:prod": "prisma migrate deploy", + "prisma:studio": "prisma studio" + }, + "dependencies": { + "@nestjs/common": "^10.3.0", + "@nestjs/config": "^3.1.1", + "@nestjs/core": "^10.3.0", + "@nestjs/microservices": "^10.3.0", + "@nestjs/platform-express": "^10.3.0", + "@nestjs/schedule": "^4.0.0", + "@nestjs/swagger": "^7.1.17", + "@prisma/client": "^5.7.1", + "class-transformer": "^0.5.1", + "class-validator": "^0.14.0", + "decimal.js": "^10.4.3", + "ethers": "^6.9.0", + "ioredis": "^5.3.2", + "jsonwebtoken": "^9.0.2", + "kafkajs": "^2.2.4", + "reflect-metadata": "^0.1.14", + "rxjs": "^7.8.1", + "swagger-ui-express": "^5.0.0" + }, + "devDependencies": { + "@nestjs/cli": "^10.2.1", + "@nestjs/schematics": "^10.0.3", + "@nestjs/testing": "^10.3.0", + "@types/express": "^4.17.21", + "@types/jest": "^29.5.11", + "@types/jsonwebtoken": "^9.0.5", + "@types/node": "^20.10.5", + "@typescript-eslint/eslint-plugin": "^6.15.0", + "@typescript-eslint/parser": "^6.15.0", + "eslint": "^8.56.0", + "eslint-config-prettier": "^9.1.0", + "eslint-plugin-prettier": "^5.1.0", + "jest": "^29.7.0", + "prettier": "^3.1.1", + "prisma": "^5.7.1", + "ts-jest": "^29.1.1", + "ts-loader": "^9.5.1", + "ts-node": "^10.9.2", + "tsconfig-paths": "^4.2.0", + "typescript": "^5.3.3" + } +} diff --git a/backend/services/mining-wallet-service/prisma/schema.prisma b/backend/services/mining-wallet-service/prisma/schema.prisma new file mode 100644 index 00000000..5a1c75dc --- /dev/null +++ b/backend/services/mining-wallet-service/prisma/schema.prisma @@ -0,0 +1,682 @@ +// ============================================================================= +// Mining Wallet Service - Prisma Schema +// 100% 独立于 1.0 系统,完全隔离的钱包管理服务 +// ============================================================================= + +generator client { + provider = "prisma-client-js" +} + +datasource db { + provider = "postgresql" + url = env("DATABASE_URL") +} + +// ============================================================================= +// 枚举定义 +// ============================================================================= + +// 系统账户类型 +enum SystemAccountType { + HEADQUARTERS // 总部账户 + OPERATION // 运营账户 + PROVINCE // 省级公司账户 + CITY // 市级公司账户 + FEE // 手续费账户 + HOT_WALLET // 热钱包(KAVA链上) + COLD_WALLET // 冷钱包(离线存储) +} + +// 池账户类型 +enum PoolAccountType { + SHARE_POOL // 积分股池 - 总股池 + BLACK_HOLE_POOL // 黑洞积分股池 - 销毁池 + CIRCULATION_POOL // 流通积分股池 - 交易流通池 +} + +// 用户钱包账户类型 +enum UserWalletType { + CONTRIBUTION // 算力账户(hashpower/contribution) + TOKEN_STORAGE // 积分股存储账户 + GREEN_POINTS // 绿色积分账户 +} + +// 资产类型 +enum AssetType { + SHARE // 积分股 (RWA Token) + USDT // USDT 稳定币 + GREEN_POINT // 绿色积分 + CONTRIBUTION // 算力值 +} + +// 交易类型 +enum TransactionType { + // 挖矿相关 + MINING_REWARD // 挖矿奖励 + MINING_DISTRIBUTE // 挖矿分配 + + // 划转相关 + TRANSFER_IN // 划入 + TRANSFER_OUT // 划出 + INTERNAL_TRANSFER // 内部划转 + + // 交易相关 + TRADE_BUY // 买入 + TRADE_SELL // 卖出 + + // 提现/充值 + WITHDRAW // 提现到链上 + DEPOSIT // 从链上充值 + + // 销毁 + BURN // 销毁到黑洞 + + // 冻结/解冻 + FREEZE // 冻结 + UNFREEZE // 解冻 + + // 手续费 + FEE_COLLECT // 收取手续费 + FEE_DISTRIBUTE // 分发手续费 + + // 池操作 + POOL_INJECT // 注入池 + POOL_EXTRACT // 从池提取 + + // 系统调整 + ADJUSTMENT // 系统调整 + INITIAL_INJECT // 初始注入 +} + +// 交易对手方类型 +enum CounterpartyType { + USER // 用户 + SYSTEM_ACCOUNT // 系统账户 + POOL // 池账户 + BLOCKCHAIN // 区块链地址 + EXTERNAL // 外部 +} + +// 提现状态 +enum WithdrawStatus { + PENDING // 待处理 + PROCESSING // 处理中 + CONFIRMING // 链上确认中 + COMPLETED // 已完成 + FAILED // 失败 + CANCELLED // 已取消 +} + +// Outbox 状态 +enum OutboxStatus { + PENDING + PUBLISHED + FAILED +} + +// ============================================================================= +// 区域管理(100% 独立于 1.0) +// ============================================================================= + +// 省份 +model Province { + id String @id @default(uuid()) + code String @unique // 省份代码 e.g. "GD" + name String // 省份名称 e.g. "广东省" + status String @default("ACTIVE") // ACTIVE, DISABLED + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + cities City[] + systemAccounts SystemAccount[] + + @@index([code]) + @@map("provinces") +} + +// 城市 +model City { + id String @id @default(uuid()) + provinceId String @map("province_id") + code String @unique // 城市代码 e.g. "SZ" + name String // 城市名称 e.g. "深圳市" + status String @default("ACTIVE") // ACTIVE, DISABLED + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + province Province @relation(fields: [provinceId], references: [id]) + userMappings UserRegionMapping[] + systemAccounts SystemAccount[] + + @@index([provinceId]) + @@index([code]) + @@map("cities") +} + +// 用户区域映射(100% 独立于 1.0) +model UserRegionMapping { + id String @id @default(uuid()) + accountSequence String @unique @map("account_sequence") // 用户账户序列号 + cityId String @map("city_id") + assignedAt DateTime @default(now()) @map("assigned_at") + assignedBy String? @map("assigned_by") // 分配人(管理员ID) + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + city City @relation(fields: [cityId], references: [id]) + + @@index([cityId]) + @@map("user_region_mappings") +} + +// ============================================================================= +// 系统账户(HQ、运营、省/市公司、手续费、热/冷钱包) +// ============================================================================= + +model SystemAccount { + id String @id @default(uuid()) + accountType SystemAccountType @map("account_type") + name String // 账户名称 + code String @unique // 账户代码 e.g. "HQ", "OP", "GD-SZ" + + // 关联区域(省/市级公司账户使用) + provinceId String? @map("province_id") + cityId String? @map("city_id") + + // 余额信息 + shareBalance Decimal @default(0) @map("share_balance") @db.Decimal(30, 8) + usdtBalance Decimal @default(0) @map("usdt_balance") @db.Decimal(30, 8) + greenPointBalance Decimal @default(0) @map("green_point_balance") @db.Decimal(30, 8) + frozenShare Decimal @default(0) @map("frozen_share") @db.Decimal(30, 8) + frozenUsdt Decimal @default(0) @map("frozen_usdt") @db.Decimal(30, 8) + + // 累计统计 + totalInflow Decimal @default(0) @map("total_inflow") @db.Decimal(30, 8) + totalOutflow Decimal @default(0) @map("total_outflow") @db.Decimal(30, 8) + + // 链上地址(热/冷钱包使用) + blockchainAddress String? @map("blockchain_address") + + description String? + isActive Boolean @default(true) @map("is_active") + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + province Province? @relation(fields: [provinceId], references: [id]) + city City? @relation(fields: [cityId], references: [id]) + transactions SystemAccountTransaction[] + + @@unique([accountType, provinceId, cityId]) + @@index([accountType]) + @@index([provinceId]) + @@index([cityId]) + @@map("system_accounts") +} + +// 系统账户交易明细 +model SystemAccountTransaction { + id String @id @default(uuid()) + systemAccountId String @map("system_account_id") + transactionType TransactionType @map("transaction_type") + assetType AssetType @map("asset_type") + + // 金额信息 + amount Decimal @db.Decimal(30, 8) + balanceBefore Decimal @map("balance_before") @db.Decimal(30, 8) + balanceAfter Decimal @map("balance_after") @db.Decimal(30, 8) + + // 交易对手方信息 + counterpartyType CounterpartyType? @map("counterparty_type") + counterpartyAccountSeq String? @map("counterparty_account_seq") + counterpartyUserId String? @map("counterparty_user_id") + counterpartySystemId String? @map("counterparty_system_id") + counterpartyPoolType PoolAccountType? @map("counterparty_pool_type") + counterpartyAddress String? @map("counterparty_address") + + // 关联信息 + referenceId String? @map("reference_id") + referenceType String? @map("reference_type") + txHash String? @map("tx_hash") + + // 详细备注(格式: "手续费收取自用户[U123456], 交易订单ORD20240110001") + memo String? @db.Text + metadata Json? + + createdAt DateTime @default(now()) @map("created_at") + + systemAccount SystemAccount @relation(fields: [systemAccountId], references: [id]) + + @@index([systemAccountId, createdAt(sort: Desc)]) + @@index([transactionType]) + @@index([counterpartyAccountSeq]) + @@index([counterpartyUserId]) + @@index([referenceId]) + @@index([createdAt(sort: Desc)]) + @@map("system_account_transactions") +} + +// ============================================================================= +// 池账户(积分股池、黑洞池、流通池) +// ============================================================================= + +model PoolAccount { + id String @id @default(uuid()) + poolType PoolAccountType @unique @map("pool_type") + name String + + // 余额信息 + balance Decimal @default(0) @db.Decimal(30, 8) + totalInflow Decimal @default(0) @map("total_inflow") @db.Decimal(30, 8) + totalOutflow Decimal @default(0) @map("total_outflow") @db.Decimal(30, 8) + + // 黑洞池特有字段 + targetBurn Decimal? @map("target_burn") @db.Decimal(30, 8) + remainingBurn Decimal? @map("remaining_burn") @db.Decimal(30, 8) + + isActive Boolean @default(true) @map("is_active") + description String? + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + transactions PoolAccountTransaction[] + + @@index([poolType]) + @@map("pool_accounts") +} + +// 池账户交易明细 +model PoolAccountTransaction { + id String @id @default(uuid()) + poolAccountId String @map("pool_account_id") + poolType PoolAccountType @map("pool_type") + transactionType TransactionType @map("transaction_type") + + // 金额信息 + amount Decimal @db.Decimal(30, 8) + balanceBefore Decimal @map("balance_before") @db.Decimal(30, 8) + balanceAfter Decimal @map("balance_after") @db.Decimal(30, 8) + + // 交易对手方信息(关键:用户ID和账户序列号) + counterpartyType CounterpartyType? @map("counterparty_type") + counterpartyAccountSeq String? @map("counterparty_account_seq") + counterpartyUserId String? @map("counterparty_user_id") + counterpartySystemId String? @map("counterparty_system_id") + counterpartyPoolType PoolAccountType? @map("counterparty_pool_type") + counterpartyAddress String? @map("counterparty_address") + + // 关联信息 + referenceId String? @map("reference_id") + referenceType String? @map("reference_type") + txHash String? @map("tx_hash") + + // 详细备注(格式: "挖矿分配给用户[U123456], 算力占比0.5%, 分钟2024-01-10 10:30") + memo String? @db.Text + metadata Json? + + createdAt DateTime @default(now()) @map("created_at") + + poolAccount PoolAccount @relation(fields: [poolAccountId], references: [id]) + + @@index([poolAccountId, createdAt(sort: Desc)]) + @@index([poolType, transactionType]) + @@index([counterpartyAccountSeq]) + @@index([counterpartyUserId]) + @@index([referenceId]) + @@index([createdAt(sort: Desc)]) + @@map("pool_account_transactions") +} + +// ============================================================================= +// 用户钱包账户 +// ============================================================================= + +// 用户钱包(每个用户有多个钱包类型) +model UserWallet { + id String @id @default(uuid()) + accountSequence String @map("account_sequence") + walletType UserWalletType @map("wallet_type") + + // 余额信息 + balance Decimal @default(0) @db.Decimal(30, 8) + frozenBalance Decimal @default(0) @map("frozen_balance") @db.Decimal(30, 8) + + // 累计统计 + totalInflow Decimal @default(0) @map("total_inflow") @db.Decimal(30, 8) + totalOutflow Decimal @default(0) @map("total_outflow") @db.Decimal(30, 8) + + isActive Boolean @default(true) @map("is_active") + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + transactions UserWalletTransaction[] + + @@unique([accountSequence, walletType]) + @@index([accountSequence]) + @@index([walletType]) + @@map("user_wallets") +} + +// 用户钱包交易明细 +model UserWalletTransaction { + id String @id @default(uuid()) + userWalletId String @map("user_wallet_id") + accountSequence String @map("account_sequence") + walletType UserWalletType @map("wallet_type") + transactionType TransactionType @map("transaction_type") + assetType AssetType @map("asset_type") + + // 金额信息 + amount Decimal @db.Decimal(30, 8) + balanceBefore Decimal @map("balance_before") @db.Decimal(30, 8) + balanceAfter Decimal @map("balance_after") @db.Decimal(30, 8) + + // 交易对手方信息(关键:用户ID和账户序列号) + counterpartyType CounterpartyType? @map("counterparty_type") + counterpartyAccountSeq String? @map("counterparty_account_seq") + counterpartyUserId String? @map("counterparty_user_id") + counterpartySystemId String? @map("counterparty_system_id") + counterpartyPoolType PoolAccountType? @map("counterparty_pool_type") + counterpartyAddress String? @map("counterparty_address") + + // 关联信息 + referenceId String? @map("reference_id") + referenceType String? @map("reference_type") + txHash String? @map("tx_hash") + + // 详细备注(格式: "划转到交易账户, 接收方用户[U789012]") + memo String? @db.Text + metadata Json? + + createdAt DateTime @default(now()) @map("created_at") + + userWallet UserWallet @relation(fields: [userWalletId], references: [id]) + + @@index([userWalletId, createdAt(sort: Desc)]) + @@index([accountSequence, walletType]) + @@index([transactionType]) + @@index([counterpartyAccountSeq]) + @@index([counterpartyUserId]) + @@index([referenceId]) + @@index([createdAt(sort: Desc)]) + @@map("user_wallet_transactions") +} + +// ============================================================================= +// KAVA 区块链集成 +// ============================================================================= + +// 提现请求 +model WithdrawRequest { + id String @id @default(uuid()) + requestNo String @unique @map("request_no") + accountSequence String @map("account_sequence") + + // 提现信息 + assetType AssetType @map("asset_type") + amount Decimal @db.Decimal(30, 8) + fee Decimal @default(0) @db.Decimal(30, 8) + netAmount Decimal @map("net_amount") @db.Decimal(30, 8) + + // 目标地址 + toAddress String @map("to_address") + + // 状态 + status WithdrawStatus @default(PENDING) + + // 链上信息 + txHash String? @map("tx_hash") + blockNumber BigInt? @map("block_number") + confirmations Int @default(0) + + // 错误信息 + errorMessage String? @map("error_message") + + // 审核信息 + approvedBy String? @map("approved_by") + approvedAt DateTime? @map("approved_at") + + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + completedAt DateTime? @map("completed_at") + + @@index([accountSequence]) + @@index([status]) + @@index([txHash]) + @@index([createdAt(sort: Desc)]) + @@map("withdraw_requests") +} + +// 充值记录(链上检测到的充值) +model DepositRecord { + id String @id @default(uuid()) + txHash String @unique @map("tx_hash") + + // 来源信息 + fromAddress String @map("from_address") + toAddress String @map("to_address") + + // 充值信息 + assetType AssetType @map("asset_type") + amount Decimal @db.Decimal(30, 8) + + // 链上信息 + blockNumber BigInt @map("block_number") + confirmations Int @default(0) + + // 匹配的用户(如果能匹配到) + matchedAccountSeq String? @map("matched_account_seq") + isProcessed Boolean @default(false) @map("is_processed") + processedAt DateTime? @map("processed_at") + + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + @@index([fromAddress]) + @@index([toAddress]) + @@index([matchedAccountSeq]) + @@index([isProcessed]) + @@index([createdAt(sort: Desc)]) + @@map("deposit_records") +} + +// DEX Swap 记录 +model DexSwapRecord { + id String @id @default(uuid()) + swapNo String @unique @map("swap_no") + accountSequence String @map("account_sequence") + + // Swap 信息 + fromAsset AssetType @map("from_asset") + toAsset AssetType @map("to_asset") + fromAmount Decimal @map("from_amount") @db.Decimal(30, 8) + toAmount Decimal @map("to_amount") @db.Decimal(30, 8) + exchangeRate Decimal @map("exchange_rate") @db.Decimal(30, 18) + + // 滑点/手续费 + slippage Decimal @default(0) @db.Decimal(10, 4) + fee Decimal @default(0) @db.Decimal(30, 8) + + // 状态 + status WithdrawStatus @default(PENDING) + + // 链上信息 + txHash String? @map("tx_hash") + blockNumber BigInt? @map("block_number") + + errorMessage String? @map("error_message") + + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + completedAt DateTime? @map("completed_at") + + @@index([accountSequence]) + @@index([status]) + @@index([txHash]) + @@index([createdAt(sort: Desc)]) + @@map("dex_swap_records") +} + +// 链上地址绑定 +model BlockchainAddressBinding { + id String @id @default(uuid()) + accountSequence String @unique @map("account_sequence") + + // KAVA 地址 + kavaAddress String @unique @map("kava_address") + + // 验证信息 + isVerified Boolean @default(false) @map("is_verified") + verifiedAt DateTime? @map("verified_at") + verificationTxHash String? @map("verification_tx_hash") + + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + @@index([kavaAddress]) + @@map("blockchain_address_bindings") +} + +// 黑洞合约(KAVA 链上销毁地址) +model BlackHoleContract { + id String @id @default(uuid()) + contractAddress String @unique @map("contract_address") + name String + + // 累计销毁 + totalBurned Decimal @default(0) @map("total_burned") @db.Decimal(30, 8) + targetBurn Decimal @map("target_burn") @db.Decimal(30, 8) + remainingBurn Decimal @map("remaining_burn") @db.Decimal(30, 8) + + isActive Boolean @default(true) @map("is_active") + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + burnRecords BurnToBlackHoleRecord[] + + @@map("black_hole_contracts") +} + +// 销毁到黑洞的记录 +model BurnToBlackHoleRecord { + id String @id @default(uuid()) + blackHoleId String @map("black_hole_id") + + // 销毁信息 + amount Decimal @db.Decimal(30, 8) + + // 来源 + sourceType CounterpartyType @map("source_type") + sourceAccountSeq String? @map("source_account_seq") + sourceUserId String? @map("source_user_id") + sourcePoolType PoolAccountType? @map("source_pool_type") + + // 链上信息 + txHash String? @map("tx_hash") + blockNumber BigInt? @map("block_number") + + // 备注 + memo String? @db.Text + + createdAt DateTime @default(now()) @map("created_at") + + blackHole BlackHoleContract @relation(fields: [blackHoleId], references: [id]) + + @@index([blackHoleId]) + @@index([sourceAccountSeq]) + @@index([txHash]) + @@index([createdAt(sort: Desc)]) + @@map("burn_to_black_hole_records") +} + +// ============================================================================= +// 手续费配置 +// ============================================================================= + +model FeeConfig { + id String @id @default(uuid()) + feeType String @unique @map("fee_type") // WITHDRAW, TRADE, SWAP, TRANSFER + + // 费率配置 + feeRate Decimal @map("fee_rate") @db.Decimal(10, 6) // 费率 e.g. 0.001 = 0.1% + minFee Decimal @map("min_fee") @db.Decimal(30, 8) // 最低手续费 + maxFee Decimal? @map("max_fee") @db.Decimal(30, 8) // 最高手续费(可选) + + // 分配比例 + headquartersRate Decimal @map("headquarters_rate") @db.Decimal(10, 6) // 总部分成比例 + operationRate Decimal @map("operation_rate") @db.Decimal(10, 6) // 运营分成比例 + provinceRate Decimal @map("province_rate") @db.Decimal(10, 6) // 省级分成比例 + cityRate Decimal @map("city_rate") @db.Decimal(10, 6) // 市级分成比例 + + isActive Boolean @default(true) @map("is_active") + description String? + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + @@map("fee_configs") +} + +// ============================================================================= +// Outbox Pattern(事件发布) +// ============================================================================= + +model OutboxEvent { + id String @id @default(uuid()) + aggregateType String @map("aggregate_type") + aggregateId String @map("aggregate_id") + eventType String @map("event_type") + payload Json + topic String @default("mining-wallet.events") + key String? + status OutboxStatus @default(PENDING) + retryCount Int @default(0) @map("retry_count") + maxRetries Int @default(10) @map("max_retries") + lastError String? @map("last_error") + publishedAt DateTime? @map("published_at") + nextRetryAt DateTime? @map("next_retry_at") + createdAt DateTime @default(now()) @map("created_at") + + @@index([status]) + @@index([nextRetryAt]) + @@index([createdAt]) + @@map("outbox_events") +} + +// ============================================================================= +// 已处理事件(幂等性) +// ============================================================================= + +model ProcessedEvent { + id String @id @default(uuid()) + eventId String @unique @map("event_id") + eventType String @map("event_type") + sourceService String @map("source_service") + processedAt DateTime @default(now()) @map("processed_at") + + @@index([sourceService]) + @@index([processedAt]) + @@map("processed_events") +} + +// ============================================================================= +// 审计日志 +// ============================================================================= + +model AuditLog { + id String @id @default(uuid()) + operatorId String @map("operator_id") + operatorType String @map("operator_type") // ADMIN, SYSTEM, USER + action String // CREATE, UPDATE, DELETE, TRANSFER, WITHDRAW, etc. + resource String // SYSTEM_ACCOUNT, USER_WALLET, POOL, etc. + resourceId String? @map("resource_id") + oldValue Json? @map("old_value") + newValue Json? @map("new_value") + ipAddress String? @map("ip_address") + userAgent String? @map("user_agent") + createdAt DateTime @default(now()) @map("created_at") + + @@index([operatorId]) + @@index([action]) + @@index([resource]) + @@index([createdAt(sort: Desc)]) + @@map("audit_logs") +} diff --git a/backend/services/mining-wallet-service/src/api/api.module.ts b/backend/services/mining-wallet-service/src/api/api.module.ts new file mode 100644 index 00000000..37caa5ee --- /dev/null +++ b/backend/services/mining-wallet-service/src/api/api.module.ts @@ -0,0 +1,21 @@ +import { Module } from '@nestjs/common'; +import { HealthController } from './controllers/health.controller'; +import { SystemAccountController } from './controllers/system-account.controller'; +import { PoolAccountController } from './controllers/pool-account.controller'; +import { UserWalletController } from './controllers/user-wallet.controller'; +import { RegionController } from './controllers/region.controller'; +import { BlockchainController } from './controllers/blockchain.controller'; +import { ApplicationModule } from '../application/application.module'; + +@Module({ + imports: [ApplicationModule], + controllers: [ + HealthController, + SystemAccountController, + PoolAccountController, + UserWalletController, + RegionController, + BlockchainController, + ], +}) +export class ApiModule {} diff --git a/backend/services/mining-wallet-service/src/api/controllers/blockchain.controller.ts b/backend/services/mining-wallet-service/src/api/controllers/blockchain.controller.ts new file mode 100644 index 00000000..ff9ae6d0 --- /dev/null +++ b/backend/services/mining-wallet-service/src/api/controllers/blockchain.controller.ts @@ -0,0 +1,129 @@ +import { Controller, Get, Post, Body, Param, Query } from '@nestjs/common'; +import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth, ApiQuery } from '@nestjs/swagger'; +import { BlockchainIntegrationService } from '../../application/services/blockchain.service'; +import { CurrentUser, CurrentUserPayload } from '../../shared/decorators/current-user.decorator'; +import { AdminOnly } from '../../shared/guards/jwt-auth.guard'; +import { AssetType, WithdrawStatus } from '@prisma/client'; +import Decimal from 'decimal.js'; + +class CreateWithdrawRequestDto { + assetType: AssetType; + amount: string; + toAddress: string; +} + +class BindAddressDto { + kavaAddress: string; +} + +class CreateSwapRequestDto { + fromAsset: AssetType; + toAsset: AssetType; + fromAmount: string; + minToAmount: string; +} + +@ApiTags('Blockchain') +@Controller('blockchain') +@ApiBearerAuth() +export class BlockchainController { + constructor(private readonly blockchainService: BlockchainIntegrationService) {} + + // ==================== User Operations ==================== + + @Get('my/address') + @ApiOperation({ summary: '获取我的绑定地址' }) + async getMyAddress(@CurrentUser() user: CurrentUserPayload) { + return this.blockchainService.getUserAddressBinding(user.accountSequence); + } + + @Post('my/address') + @ApiOperation({ summary: '绑定KAVA地址' }) + async bindAddress( + @CurrentUser() user: CurrentUserPayload, + @Body() dto: BindAddressDto, + ) { + await this.blockchainService.bindUserAddress(user.accountSequence, dto.kavaAddress); + return { success: true, kavaAddress: dto.kavaAddress }; + } + + @Get('my/withdrawals') + @ApiOperation({ summary: '获取我的提现记录' }) + @ApiQuery({ name: 'status', required: false }) + @ApiQuery({ name: 'limit', required: false, type: Number }) + @ApiQuery({ name: 'offset', required: false, type: Number }) + async getMyWithdrawals( + @CurrentUser() user: CurrentUserPayload, + @Query('status') status?: WithdrawStatus, + @Query('limit') limit?: number, + @Query('offset') offset?: number, + ) { + return this.blockchainService.getUserWithdrawRequests(user.accountSequence, { + status, + limit: limit ? Number(limit) : undefined, + offset: offset ? Number(offset) : undefined, + }); + } + + @Post('my/withdraw') + @ApiOperation({ summary: '创建提现请求' }) + async createWithdrawRequest( + @CurrentUser() user: CurrentUserPayload, + @Body() dto: CreateWithdrawRequestDto, + ) { + return this.blockchainService.createWithdrawRequest( + user.accountSequence, + dto.assetType, + new Decimal(dto.amount), + dto.toAddress, + ); + } + + @Post('my/swap') + @ApiOperation({ summary: '创建DEX Swap请求' }) + async createSwapRequest( + @CurrentUser() user: CurrentUserPayload, + @Body() dto: CreateSwapRequestDto, + ) { + return this.blockchainService.createSwapRequest( + user.accountSequence, + dto.fromAsset, + dto.toAsset, + new Decimal(dto.fromAmount), + new Decimal(dto.minToAmount), + ); + } + + // ==================== Admin Operations ==================== + + @Get('withdrawals/:id') + @AdminOnly() + @ApiOperation({ summary: '获取提现请求详情' }) + async getWithdrawRequest(@Param('id') id: string) { + return this.blockchainService.getWithdrawRequest(id); + } + + @Post('withdrawals/:id/approve') + @AdminOnly() + @ApiOperation({ summary: '审批提现请求' }) + async approveWithdraw( + @Param('id') id: string, + @CurrentUser() user: CurrentUserPayload, + ) { + return this.blockchainService.approveWithdrawRequest(id, user.userId); + } + + @Post('withdrawals/:id/execute') + @AdminOnly() + @ApiOperation({ summary: '执行链上提现' }) + async executeWithdraw(@Param('id') id: string) { + return this.blockchainService.executeWithdraw(id); + } + + @Post('withdrawals/:id/confirm') + @AdminOnly() + @ApiOperation({ summary: '确认提现完成' }) + async confirmWithdraw(@Param('id') id: string) { + return this.blockchainService.confirmWithdrawComplete(id); + } +} diff --git a/backend/services/mining-wallet-service/src/api/controllers/health.controller.ts b/backend/services/mining-wallet-service/src/api/controllers/health.controller.ts new file mode 100644 index 00000000..e118d97d --- /dev/null +++ b/backend/services/mining-wallet-service/src/api/controllers/health.controller.ts @@ -0,0 +1,75 @@ +import { Controller, Get } from '@nestjs/common'; +import { ApiTags, ApiOperation, ApiResponse } from '@nestjs/swagger'; +import { Public } from '../../shared/guards/jwt-auth.guard'; +import { PrismaService } from '../../infrastructure/persistence/prisma/prisma.service'; +import { RedisService } from '../../infrastructure/redis/redis.service'; +import { KavaBlockchainService } from '../../infrastructure/blockchain/kava-blockchain.service'; + +@ApiTags('Health') +@Controller('health') +export class HealthController { + constructor( + private readonly prisma: PrismaService, + private readonly redis: RedisService, + private readonly kava: KavaBlockchainService, + ) {} + + @Get() + @Public() + @ApiOperation({ summary: '健康检查' }) + @ApiResponse({ status: 200, description: '服务健康' }) + async check() { + const checks = { + service: 'mining-wallet-service', + status: 'healthy', + timestamp: new Date().toISOString(), + checks: { + database: 'unknown', + redis: 'unknown', + blockchain: 'unknown', + }, + }; + + // Database check + try { + await this.prisma.$queryRaw`SELECT 1`; + checks.checks.database = 'healthy'; + } catch { + checks.checks.database = 'unhealthy'; + checks.status = 'degraded'; + } + + // Redis check + try { + await this.redis.set('health:check', 'ok', 10); + checks.checks.redis = 'healthy'; + } catch { + checks.checks.redis = 'unhealthy'; + checks.status = 'degraded'; + } + + // Blockchain check + checks.checks.blockchain = this.kava.isReady() ? 'healthy' : 'degraded'; + + return checks; + } + + @Get('live') + @Public() + @ApiOperation({ summary: 'Liveness 探针' }) + async liveness() { + return { status: 'ok' }; + } + + @Get('ready') + @Public() + @ApiOperation({ summary: 'Readiness 探针' }) + async readiness() { + try { + await this.prisma.$queryRaw`SELECT 1`; + return { status: 'ready' }; + } catch { + return { status: 'not ready' }; + } + } +} diff --git a/backend/services/mining-wallet-service/src/api/controllers/pool-account.controller.ts b/backend/services/mining-wallet-service/src/api/controllers/pool-account.controller.ts new file mode 100644 index 00000000..1b9d5775 --- /dev/null +++ b/backend/services/mining-wallet-service/src/api/controllers/pool-account.controller.ts @@ -0,0 +1,111 @@ +import { Controller, Get, Post, Body, Param, Query } from '@nestjs/common'; +import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth, ApiQuery } from '@nestjs/swagger'; +import { PoolAccountService } from '../../application/services/pool-account.service'; +import { AdminOnly } from '../../shared/guards/jwt-auth.guard'; +import { PoolAccountType, TransactionType } from '@prisma/client'; +import Decimal from 'decimal.js'; + +class InitializePoolsDto { + sharePool: { name: string; initialBalance?: string }; + blackHolePool: { name: string; targetBurn: string }; + circulationPool: { name: string; initialBalance?: string }; +} + +class BurnToBlackHoleDto { + amount: string; + fromPoolType: PoolAccountType; + counterpartyAccountSeq?: string; + counterpartyUserId?: string; + referenceId?: string; +} + +@ApiTags('Pool Accounts') +@Controller('pool-accounts') +@ApiBearerAuth() +export class PoolAccountController { + constructor(private readonly poolAccountService: PoolAccountService) {} + + @Get() + @AdminOnly() + @ApiOperation({ summary: '获取所有池账户' }) + @ApiResponse({ status: 200, description: '池账户列表' }) + async findAll() { + return this.poolAccountService.findAll(); + } + + @Get('stats') + @AdminOnly() + @ApiOperation({ summary: '获取池账户统计' }) + async getStats() { + return this.poolAccountService.getPoolStats(); + } + + @Get(':type') + @AdminOnly() + @ApiOperation({ summary: '获取特定类型池账户' }) + async findByType(@Param('type') type: PoolAccountType) { + return this.poolAccountService.findByType(type); + } + + @Get(':type/transactions') + @AdminOnly() + @ApiOperation({ summary: '获取池账户交易记录' }) + @ApiQuery({ name: 'transactionType', required: false }) + @ApiQuery({ name: 'startDate', required: false }) + @ApiQuery({ name: 'endDate', required: false }) + @ApiQuery({ name: 'limit', required: false, type: Number }) + @ApiQuery({ name: 'offset', required: false, type: Number }) + async getTransactions( + @Param('type') type: PoolAccountType, + @Query('transactionType') transactionType?: TransactionType, + @Query('startDate') startDate?: string, + @Query('endDate') endDate?: string, + @Query('limit') limit?: number, + @Query('offset') offset?: number, + ) { + return this.poolAccountService.getTransactions(type, { + transactionType, + startDate: startDate ? new Date(startDate) : undefined, + endDate: endDate ? new Date(endDate) : undefined, + limit: limit ? Number(limit) : undefined, + offset: offset ? Number(offset) : undefined, + }); + } + + @Post('initialize') + @AdminOnly() + @ApiOperation({ summary: '初始化池账户' }) + @ApiResponse({ status: 201, description: '池账户初始化成功' }) + async initialize(@Body() dto: InitializePoolsDto) { + return this.poolAccountService.initializePools({ + sharePool: { + name: dto.sharePool.name, + initialBalance: dto.sharePool.initialBalance + ? new Decimal(dto.sharePool.initialBalance) + : undefined, + }, + blackHolePool: { + name: dto.blackHolePool.name, + targetBurn: new Decimal(dto.blackHolePool.targetBurn), + }, + circulationPool: { + name: dto.circulationPool.name, + initialBalance: dto.circulationPool.initialBalance + ? new Decimal(dto.circulationPool.initialBalance) + : undefined, + }, + }); + } + + @Post('burn') + @AdminOnly() + @ApiOperation({ summary: '销毁到黑洞池' }) + async burnToBlackHole(@Body() dto: BurnToBlackHoleDto) { + return this.poolAccountService.burnToBlackHole(new Decimal(dto.amount), { + fromPoolType: dto.fromPoolType, + counterpartyAccountSeq: dto.counterpartyAccountSeq, + counterpartyUserId: dto.counterpartyUserId, + referenceId: dto.referenceId, + }); + } +} diff --git a/backend/services/mining-wallet-service/src/api/controllers/region.controller.ts b/backend/services/mining-wallet-service/src/api/controllers/region.controller.ts new file mode 100644 index 00000000..e1008d4a --- /dev/null +++ b/backend/services/mining-wallet-service/src/api/controllers/region.controller.ts @@ -0,0 +1,162 @@ +import { Controller, Get, Post, Body, Param, Query, Delete } from '@nestjs/common'; +import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth, ApiQuery } from '@nestjs/swagger'; +import { RegionRepository } from '../../infrastructure/persistence/repositories/region.repository'; +import { AdminOnly } from '../../shared/guards/jwt-auth.guard'; + +class CreateProvinceDto { + code: string; + name: string; +} + +class CreateCityDto { + provinceId: string; + code: string; + name: string; +} + +class AssignUserRegionDto { + accountSequence: string; + cityId: string; + assignedBy?: string; +} + +class BulkInitializeRegionsDto { + regions: { + provinceCode: string; + provinceName: string; + cities: { code: string; name: string }[]; + }[]; +} + +@ApiTags('Regions') +@Controller('regions') +@ApiBearerAuth() +export class RegionController { + constructor(private readonly regionRepo: RegionRepository) {} + + // ==================== Province ==================== + + @Get('provinces') + @AdminOnly() + @ApiOperation({ summary: '获取所有省份' }) + async getAllProvinces() { + return this.regionRepo.findAllProvinces(); + } + + @Get('provinces/:id') + @AdminOnly() + @ApiOperation({ summary: '根据ID获取省份' }) + async getProvinceById(@Param('id') id: string) { + return this.regionRepo.findProvinceById(id); + } + + @Post('provinces') + @AdminOnly() + @ApiOperation({ summary: '创建省份' }) + async createProvince(@Body() dto: CreateProvinceDto) { + return this.regionRepo.createProvince(dto); + } + + // ==================== City ==================== + + @Get('cities') + @AdminOnly() + @ApiOperation({ summary: '获取所有城市' }) + async getAllCities() { + return this.regionRepo.findAllCities(); + } + + @Get('cities/by-province/:provinceId') + @AdminOnly() + @ApiOperation({ summary: '根据省份获取城市' }) + async getCitiesByProvince(@Param('provinceId') provinceId: string) { + return this.regionRepo.findCitiesByProvince(provinceId); + } + + @Get('cities/:id') + @AdminOnly() + @ApiOperation({ summary: '根据ID获取城市' }) + async getCityById(@Param('id') id: string) { + return this.regionRepo.findCityById(id); + } + + @Post('cities') + @AdminOnly() + @ApiOperation({ summary: '创建城市' }) + async createCity(@Body() dto: CreateCityDto) { + return this.regionRepo.createCity(dto); + } + + // ==================== User Region Mapping ==================== + + @Get('user-mappings/:accountSequence') + @AdminOnly() + @ApiOperation({ summary: '获取用户区域映射' }) + async getUserRegion(@Param('accountSequence') accountSequence: string) { + return this.regionRepo.findUserRegion(accountSequence); + } + + @Get('user-mappings/by-city/:cityId') + @AdminOnly() + @ApiOperation({ summary: '根据城市获取用户列表' }) + @ApiQuery({ name: 'limit', required: false, type: Number }) + @ApiQuery({ name: 'offset', required: false, type: Number }) + async getUsersByCity( + @Param('cityId') cityId: string, + @Query('limit') limit?: number, + @Query('offset') offset?: number, + ) { + return this.regionRepo.findUsersByCity(cityId, { + limit: limit ? Number(limit) : undefined, + offset: offset ? Number(offset) : undefined, + }); + } + + @Get('user-mappings/by-province/:provinceId') + @AdminOnly() + @ApiOperation({ summary: '根据省份获取用户列表' }) + @ApiQuery({ name: 'limit', required: false, type: Number }) + @ApiQuery({ name: 'offset', required: false, type: Number }) + async getUsersByProvince( + @Param('provinceId') provinceId: string, + @Query('limit') limit?: number, + @Query('offset') offset?: number, + ) { + return this.regionRepo.findUsersByProvince(provinceId, { + limit: limit ? Number(limit) : undefined, + offset: offset ? Number(offset) : undefined, + }); + } + + @Post('user-mappings') + @AdminOnly() + @ApiOperation({ summary: '分配用户到区域' }) + async assignUserToRegion(@Body() dto: AssignUserRegionDto) { + return this.regionRepo.assignUserToRegion(dto); + } + + @Delete('user-mappings/:accountSequence') + @AdminOnly() + @ApiOperation({ summary: '移除用户区域映射' }) + async removeUserFromRegion(@Param('accountSequence') accountSequence: string) { + await this.regionRepo.removeUserFromRegion(accountSequence); + return { success: true }; + } + + // ==================== Statistics & Initialization ==================== + + @Get('stats') + @AdminOnly() + @ApiOperation({ summary: '获取区域统计' }) + async getRegionStats() { + return this.regionRepo.getRegionStats(); + } + + @Post('bulk-initialize') + @AdminOnly() + @ApiOperation({ summary: '批量初始化区域数据' }) + async bulkInitializeRegions(@Body() dto: BulkInitializeRegionsDto) { + await this.regionRepo.bulkInitializeRegions(dto.regions); + return { success: true, message: 'Regions initialized successfully' }; + } +} diff --git a/backend/services/mining-wallet-service/src/api/controllers/system-account.controller.ts b/backend/services/mining-wallet-service/src/api/controllers/system-account.controller.ts new file mode 100644 index 00000000..fa368221 --- /dev/null +++ b/backend/services/mining-wallet-service/src/api/controllers/system-account.controller.ts @@ -0,0 +1,70 @@ +import { Controller, Get, Post, Body, Param, Query } from '@nestjs/common'; +import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth } from '@nestjs/swagger'; +import { SystemAccountService } from '../../application/services/system-account.service'; +import { AdminOnly } from '../../shared/guards/jwt-auth.guard'; +import { SystemAccountType } from '@prisma/client'; + +class InitializeSystemAccountsDto { + headquarters: { name: string; code: string }; + operation: { name: string; code: string }; + fee: { name: string; code: string }; + hotWallet: { name: string; code: string; blockchainAddress?: string }; + coldWallet?: { name: string; code: string; blockchainAddress?: string }; +} + +class CreateRegionAccountDto { + name: string; + code: string; + regionId: string; +} + +@ApiTags('System Accounts') +@Controller('system-accounts') +@ApiBearerAuth() +export class SystemAccountController { + constructor(private readonly systemAccountService: SystemAccountService) {} + + @Get() + @AdminOnly() + @ApiOperation({ summary: '获取所有系统账户' }) + @ApiResponse({ status: 200, description: '系统账户列表' }) + async findAll() { + return this.systemAccountService.findAll(); + } + + @Get('by-type/:type') + @AdminOnly() + @ApiOperation({ summary: '按类型获取系统账户' }) + async findByType(@Param('type') type: SystemAccountType) { + return this.systemAccountService.findByType(type); + } + + @Get('by-code/:code') + @AdminOnly() + @ApiOperation({ summary: '按代码获取系统账户' }) + async findByCode(@Param('code') code: string) { + return this.systemAccountService.findByCode(code); + } + + @Post('initialize') + @AdminOnly() + @ApiOperation({ summary: '初始化核心系统账户' }) + @ApiResponse({ status: 201, description: '系统账户初始化成功' }) + async initialize(@Body() dto: InitializeSystemAccountsDto) { + return this.systemAccountService.initializeCoreAccounts(dto); + } + + @Post('province') + @AdminOnly() + @ApiOperation({ summary: '创建省级公司账户' }) + async createProvinceAccount(@Body() dto: CreateRegionAccountDto) { + return this.systemAccountService.createProvinceAccount(dto.regionId, dto.name, dto.code); + } + + @Post('city') + @AdminOnly() + @ApiOperation({ summary: '创建市级公司账户' }) + async createCityAccount(@Body() dto: CreateRegionAccountDto) { + return this.systemAccountService.createCityAccount(dto.regionId, dto.name, dto.code); + } +} diff --git a/backend/services/mining-wallet-service/src/api/controllers/user-wallet.controller.ts b/backend/services/mining-wallet-service/src/api/controllers/user-wallet.controller.ts new file mode 100644 index 00000000..61e873ef --- /dev/null +++ b/backend/services/mining-wallet-service/src/api/controllers/user-wallet.controller.ts @@ -0,0 +1,95 @@ +import { Controller, Get, Post, Body, Param, Query } from '@nestjs/common'; +import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth, ApiQuery } from '@nestjs/swagger'; +import { UserWalletService } from '../../application/services/user-wallet.service'; +import { CurrentUser, CurrentUserPayload } from '../../shared/decorators/current-user.decorator'; +import { UserWalletType, AssetType, TransactionType } from '@prisma/client'; +import Decimal from 'decimal.js'; + +class TransferDto { + toAccountSeq: string; + walletType: UserWalletType; + assetType: AssetType; + amount: string; + memo?: string; +} + +@ApiTags('User Wallets') +@Controller('user-wallets') +@ApiBearerAuth() +export class UserWalletController { + constructor(private readonly userWalletService: UserWalletService) {} + + @Get('my') + @ApiOperation({ summary: '获取当前用户所有钱包' }) + @ApiResponse({ status: 200, description: '用户钱包列表' }) + async getMyWallets(@CurrentUser() user: CurrentUserPayload) { + return this.userWalletService.getUserWallets(user.accountSequence); + } + + @Get('my/summary') + @ApiOperation({ summary: '获取当前用户钱包汇总' }) + async getMyWalletSummary(@CurrentUser() user: CurrentUserPayload) { + return this.userWalletService.getUserWalletSummary(user.accountSequence); + } + + @Get('my/:walletType') + @ApiOperation({ summary: '获取当前用户特定类型钱包' }) + async getMyWallet( + @CurrentUser() user: CurrentUserPayload, + @Param('walletType') walletType: UserWalletType, + ) { + return this.userWalletService.getUserWallet(user.accountSequence, walletType); + } + + @Get('my/transactions') + @ApiOperation({ summary: '获取当前用户交易记录' }) + @ApiQuery({ name: 'walletType', required: false }) + @ApiQuery({ name: 'transactionType', required: false }) + @ApiQuery({ name: 'assetType', required: false }) + @ApiQuery({ name: 'startDate', required: false }) + @ApiQuery({ name: 'endDate', required: false }) + @ApiQuery({ name: 'limit', required: false, type: Number }) + @ApiQuery({ name: 'offset', required: false, type: Number }) + async getMyTransactions( + @CurrentUser() user: CurrentUserPayload, + @Query('walletType') walletType?: UserWalletType, + @Query('transactionType') transactionType?: TransactionType, + @Query('assetType') assetType?: AssetType, + @Query('startDate') startDate?: string, + @Query('endDate') endDate?: string, + @Query('limit') limit?: number, + @Query('offset') offset?: number, + ) { + return this.userWalletService.getTransactions(user.accountSequence, { + walletType, + transactionType, + assetType, + startDate: startDate ? new Date(startDate) : undefined, + endDate: endDate ? new Date(endDate) : undefined, + limit: limit ? Number(limit) : undefined, + offset: offset ? Number(offset) : undefined, + }); + } + + @Post('my/transfer') + @ApiOperation({ summary: '用户间转账' }) + async transfer( + @CurrentUser() user: CurrentUserPayload, + @Body() dto: TransferDto, + ) { + return this.userWalletService.transferBetweenUsers( + user.accountSequence, + dto.toAccountSeq, + dto.walletType, + dto.assetType, + new Decimal(dto.amount), + { memo: dto.memo }, + ); + } + + @Post('create') + @ApiOperation({ summary: '为用户创建所有钱包' }) + async createWallets(@CurrentUser() user: CurrentUserPayload) { + return this.userWalletService.createWalletsForUser(user.accountSequence); + } +} diff --git a/backend/services/mining-wallet-service/src/app.module.ts b/backend/services/mining-wallet-service/src/app.module.ts new file mode 100644 index 00000000..73a796fd --- /dev/null +++ b/backend/services/mining-wallet-service/src/app.module.ts @@ -0,0 +1,45 @@ +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 { InfrastructureModule } from './infrastructure/infrastructure.module'; +import { ApplicationModule } from './application/application.module'; +import { DomainExceptionFilter } from './shared/filters/domain-exception.filter'; +import { TransformInterceptor } from './shared/interceptors/transform.interceptor'; +import { LoggingInterceptor } from './shared/interceptors/logging.interceptor'; +import { JwtAuthGuard } from './shared/guards/jwt-auth.guard'; + +@Module({ + imports: [ + ConfigModule.forRoot({ + isGlobal: true, + envFilePath: [ + `.env.${process.env.NODE_ENV || 'development'}`, + '.env', + ], + ignoreEnvFile: false, + }), + InfrastructureModule, + ApplicationModule, + ApiModule, + ], + providers: [ + { + provide: APP_FILTER, + useClass: DomainExceptionFilter, + }, + { + provide: APP_INTERCEPTOR, + useClass: LoggingInterceptor, + }, + { + provide: APP_INTERCEPTOR, + useClass: TransformInterceptor, + }, + { + provide: APP_GUARD, + useClass: JwtAuthGuard, + }, + ], +}) +export class AppModule {} diff --git a/backend/services/mining-wallet-service/src/application/application.module.ts b/backend/services/mining-wallet-service/src/application/application.module.ts new file mode 100644 index 00000000..d09a163b --- /dev/null +++ b/backend/services/mining-wallet-service/src/application/application.module.ts @@ -0,0 +1,31 @@ +import { Module } from '@nestjs/common'; +import { ScheduleModule } from '@nestjs/schedule'; + +// Services +import { SystemAccountService } from './services/system-account.service'; +import { PoolAccountService } from './services/pool-account.service'; +import { UserWalletService } from './services/user-wallet.service'; +import { BlockchainIntegrationService } from './services/blockchain.service'; + +// Schedulers +import { OutboxScheduler } from './schedulers/outbox.scheduler'; + +@Module({ + imports: [ScheduleModule.forRoot()], + providers: [ + // Services + SystemAccountService, + PoolAccountService, + UserWalletService, + BlockchainIntegrationService, + // Schedulers + OutboxScheduler, + ], + exports: [ + SystemAccountService, + PoolAccountService, + UserWalletService, + BlockchainIntegrationService, + ], +}) +export class ApplicationModule {} diff --git a/backend/services/mining-wallet-service/src/application/schedulers/outbox.scheduler.ts b/backend/services/mining-wallet-service/src/application/schedulers/outbox.scheduler.ts new file mode 100644 index 00000000..72ba6035 --- /dev/null +++ b/backend/services/mining-wallet-service/src/application/schedulers/outbox.scheduler.ts @@ -0,0 +1,97 @@ +import { Injectable, Logger, OnModuleInit } from '@nestjs/common'; +import { Cron, CronExpression } from '@nestjs/schedule'; +import { OutboxRepository } from '../../infrastructure/persistence/repositories/outbox.repository'; +import { KafkaProducerService } from '../../infrastructure/kafka/kafka-producer.service'; +import { RedisService } from '../../infrastructure/redis/redis.service'; + +@Injectable() +export class OutboxScheduler implements OnModuleInit { + private readonly logger = new Logger(OutboxScheduler.name); + private readonly LOCK_KEY = 'mining-wallet:outbox:scheduler:lock'; + private readonly LOCK_TTL = 30; // seconds + + constructor( + private readonly outboxRepo: OutboxRepository, + private readonly kafkaProducer: KafkaProducerService, + private readonly redis: RedisService, + ) {} + + async onModuleInit() { + this.logger.log('Outbox scheduler initialized'); + } + + @Cron(CronExpression.EVERY_5_SECONDS) + async processOutboxEvents() { + const lockValue = await this.redis.acquireLock(this.LOCK_KEY, this.LOCK_TTL); + if (!lockValue) { + return; // Another instance is processing + } + + try { + const events = await this.outboxRepo.findPendingEvents(100); + + if (events.length === 0) { + return; + } + + this.logger.debug(`Processing ${events.length} outbox events`); + + for (const event of events) { + try { + await this.kafkaProducer.emit(event.topic, { + key: event.key || event.aggregateId, + value: { + eventId: event.id, + aggregateType: event.aggregateType, + aggregateId: event.aggregateId, + eventType: event.eventType, + payload: event.payload, + createdAt: event.createdAt.toISOString(), + }, + headers: { + 'event-type': event.eventType, + 'aggregate-type': event.aggregateType, + }, + }); + + await this.outboxRepo.markAsPublished(event.id); + this.logger.debug(`Published event: ${event.id} (${event.eventType})`); + } catch (error) { + const errorMessage = error instanceof Error ? error.message : 'Unknown error'; + this.logger.error(`Failed to publish event ${event.id}: ${errorMessage}`); + + await this.outboxRepo.markAsFailed( + event.id, + errorMessage, + event.retryCount, + event.maxRetries, + ); + } + } + } finally { + await this.redis.releaseLock(this.LOCK_KEY, lockValue); + } + } + + @Cron(CronExpression.EVERY_HOUR) + async cleanupOldEvents() { + const lockValue = await this.redis.acquireLock(`${this.LOCK_KEY}:cleanup`, 60); + if (!lockValue) { + return; + } + + try { + // 删除 7 天前已发布的事件 + const before = new Date(); + before.setDate(before.getDate() - 7); + + const deletedCount = await this.outboxRepo.deletePublished(before); + + if (deletedCount > 0) { + this.logger.log(`Cleaned up ${deletedCount} old outbox events`); + } + } finally { + await this.redis.releaseLock(`${this.LOCK_KEY}:cleanup`, lockValue); + } + } +} diff --git a/backend/services/mining-wallet-service/src/application/services/blockchain.service.ts b/backend/services/mining-wallet-service/src/application/services/blockchain.service.ts new file mode 100644 index 00000000..c561ee45 --- /dev/null +++ b/backend/services/mining-wallet-service/src/application/services/blockchain.service.ts @@ -0,0 +1,353 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { BlockchainRepository } from '../../infrastructure/persistence/repositories/blockchain.repository'; +import { UserWalletRepository } from '../../infrastructure/persistence/repositories/user-wallet.repository'; +import { OutboxRepository } from '../../infrastructure/persistence/repositories/outbox.repository'; +import { KavaBlockchainService } from '../../infrastructure/blockchain/kava-blockchain.service'; +import { WithdrawRequest, DepositRecord, DexSwapRecord, AssetType, WithdrawStatus } from '@prisma/client'; +import Decimal from 'decimal.js'; +import { DomainException } from '../../shared/filters/domain-exception.filter'; + +@Injectable() +export class BlockchainIntegrationService { + private readonly logger = new Logger(BlockchainIntegrationService.name); + + constructor( + private readonly blockchainRepo: BlockchainRepository, + private readonly userWalletRepo: UserWalletRepository, + private readonly outboxRepo: OutboxRepository, + private readonly kavaService: KavaBlockchainService, + ) {} + + /** + * 创建提现请求 + */ + async createWithdrawRequest( + accountSequence: string, + assetType: AssetType, + amount: Decimal, + toAddress: string, + feeRate: Decimal = new Decimal('0.001'), + ): Promise { + // 验证地址格式 + if (!this.kavaService.isValidAddress(toAddress)) { + throw new DomainException('Invalid blockchain address', 'INVALID_ADDRESS'); + } + + // 检查用户余额 + const wallet = await this.userWalletRepo.findByAccountAndType(accountSequence, 'TOKEN_STORAGE'); + if (!wallet) { + throw new DomainException('User wallet not found', 'WALLET_NOT_FOUND'); + } + + const balance = new Decimal(wallet.balance.toString()); + if (balance.lessThan(amount)) { + throw new DomainException(`Insufficient balance: ${balance} < ${amount}`, 'INSUFFICIENT_BALANCE'); + } + + // 计算手续费 + const fee = amount.mul(feeRate); + if (fee.greaterThan(amount)) { + throw new DomainException('Fee exceeds amount', 'FEE_TOO_HIGH'); + } + + // 生成请求号 + const requestNo = `WD${Date.now()}${Math.random().toString(36).substring(2, 8).toUpperCase()}`; + + // 创建提现请求 + const request = await this.blockchainRepo.createWithdrawRequest({ + requestNo, + accountSequence, + assetType, + amount, + fee, + toAddress, + }); + + // 冻结用户余额 + await this.userWalletRepo.freezeBalance( + accountSequence, + 'TOKEN_STORAGE', + 'SHARE', + amount, + true, + { + referenceId: request.id, + referenceType: 'WITHDRAW_REQUEST', + memo: `提现冻结, 目标地址${toAddress}, 数量${amount.toFixed(8)}`, + }, + ); + + await this.outboxRepo.create({ + aggregateType: 'WithdrawRequest', + aggregateId: request.id, + eventType: 'WITHDRAW_REQUEST_CREATED', + payload: { + requestId: request.id, + requestNo, + accountSequence, + assetType, + amount: amount.toString(), + fee: fee.toString(), + toAddress, + }, + }); + + this.logger.log(`Withdraw request created: ${requestNo}`); + return request; + } + + /** + * 审批提现请求 + */ + async approveWithdrawRequest( + requestId: string, + approvedBy: string, + ): Promise { + const request = await this.blockchainRepo.findWithdrawById(requestId); + if (!request) { + throw new DomainException('Withdraw request not found', 'REQUEST_NOT_FOUND'); + } + + if (request.status !== 'PENDING') { + throw new DomainException(`Cannot approve request in status: ${request.status}`, 'INVALID_STATUS'); + } + + const updated = await this.blockchainRepo.updateWithdrawStatus(requestId, 'PROCESSING', { + approvedBy, + }); + + await this.outboxRepo.create({ + aggregateType: 'WithdrawRequest', + aggregateId: requestId, + eventType: 'WITHDRAW_REQUEST_APPROVED', + payload: { + requestId, + requestNo: request.requestNo, + approvedBy, + approvedAt: updated.approvedAt?.toISOString(), + }, + }); + + return updated; + } + + /** + * 执行链上提现 + */ + async executeWithdraw(requestId: string): Promise { + const request = await this.blockchainRepo.findWithdrawById(requestId); + if (!request) { + throw new DomainException('Withdraw request not found', 'REQUEST_NOT_FOUND'); + } + + if (request.status !== 'PROCESSING') { + throw new DomainException(`Cannot execute request in status: ${request.status}`, 'INVALID_STATUS'); + } + + if (!this.kavaService.isReady()) { + throw new DomainException('Blockchain service not ready', 'BLOCKCHAIN_NOT_READY'); + } + + try { + // 执行链上转账 + const netAmount = new Decimal(request.netAmount.toString()); + const result = await this.kavaService.sendNative(request.toAddress, netAmount); + + // 更新状态为确认中 + const updated = await this.blockchainRepo.updateWithdrawStatus(requestId, 'CONFIRMING', { + txHash: result.txHash, + blockNumber: BigInt(result.blockNumber), + }); + + await this.outboxRepo.create({ + aggregateType: 'WithdrawRequest', + aggregateId: requestId, + eventType: 'WITHDRAW_TX_SENT', + payload: { + requestId, + txHash: result.txHash, + blockNumber: result.blockNumber, + }, + }); + + return updated; + } catch (error) { + // 标记失败 + await this.blockchainRepo.updateWithdrawStatus(requestId, 'FAILED', { + errorMessage: error instanceof Error ? error.message : 'Unknown error', + }); + + // 解冻用户余额 + await this.userWalletRepo.freezeBalance( + request.accountSequence, + 'TOKEN_STORAGE', + 'SHARE', + new Decimal(request.amount.toString()), + false, + { + referenceId: requestId, + referenceType: 'WITHDRAW_REQUEST', + memo: `提现失败解冻, 原因: ${error instanceof Error ? error.message : 'Unknown error'}`, + }, + ); + + throw error; + } + } + + /** + * 确认提现完成 + */ + async confirmWithdrawComplete(requestId: string): Promise { + const request = await this.blockchainRepo.findWithdrawById(requestId); + if (!request) { + throw new DomainException('Withdraw request not found', 'REQUEST_NOT_FOUND'); + } + + if (request.status !== 'CONFIRMING' || !request.txHash) { + throw new DomainException(`Cannot confirm request in status: ${request.status}`, 'INVALID_STATUS'); + } + + // 检查链上确认 + const txStatus = await this.kavaService.getTransactionStatus(request.txHash); + if (!txStatus.confirmed || txStatus.status !== 'success') { + throw new DomainException('Transaction not confirmed yet', 'TX_NOT_CONFIRMED'); + } + + // 更新为完成 + const updated = await this.blockchainRepo.updateWithdrawStatus(requestId, 'COMPLETED', { + confirmations: txStatus.confirmations, + }); + + // 从冻结余额扣除(实际扣款) + await this.userWalletRepo.updateBalanceWithTransaction( + request.accountSequence, + 'TOKEN_STORAGE', + 'SHARE', + new Decimal(request.amount.toString()).negated(), + { + transactionType: 'WITHDRAW', + counterpartyType: 'BLOCKCHAIN', + counterpartyAddress: request.toAddress, + referenceId: requestId, + referenceType: 'WITHDRAW_REQUEST', + txHash: request.txHash, + memo: `提现成功, 目标地址${request.toAddress}, 数量${request.netAmount}, 手续费${request.fee}`, + }, + ); + + await this.outboxRepo.create({ + aggregateType: 'WithdrawRequest', + aggregateId: requestId, + eventType: 'WITHDRAW_COMPLETED', + payload: { + requestId, + txHash: request.txHash, + confirmations: txStatus.confirmations, + completedAt: updated.completedAt?.toISOString(), + }, + }); + + this.logger.log(`Withdraw completed: ${request.requestNo}`); + return updated; + } + + /** + * 绑定用户区块链地址 + */ + async bindUserAddress( + accountSequence: string, + kavaAddress: string, + ): Promise { + if (!this.kavaService.isValidAddress(kavaAddress)) { + throw new DomainException('Invalid KAVA address', 'INVALID_ADDRESS'); + } + + await this.blockchainRepo.bindAddress({ + accountSequence, + kavaAddress, + }); + + await this.outboxRepo.create({ + aggregateType: 'BlockchainAddress', + aggregateId: accountSequence, + eventType: 'ADDRESS_BOUND', + payload: { + accountSequence, + kavaAddress, + }, + }); + + this.logger.log(`Address bound for user ${accountSequence}: ${kavaAddress}`); + } + + /** + * 创建 DEX Swap 请求 + */ + async createSwapRequest( + accountSequence: string, + fromAsset: AssetType, + toAsset: AssetType, + fromAmount: Decimal, + minToAmount: Decimal, + ): Promise { + const swapNo = `SW${Date.now()}${Math.random().toString(36).substring(2, 8).toUpperCase()}`; + + // 这里应该调用 DEX 获取实际汇率 + // 简化处理:假设 1:1 汇率 + const exchangeRate = new Decimal(1); + const toAmount = fromAmount.mul(exchangeRate); + + if (toAmount.lessThan(minToAmount)) { + throw new DomainException('Slippage too high', 'SLIPPAGE_EXCEEDED'); + } + + const slippage = toAmount.minus(minToAmount).div(minToAmount).mul(100); + const fee = fromAmount.mul(new Decimal('0.003')); // 0.3% 手续费 + + const swap = await this.blockchainRepo.createSwapRecord({ + swapNo, + accountSequence, + fromAsset, + toAsset, + fromAmount, + toAmount, + exchangeRate, + slippage, + fee, + }); + + await this.outboxRepo.create({ + aggregateType: 'DexSwap', + aggregateId: swap.id, + eventType: 'SWAP_REQUEST_CREATED', + payload: { + swapId: swap.id, + swapNo, + accountSequence, + fromAsset, + toAsset, + fromAmount: fromAmount.toString(), + toAmount: toAmount.toString(), + exchangeRate: exchangeRate.toString(), + }, + }); + + return swap; + } + + async getWithdrawRequest(requestId: string): Promise { + return this.blockchainRepo.findWithdrawById(requestId); + } + + async getUserWithdrawRequests( + accountSequence: string, + options?: { status?: WithdrawStatus; limit?: number; offset?: number }, + ) { + return this.blockchainRepo.findWithdrawsByAccount(accountSequence, options); + } + + async getUserAddressBinding(accountSequence: string) { + return this.blockchainRepo.findAddressByAccount(accountSequence); + } +} diff --git a/backend/services/mining-wallet-service/src/application/services/pool-account.service.ts b/backend/services/mining-wallet-service/src/application/services/pool-account.service.ts new file mode 100644 index 00000000..152c4270 --- /dev/null +++ b/backend/services/mining-wallet-service/src/application/services/pool-account.service.ts @@ -0,0 +1,295 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { PoolAccountRepository } from '../../infrastructure/persistence/repositories/pool-account.repository'; +import { OutboxRepository } from '../../infrastructure/persistence/repositories/outbox.repository'; +import { PoolAccount, PoolAccountType, TransactionType, CounterpartyType } from '@prisma/client'; +import Decimal from 'decimal.js'; +import { DomainException } from '../../shared/filters/domain-exception.filter'; + +export interface InitializePoolsInput { + sharePool: { + name: string; + initialBalance?: Decimal; + }; + blackHolePool: { + name: string; + targetBurn: Decimal; + }; + circulationPool: { + name: string; + initialBalance?: Decimal; + }; +} + +@Injectable() +export class PoolAccountService { + private readonly logger = new Logger(PoolAccountService.name); + + constructor( + private readonly poolAccountRepo: PoolAccountRepository, + private readonly outboxRepo: OutboxRepository, + ) {} + + /** + * 初始化三大池账户 + */ + async initializePools(input: InitializePoolsInput): Promise { + const pools: PoolAccount[] = []; + + // 积分股池 + const sharePool = await this.poolAccountRepo.create({ + poolType: 'SHARE_POOL', + name: input.sharePool.name, + }); + pools.push(sharePool); + + // 如果有初始余额,注入资金 + if (input.sharePool.initialBalance?.greaterThan(0)) { + await this.poolAccountRepo.updateBalanceWithTransaction( + 'SHARE_POOL', + input.sharePool.initialBalance, + { + transactionType: 'INITIAL_INJECT', + counterpartyType: 'EXTERNAL', + memo: `初始注入, 数量${input.sharePool.initialBalance.toFixed(8)}`, + }, + ); + } + + // 黑洞积分股池 + const blackHolePool = await this.poolAccountRepo.create({ + poolType: 'BLACK_HOLE_POOL', + name: input.blackHolePool.name, + targetBurn: input.blackHolePool.targetBurn, + }); + pools.push(blackHolePool); + + // 流通积分股池 + const circulationPool = await this.poolAccountRepo.create({ + poolType: 'CIRCULATION_POOL', + name: input.circulationPool.name, + }); + pools.push(circulationPool); + + if (input.circulationPool.initialBalance?.greaterThan(0)) { + await this.poolAccountRepo.updateBalanceWithTransaction( + 'CIRCULATION_POOL', + input.circulationPool.initialBalance, + { + transactionType: 'INITIAL_INJECT', + counterpartyType: 'EXTERNAL', + memo: `初始注入, 数量${input.circulationPool.initialBalance.toFixed(8)}`, + }, + ); + } + + // 发布初始化事件 + await this.outboxRepo.create({ + aggregateType: 'PoolAccount', + aggregateId: 'ALL_POOLS', + eventType: 'POOLS_INITIALIZED', + payload: { + pools: pools.map((p) => ({ + id: p.id, + type: p.poolType, + name: p.name, + })), + sharePoolInitialBalance: input.sharePool.initialBalance?.toString() || '0', + blackHoleTargetBurn: input.blackHolePool.targetBurn.toString(), + circulationPoolInitialBalance: input.circulationPool.initialBalance?.toString() || '0', + initializedAt: new Date().toISOString(), + }, + }); + + this.logger.log('Pool accounts initialized'); + return pools; + } + + /** + * 从积分股池分配(挖矿分配) + */ + async distributeMiningReward( + toAccountSeq: string, + toUserId: string, + amount: Decimal, + miningInfo: { + miningMinute: Date; + contributionRatio: Decimal; + referenceId?: string; + }, + ): Promise { + const memo = `挖矿分配给用户[${toAccountSeq}], 算力占比${miningInfo.contributionRatio.mul(100).toFixed(4)}%, 分钟${miningInfo.miningMinute.toISOString()}`; + + await this.poolAccountRepo.updateBalanceWithTransaction( + 'SHARE_POOL', + amount.negated(), + { + transactionType: 'MINING_DISTRIBUTE', + counterpartyType: 'USER', + counterpartyAccountSeq: toAccountSeq, + counterpartyUserId: toUserId, + referenceId: miningInfo.referenceId, + referenceType: 'MINING_RECORD', + memo, + metadata: { + miningMinute: miningInfo.miningMinute.toISOString(), + contributionRatio: miningInfo.contributionRatio.toString(), + }, + }, + ); + + await this.outboxRepo.create({ + aggregateType: 'PoolAccount', + aggregateId: 'SHARE_POOL', + eventType: 'MINING_DISTRIBUTED', + payload: { + toAccountSeq, + toUserId, + amount: amount.toString(), + miningMinute: miningInfo.miningMinute.toISOString(), + contributionRatio: miningInfo.contributionRatio.toString(), + }, + }); + } + + /** + * 用户划入流通池(准备卖出) + */ + async userTransferToCirculation( + fromAccountSeq: string, + fromUserId: string, + amount: Decimal, + referenceId?: string, + ): Promise { + const memo = `用户[${fromAccountSeq}]划入流通池, 数量${amount.toFixed(8)}`; + + await this.poolAccountRepo.updateBalanceWithTransaction( + 'CIRCULATION_POOL', + amount, + { + transactionType: 'TRANSFER_IN', + counterpartyType: 'USER', + counterpartyAccountSeq: fromAccountSeq, + counterpartyUserId: fromUserId, + referenceId, + referenceType: 'TRANSFER', + memo, + }, + ); + + await this.outboxRepo.create({ + aggregateType: 'PoolAccount', + aggregateId: 'CIRCULATION_POOL', + eventType: 'USER_TRANSFER_IN', + payload: { + fromAccountSeq, + fromUserId, + amount: amount.toString(), + referenceId, + }, + }); + } + + /** + * 用户从流通池划出(买入成功) + */ + async userTransferFromCirculation( + toAccountSeq: string, + toUserId: string, + amount: Decimal, + referenceId?: string, + ): Promise { + const memo = `用户[${toAccountSeq}]从流通池划出, 数量${amount.toFixed(8)}`; + + await this.poolAccountRepo.updateBalanceWithTransaction( + 'CIRCULATION_POOL', + amount.negated(), + { + transactionType: 'TRANSFER_OUT', + counterpartyType: 'USER', + counterpartyAccountSeq: toAccountSeq, + counterpartyUserId: toUserId, + referenceId, + referenceType: 'TRANSFER', + memo, + }, + ); + + await this.outboxRepo.create({ + aggregateType: 'PoolAccount', + aggregateId: 'CIRCULATION_POOL', + eventType: 'USER_TRANSFER_OUT', + payload: { + toAccountSeq, + toUserId, + amount: amount.toString(), + referenceId, + }, + }); + } + + /** + * 销毁到黑洞池 + */ + async burnToBlackHole( + amount: Decimal, + sourceInfo: { + fromPoolType: PoolAccountType; + counterpartyAccountSeq?: string; + counterpartyUserId?: string; + referenceId?: string; + }, + ): Promise { + const result = await this.poolAccountRepo.burnToBlackHole( + sourceInfo.fromPoolType, + amount, + { + counterpartyType: sourceInfo.counterpartyAccountSeq ? 'USER' : 'POOL', + counterpartyAccountSeq: sourceInfo.counterpartyAccountSeq, + counterpartyUserId: sourceInfo.counterpartyUserId, + memo: `销毁到黑洞, 来源${sourceInfo.fromPoolType}, 数量${amount.toFixed(8)}`, + }, + ); + + await this.outboxRepo.create({ + aggregateType: 'PoolAccount', + aggregateId: 'BLACK_HOLE_POOL', + eventType: 'BURN_TO_BLACK_HOLE', + payload: { + amount: amount.toString(), + fromPoolType: sourceInfo.fromPoolType, + counterpartyAccountSeq: sourceInfo.counterpartyAccountSeq, + counterpartyUserId: sourceInfo.counterpartyUserId, + referenceId: sourceInfo.referenceId, + newBlackHoleBalance: result.blackHolePool.balance.toString(), + remainingBurn: result.blackHolePool.remainingBurn?.toString(), + }, + }); + + this.logger.log(`Burned ${amount.toFixed(8)} to black hole from ${sourceInfo.fromPoolType}`); + } + + async findByType(poolType: PoolAccountType): Promise { + return this.poolAccountRepo.findByType(poolType); + } + + async findAll(): Promise { + return this.poolAccountRepo.findAll(); + } + + async getPoolStats() { + return this.poolAccountRepo.getPoolStats(); + } + + async getTransactions( + poolType: PoolAccountType, + options?: { + transactionType?: TransactionType; + startDate?: Date; + endDate?: Date; + limit?: number; + offset?: number; + }, + ) { + return this.poolAccountRepo.getTransactions(poolType, options); + } +} diff --git a/backend/services/mining-wallet-service/src/application/services/system-account.service.ts b/backend/services/mining-wallet-service/src/application/services/system-account.service.ts new file mode 100644 index 00000000..8321dd65 --- /dev/null +++ b/backend/services/mining-wallet-service/src/application/services/system-account.service.ts @@ -0,0 +1,382 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { SystemAccountRepository } from '../../infrastructure/persistence/repositories/system-account.repository'; +import { RegionRepository } from '../../infrastructure/persistence/repositories/region.repository'; +import { OutboxRepository } from '../../infrastructure/persistence/repositories/outbox.repository'; +import { SystemAccount, SystemAccountType, AssetType, TransactionType, CounterpartyType } from '@prisma/client'; +import Decimal from 'decimal.js'; +import { DomainException } from '../../shared/filters/domain-exception.filter'; + +export interface InitializeSystemAccountsInput { + headquarters: { name: string; code: string }; + operation: { name: string; code: string }; + fee: { name: string; code: string }; + hotWallet: { name: string; code: string; blockchainAddress?: string }; + coldWallet?: { name: string; code: string; blockchainAddress?: string }; +} + +@Injectable() +export class SystemAccountService { + private readonly logger = new Logger(SystemAccountService.name); + + constructor( + private readonly systemAccountRepo: SystemAccountRepository, + private readonly regionRepo: RegionRepository, + private readonly outboxRepo: OutboxRepository, + ) {} + + /** + * 初始化核心系统账户(总部、运营、手续费、热钱包) + */ + async initializeCoreAccounts(input: InitializeSystemAccountsInput): Promise { + const accounts: SystemAccount[] = []; + + // 总部账户 + const hq = await this.systemAccountRepo.create({ + accountType: 'HEADQUARTERS', + name: input.headquarters.name, + code: input.headquarters.code, + }); + accounts.push(hq); + + // 运营账户 + const op = await this.systemAccountRepo.create({ + accountType: 'OPERATION', + name: input.operation.name, + code: input.operation.code, + }); + accounts.push(op); + + // 手续费账户 + const fee = await this.systemAccountRepo.create({ + accountType: 'FEE', + name: input.fee.name, + code: input.fee.code, + }); + accounts.push(fee); + + // 热钱包 + const hotWallet = await this.systemAccountRepo.create({ + accountType: 'HOT_WALLET', + name: input.hotWallet.name, + code: input.hotWallet.code, + blockchainAddress: input.hotWallet.blockchainAddress, + }); + accounts.push(hotWallet); + + // 冷钱包(可选) + if (input.coldWallet) { + const coldWallet = await this.systemAccountRepo.create({ + accountType: 'COLD_WALLET', + name: input.coldWallet.name, + code: input.coldWallet.code, + blockchainAddress: input.coldWallet.blockchainAddress, + }); + accounts.push(coldWallet); + } + + // 发布初始化事件 + await this.outboxRepo.create({ + aggregateType: 'SystemAccount', + aggregateId: 'CORE_ACCOUNTS', + eventType: 'CORE_ACCOUNTS_INITIALIZED', + payload: { + accounts: accounts.map((a) => ({ + id: a.id, + type: a.accountType, + code: a.code, + })), + initializedAt: new Date().toISOString(), + }, + }); + + this.logger.log(`Core system accounts initialized: ${accounts.length} accounts`); + return accounts; + } + + /** + * 创建省级公司账户 + */ + async createProvinceAccount( + provinceId: string, + name: string, + code: string, + ): Promise { + const province = await this.regionRepo.findProvinceById(provinceId); + if (!province) { + throw new DomainException(`Province not found: ${provinceId}`, 'PROVINCE_NOT_FOUND'); + } + + const account = await this.systemAccountRepo.create({ + accountType: 'PROVINCE', + name, + code, + provinceId, + }); + + await this.outboxRepo.create({ + aggregateType: 'SystemAccount', + aggregateId: account.id, + eventType: 'PROVINCE_ACCOUNT_CREATED', + payload: { + accountId: account.id, + provinceId, + provinceName: province.name, + code, + }, + }); + + this.logger.log(`Province account created: ${code} for ${province.name}`); + return account; + } + + /** + * 创建市级公司账户 + */ + async createCityAccount( + cityId: string, + name: string, + code: string, + ): Promise { + const city = await this.regionRepo.findCityById(cityId); + if (!city) { + throw new DomainException(`City not found: ${cityId}`, 'CITY_NOT_FOUND'); + } + + const account = await this.systemAccountRepo.create({ + accountType: 'CITY', + name, + code, + provinceId: city.provinceId, + cityId, + }); + + await this.outboxRepo.create({ + aggregateType: 'SystemAccount', + aggregateId: account.id, + eventType: 'CITY_ACCOUNT_CREATED', + payload: { + accountId: account.id, + cityId, + cityName: city.name, + provinceId: city.provinceId, + code, + }, + }); + + this.logger.log(`City account created: ${code} for ${city.name}`); + return account; + } + + /** + * 收取手续费(从用户到手续费账户) + */ + async collectFee( + assetType: AssetType, + amount: Decimal, + fromAccountSeq: string, + fromUserId: string, + referenceInfo: { + referenceId: string; + referenceType: string; + memo: string; + }, + ): Promise { + const feeAccounts = await this.systemAccountRepo.findByType('FEE'); + if (feeAccounts.length === 0) { + throw new DomainException('Fee account not found', 'FEE_ACCOUNT_NOT_FOUND'); + } + + const feeAccount = feeAccounts[0]; + + await this.systemAccountRepo.updateBalanceWithTransaction( + feeAccount.id, + assetType, + amount, + { + transactionType: 'FEE_COLLECT', + counterpartyType: 'USER', + counterpartyAccountSeq: fromAccountSeq, + counterpartyUserId: fromUserId, + referenceId: referenceInfo.referenceId, + referenceType: referenceInfo.referenceType, + memo: referenceInfo.memo || `收取手续费自用户[${fromAccountSeq}], 数量${amount.toFixed(8)}`, + }, + ); + + await this.outboxRepo.create({ + aggregateType: 'SystemAccount', + aggregateId: feeAccount.id, + eventType: 'FEE_COLLECTED', + payload: { + feeAccountId: feeAccount.id, + assetType, + amount: amount.toString(), + fromAccountSeq, + fromUserId, + referenceId: referenceInfo.referenceId, + }, + }); + } + + /** + * 分发手续费到各级账户 + */ + async distributeFee( + assetType: AssetType, + totalAmount: Decimal, + distribution: { + headquartersRate: Decimal; + operationRate: Decimal; + provinceId?: string; + provinceRate?: Decimal; + cityId?: string; + cityRate?: Decimal; + }, + referenceInfo: { + referenceId: string; + referenceType: string; + }, + ): Promise { + const feeAccounts = await this.systemAccountRepo.findByType('FEE'); + if (feeAccounts.length === 0) { + throw new DomainException('Fee account not found', 'FEE_ACCOUNT_NOT_FOUND'); + } + + const feeAccount = feeAccounts[0]; + + // 计算各账户分成 + const hqAmount = totalAmount.mul(distribution.headquartersRate); + const opAmount = totalAmount.mul(distribution.operationRate); + let provinceAmount = new Decimal(0); + let cityAmount = new Decimal(0); + + if (distribution.provinceRate && distribution.provinceId) { + provinceAmount = totalAmount.mul(distribution.provinceRate); + } + if (distribution.cityRate && distribution.cityId) { + cityAmount = totalAmount.mul(distribution.cityRate); + } + + const totalDistributed = hqAmount.plus(opAmount).plus(provinceAmount).plus(cityAmount); + + // 从手续费账户扣除 + await this.systemAccountRepo.updateBalanceWithTransaction( + feeAccount.id, + assetType, + totalDistributed.negated(), + { + transactionType: 'FEE_DISTRIBUTE', + counterpartyType: 'SYSTEM_ACCOUNT', + referenceId: referenceInfo.referenceId, + referenceType: referenceInfo.referenceType, + memo: `分发手续费, 总额${totalDistributed.toFixed(8)}`, + }, + ); + + // 分发到总部 + const hqAccounts = await this.systemAccountRepo.findByType('HEADQUARTERS'); + if (hqAccounts.length > 0 && hqAmount.greaterThan(0)) { + await this.systemAccountRepo.updateBalanceWithTransaction( + hqAccounts[0].id, + assetType, + hqAmount, + { + transactionType: 'FEE_DISTRIBUTE', + counterpartyType: 'SYSTEM_ACCOUNT', + counterpartySystemId: feeAccount.id, + referenceId: referenceInfo.referenceId, + referenceType: referenceInfo.referenceType, + memo: `收到手续费分成, 数量${hqAmount.toFixed(8)}`, + }, + ); + } + + // 分发到运营 + const opAccounts = await this.systemAccountRepo.findByType('OPERATION'); + if (opAccounts.length > 0 && opAmount.greaterThan(0)) { + await this.systemAccountRepo.updateBalanceWithTransaction( + opAccounts[0].id, + assetType, + opAmount, + { + transactionType: 'FEE_DISTRIBUTE', + counterpartyType: 'SYSTEM_ACCOUNT', + counterpartySystemId: feeAccount.id, + referenceId: referenceInfo.referenceId, + referenceType: referenceInfo.referenceType, + memo: `收到手续费分成, 数量${opAmount.toFixed(8)}`, + }, + ); + } + + // 分发到省级 + if (distribution.provinceId && provinceAmount.greaterThan(0)) { + const provinceAccounts = await this.systemAccountRepo.findByRegion(distribution.provinceId); + const provinceAccount = provinceAccounts.find((a) => a.accountType === 'PROVINCE'); + if (provinceAccount) { + await this.systemAccountRepo.updateBalanceWithTransaction( + provinceAccount.id, + assetType, + provinceAmount, + { + transactionType: 'FEE_DISTRIBUTE', + counterpartyType: 'SYSTEM_ACCOUNT', + counterpartySystemId: feeAccount.id, + referenceId: referenceInfo.referenceId, + referenceType: referenceInfo.referenceType, + memo: `收到手续费分成, 数量${provinceAmount.toFixed(8)}`, + }, + ); + } + } + + // 分发到市级 + if (distribution.cityId && cityAmount.greaterThan(0)) { + const cityAccounts = await this.systemAccountRepo.findByRegion(undefined, distribution.cityId); + const cityAccount = cityAccounts.find((a) => a.accountType === 'CITY'); + if (cityAccount) { + await this.systemAccountRepo.updateBalanceWithTransaction( + cityAccount.id, + assetType, + cityAmount, + { + transactionType: 'FEE_DISTRIBUTE', + counterpartyType: 'SYSTEM_ACCOUNT', + counterpartySystemId: feeAccount.id, + referenceId: referenceInfo.referenceId, + referenceType: referenceInfo.referenceType, + memo: `收到手续费分成, 数量${cityAmount.toFixed(8)}`, + }, + ); + } + } + + await this.outboxRepo.create({ + aggregateType: 'SystemAccount', + aggregateId: feeAccount.id, + eventType: 'FEE_DISTRIBUTED', + payload: { + totalAmount: totalAmount.toString(), + distribution: { + headquarters: hqAmount.toString(), + operation: opAmount.toString(), + province: provinceAmount.toString(), + city: cityAmount.toString(), + }, + referenceId: referenceInfo.referenceId, + }, + }); + } + + async findByCode(code: string): Promise { + return this.systemAccountRepo.findByCode(code); + } + + async findByType(accountType: SystemAccountType): Promise { + return this.systemAccountRepo.findByType(accountType); + } + + async findAll(): Promise { + return this.systemAccountRepo.findAll(); + } +} diff --git a/backend/services/mining-wallet-service/src/application/services/user-wallet.service.ts b/backend/services/mining-wallet-service/src/application/services/user-wallet.service.ts new file mode 100644 index 00000000..d95cf506 --- /dev/null +++ b/backend/services/mining-wallet-service/src/application/services/user-wallet.service.ts @@ -0,0 +1,350 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { UserWalletRepository } from '../../infrastructure/persistence/repositories/user-wallet.repository'; +import { RegionRepository } from '../../infrastructure/persistence/repositories/region.repository'; +import { OutboxRepository } from '../../infrastructure/persistence/repositories/outbox.repository'; +import { UserWallet, UserWalletType, AssetType, TransactionType } from '@prisma/client'; +import Decimal from 'decimal.js'; +import { DomainException } from '../../shared/filters/domain-exception.filter'; + +@Injectable() +export class UserWalletService { + private readonly logger = new Logger(UserWalletService.name); + + constructor( + private readonly userWalletRepo: UserWalletRepository, + private readonly regionRepo: RegionRepository, + private readonly outboxRepo: OutboxRepository, + ) {} + + /** + * 为新用户创建所有钱包 + */ + async createWalletsForUser(accountSequence: string): Promise { + const wallets = await this.userWalletRepo.createAllWalletsForUser(accountSequence); + + await this.outboxRepo.create({ + aggregateType: 'UserWallet', + aggregateId: accountSequence, + eventType: 'USER_WALLETS_CREATED', + payload: { + accountSequence, + walletTypes: wallets.map((w) => w.walletType), + createdAt: new Date().toISOString(), + }, + }); + + this.logger.log(`Created ${wallets.length} wallets for user ${accountSequence}`); + return wallets; + } + + /** + * 获取用户所有钱包 + */ + async getUserWallets(accountSequence: string): Promise { + return this.userWalletRepo.findByAccountSequence(accountSequence); + } + + /** + * 获取用户特定类型钱包 + */ + async getUserWallet( + accountSequence: string, + walletType: UserWalletType, + ): Promise { + return this.userWalletRepo.findByAccountAndType(accountSequence, walletType); + } + + /** + * 获取用户钱包汇总 + */ + async getUserWalletSummary(accountSequence: string) { + return this.userWalletRepo.getUserWalletSummary(accountSequence); + } + + /** + * 增加算力(来自挖矿系统同步) + */ + async addContribution( + accountSequence: string, + amount: Decimal, + source: { + referenceId: string; + referenceType: string; + memo?: string; + }, + ): Promise { + const { wallet } = await this.userWalletRepo.updateBalanceWithTransaction( + accountSequence, + 'CONTRIBUTION', + 'CONTRIBUTION', + amount, + { + transactionType: 'TRANSFER_IN', + counterpartyType: 'SYSTEM_ACCOUNT', + referenceId: source.referenceId, + referenceType: source.referenceType, + memo: source.memo || `算力增加, 数量${amount.toFixed(8)}`, + }, + ); + + await this.outboxRepo.create({ + aggregateType: 'UserWallet', + aggregateId: accountSequence, + eventType: 'CONTRIBUTION_ADDED', + payload: { + accountSequence, + amount: amount.toString(), + newBalance: wallet.balance.toString(), + referenceId: source.referenceId, + }, + }); + + return wallet; + } + + /** + * 接收挖矿奖励(积分股) + */ + async receiveMiningReward( + accountSequence: string, + amount: Decimal, + miningInfo: { + miningMinute: Date; + contributionRatio: Decimal; + referenceId?: string; + }, + ): Promise { + const memo = `挖矿奖励, 算力占比${miningInfo.contributionRatio.mul(100).toFixed(4)}%, 分钟${miningInfo.miningMinute.toISOString()}`; + + const { wallet } = await this.userWalletRepo.updateBalanceWithTransaction( + accountSequence, + 'TOKEN_STORAGE', + 'SHARE', + amount, + { + transactionType: 'MINING_REWARD', + counterpartyType: 'POOL', + counterpartyPoolType: 'SHARE_POOL', + referenceId: miningInfo.referenceId, + referenceType: 'MINING_RECORD', + memo, + metadata: { + miningMinute: miningInfo.miningMinute.toISOString(), + contributionRatio: miningInfo.contributionRatio.toString(), + }, + }, + ); + + await this.outboxRepo.create({ + aggregateType: 'UserWallet', + aggregateId: accountSequence, + eventType: 'MINING_REWARD_RECEIVED', + payload: { + accountSequence, + amount: amount.toString(), + newBalance: wallet.balance.toString(), + miningMinute: miningInfo.miningMinute.toISOString(), + contributionRatio: miningInfo.contributionRatio.toString(), + }, + }); + + return wallet; + } + + /** + * 增加绿色积分 + */ + async addGreenPoints( + accountSequence: string, + amount: Decimal, + source: { + referenceId: string; + referenceType: string; + memo?: string; + }, + ): Promise { + const { wallet } = await this.userWalletRepo.updateBalanceWithTransaction( + accountSequence, + 'GREEN_POINTS', + 'GREEN_POINT', + amount, + { + transactionType: 'TRANSFER_IN', + referenceId: source.referenceId, + referenceType: source.referenceType, + memo: source.memo || `绿色积分增加, 数量${amount.toFixed(8)}`, + }, + ); + + await this.outboxRepo.create({ + aggregateType: 'UserWallet', + aggregateId: accountSequence, + eventType: 'GREEN_POINTS_ADDED', + payload: { + accountSequence, + amount: amount.toString(), + newBalance: wallet.balance.toString(), + referenceId: source.referenceId, + }, + }); + + return wallet; + } + + /** + * 用户间转账 + */ + async transferBetweenUsers( + fromAccountSeq: string, + toAccountSeq: string, + walletType: UserWalletType, + assetType: AssetType, + amount: Decimal, + referenceInfo?: { + referenceId?: string; + referenceType?: string; + memo?: string; + }, + ): Promise<{ fromWallet: UserWallet; toWallet: UserWallet }> { + const result = await this.userWalletRepo.transferBetweenUsers( + fromAccountSeq, + toAccountSeq, + walletType, + assetType, + amount, + { + referenceId: referenceInfo?.referenceId, + referenceType: referenceInfo?.referenceType, + memo: referenceInfo?.memo, + }, + ); + + await this.outboxRepo.create({ + aggregateType: 'UserWallet', + aggregateId: fromAccountSeq, + eventType: 'USER_TRANSFER', + payload: { + fromAccountSeq, + toAccountSeq, + walletType, + assetType, + amount: amount.toString(), + referenceId: referenceInfo?.referenceId, + }, + }); + + return { + fromWallet: result.fromWallet, + toWallet: result.toWallet, + }; + } + + /** + * 冻结余额(准备提现或交易) + */ + async freezeBalance( + accountSequence: string, + walletType: UserWalletType, + assetType: AssetType, + amount: Decimal, + referenceInfo: { + referenceId: string; + referenceType: string; + memo?: string; + }, + ): Promise { + const { wallet } = await this.userWalletRepo.freezeBalance( + accountSequence, + walletType, + assetType, + amount, + true, + { + referenceId: referenceInfo.referenceId, + referenceType: referenceInfo.referenceType, + memo: referenceInfo.memo || `冻结${amount.toFixed(8)}用于${referenceInfo.referenceType}`, + }, + ); + + return wallet; + } + + /** + * 解冻余额 + */ + async unfreezeBalance( + accountSequence: string, + walletType: UserWalletType, + assetType: AssetType, + amount: Decimal, + referenceInfo: { + referenceId: string; + referenceType: string; + memo?: string; + }, + ): Promise { + const { wallet } = await this.userWalletRepo.freezeBalance( + accountSequence, + walletType, + assetType, + amount, + false, + { + referenceId: referenceInfo.referenceId, + referenceType: referenceInfo.referenceType, + memo: referenceInfo.memo || `解冻${amount.toFixed(8)}, ${referenceInfo.referenceType}取消`, + }, + ); + + return wallet; + } + + /** + * 扣减余额(如提现成功、交易扣款) + */ + async deductBalance( + accountSequence: string, + walletType: UserWalletType, + assetType: AssetType, + amount: Decimal, + transactionInput: { + transactionType: TransactionType; + counterpartyType?: 'USER' | 'SYSTEM_ACCOUNT' | 'POOL' | 'BLOCKCHAIN' | 'EXTERNAL'; + counterpartyAccountSeq?: string; + counterpartyUserId?: string; + counterpartyAddress?: string; + referenceId?: string; + referenceType?: string; + txHash?: string; + memo?: string; + }, + ): Promise { + const { wallet } = await this.userWalletRepo.updateBalanceWithTransaction( + accountSequence, + walletType, + assetType, + amount.negated(), + transactionInput, + ); + + return wallet; + } + + /** + * 获取用户交易记录 + */ + async getTransactions( + accountSequence: string, + options?: { + walletType?: UserWalletType; + transactionType?: TransactionType; + assetType?: AssetType; + startDate?: Date; + endDate?: Date; + limit?: number; + offset?: number; + }, + ) { + return this.userWalletRepo.getTransactions(accountSequence, options); + } +} diff --git a/backend/services/mining-wallet-service/src/infrastructure/blockchain/kava-blockchain.service.ts b/backend/services/mining-wallet-service/src/infrastructure/blockchain/kava-blockchain.service.ts new file mode 100644 index 00000000..fff9f520 --- /dev/null +++ b/backend/services/mining-wallet-service/src/infrastructure/blockchain/kava-blockchain.service.ts @@ -0,0 +1,271 @@ +import { Injectable, Logger, OnModuleInit } from '@nestjs/common'; +import { ConfigService } from '@nestjs/config'; +import { ethers } from 'ethers'; +import Decimal from 'decimal.js'; + +export interface TransactionResult { + txHash: string; + blockNumber: number; + gasUsed: bigint; + status: 'success' | 'failed'; +} + +export interface TokenBalance { + balance: Decimal; + decimals: number; +} + +@Injectable() +export class KavaBlockchainService implements OnModuleInit { + private readonly logger = new Logger(KavaBlockchainService.name); + private provider: ethers.JsonRpcProvider; + private hotWallet: ethers.Wallet | null = null; + private blackHoleAddress: string; + private isConnected = false; + + constructor(private readonly configService: ConfigService) { + this.blackHoleAddress = this.configService.get( + 'KAVA_BLACK_HOLE_ADDRESS', + '0x000000000000000000000000000000000000dEaD', + ); + } + + async onModuleInit() { + await this.connect(); + } + + private async connect(): Promise { + try { + const rpcUrl = this.configService.get('KAVA_RPC_URL', 'https://evm.kava.io'); + const chainId = this.configService.get('KAVA_CHAIN_ID', 2222); + + this.provider = new ethers.JsonRpcProvider(rpcUrl, chainId); + + // Test connection + const network = await this.provider.getNetwork(); + this.logger.log(`Connected to KAVA network: ${network.chainId}`); + + // Initialize hot wallet if private key is provided + const privateKey = this.configService.get('KAVA_HOT_WALLET_PRIVATE_KEY'); + if (privateKey) { + this.hotWallet = new ethers.Wallet(privateKey, this.provider); + this.logger.log(`Hot wallet initialized: ${this.hotWallet.address}`); + } else { + this.logger.warn('No hot wallet private key provided - blockchain operations limited'); + } + + this.isConnected = true; + } catch (error) { + this.logger.error('Failed to connect to KAVA blockchain', error); + this.isConnected = false; + } + } + + isReady(): boolean { + return this.isConnected && this.hotWallet !== null; + } + + getHotWalletAddress(): string | null { + return this.hotWallet?.address || null; + } + + getBlackHoleAddress(): string { + return this.blackHoleAddress; + } + + /** + * 获取账户的原生币余额 + */ + async getBalance(address: string): Promise { + const balance = await this.provider.getBalance(address); + return new Decimal(ethers.formatEther(balance)); + } + + /** + * 获取当前区块号 + */ + async getCurrentBlockNumber(): Promise { + return this.provider.getBlockNumber(); + } + + /** + * 获取交易状态 + */ + async getTransactionStatus(txHash: string): Promise<{ + confirmed: boolean; + blockNumber: number | null; + confirmations: number; + status: 'success' | 'failed' | 'pending'; + }> { + const receipt = await this.provider.getTransactionReceipt(txHash); + + if (!receipt) { + return { + confirmed: false, + blockNumber: null, + confirmations: 0, + status: 'pending', + }; + } + + const currentBlock = await this.getCurrentBlockNumber(); + const confirmations = currentBlock - receipt.blockNumber; + + return { + confirmed: confirmations >= 1, + blockNumber: receipt.blockNumber, + confirmations, + status: receipt.status === 1 ? 'success' : 'failed', + }; + } + + /** + * 发送原生币(KAVA) + */ + async sendNative( + toAddress: string, + amount: Decimal, + ): Promise { + if (!this.hotWallet) { + throw new Error('Hot wallet not initialized'); + } + + const tx = await this.hotWallet.sendTransaction({ + to: toAddress, + value: ethers.parseEther(amount.toString()), + }); + + const receipt = await tx.wait(); + + if (!receipt) { + throw new Error('Transaction failed - no receipt'); + } + + return { + txHash: receipt.hash, + blockNumber: receipt.blockNumber, + gasUsed: receipt.gasUsed, + status: receipt.status === 1 ? 'success' : 'failed', + }; + } + + /** + * 发送 ERC20 代币 + */ + async sendToken( + tokenAddress: string, + toAddress: string, + amount: Decimal, + decimals: number = 18, + ): Promise { + if (!this.hotWallet) { + throw new Error('Hot wallet not initialized'); + } + + const erc20Abi = [ + 'function transfer(address to, uint256 amount) returns (bool)', + 'function balanceOf(address account) view returns (uint256)', + 'function decimals() view returns (uint8)', + ]; + + const tokenContract = new ethers.Contract(tokenAddress, erc20Abi, this.hotWallet); + + const amountWei = ethers.parseUnits(amount.toString(), decimals); + const tx = await tokenContract.transfer(toAddress, amountWei); + + const receipt = await tx.wait(); + + if (!receipt) { + throw new Error('Transaction failed - no receipt'); + } + + return { + txHash: receipt.hash, + blockNumber: receipt.blockNumber, + gasUsed: receipt.gasUsed, + status: receipt.status === 1 ? 'success' : 'failed', + }; + } + + /** + * 销毁到黑洞地址 + */ + async burnToBlackHole( + tokenAddress: string, + amount: Decimal, + decimals: number = 18, + ): Promise { + return this.sendToken(tokenAddress, this.blackHoleAddress, amount, decimals); + } + + /** + * 获取 ERC20 代币余额 + */ + async getTokenBalance( + tokenAddress: string, + walletAddress: string, + ): Promise { + const erc20Abi = [ + 'function balanceOf(address account) view returns (uint256)', + 'function decimals() view returns (uint8)', + ]; + + const tokenContract = new ethers.Contract(tokenAddress, erc20Abi, this.provider); + + const [balance, decimals] = await Promise.all([ + tokenContract.balanceOf(walletAddress), + tokenContract.decimals(), + ]); + + return { + balance: new Decimal(ethers.formatUnits(balance, decimals)), + decimals, + }; + } + + /** + * 估算 Gas 费用 + */ + async estimateGas( + toAddress: string, + value: Decimal, + data?: string, + ): Promise<{ gasLimit: bigint; gasPrice: bigint; estimatedFee: Decimal }> { + const gasPrice = (await this.provider.getFeeData()).gasPrice || 0n; + + const gasLimit = await this.provider.estimateGas({ + to: toAddress, + value: ethers.parseEther(value.toString()), + data: data || '0x', + }); + + const estimatedFee = new Decimal(ethers.formatEther(gasLimit * gasPrice)); + + return { + gasLimit, + gasPrice, + estimatedFee, + }; + } + + /** + * 验证地址格式 + */ + isValidAddress(address: string): boolean { + return ethers.isAddress(address); + } + + /** + * 监听新区块(用于检测充值) + */ + onNewBlock(callback: (blockNumber: number) => void): void { + this.provider.on('block', callback); + } + + /** + * 停止监听 + */ + removeAllListeners(): void { + this.provider.removeAllListeners(); + } +} diff --git a/backend/services/mining-wallet-service/src/infrastructure/infrastructure.module.ts b/backend/services/mining-wallet-service/src/infrastructure/infrastructure.module.ts new file mode 100644 index 00000000..47777f50 --- /dev/null +++ b/backend/services/mining-wallet-service/src/infrastructure/infrastructure.module.ts @@ -0,0 +1,79 @@ +import { Module, Global } from '@nestjs/common'; +import { ConfigModule, ConfigService } from '@nestjs/config'; +import { ClientsModule, Transport } from '@nestjs/microservices'; +import { PrismaModule } from './persistence/prisma/prisma.module'; +import { SystemAccountRepository } from './persistence/repositories/system-account.repository'; +import { PoolAccountRepository } from './persistence/repositories/pool-account.repository'; +import { UserWalletRepository } from './persistence/repositories/user-wallet.repository'; +import { RegionRepository } from './persistence/repositories/region.repository'; +import { BlockchainRepository } from './persistence/repositories/blockchain.repository'; +import { OutboxRepository } from './persistence/repositories/outbox.repository'; +import { RedisService } from './redis/redis.service'; +import { KafkaProducerService } from './kafka/kafka-producer.service'; +import { KavaBlockchainService } from './blockchain/kava-blockchain.service'; + +@Global() +@Module({ + imports: [ + PrismaModule, + ClientsModule.registerAsync([ + { + name: 'KAFKA_CLIENT', + imports: [ConfigModule], + useFactory: (configService: ConfigService) => ({ + transport: Transport.KAFKA, + options: { + client: { + clientId: 'mining-wallet-service', + brokers: configService + .get('KAFKA_BROKERS', 'localhost:9092') + .split(','), + }, + producer: { + allowAutoTopicCreation: true, + }, + }, + }), + inject: [ConfigService], + }, + ]), + ], + providers: [ + // Repositories + SystemAccountRepository, + PoolAccountRepository, + UserWalletRepository, + RegionRepository, + BlockchainRepository, + OutboxRepository, + // Services + KafkaProducerService, + KavaBlockchainService, + { + provide: 'REDIS_OPTIONS', + useFactory: (configService: ConfigService) => ({ + host: configService.get('REDIS_HOST', 'localhost'), + port: configService.get('REDIS_PORT', 6379), + password: configService.get('REDIS_PASSWORD'), + db: configService.get('REDIS_DB', 15), + }), + inject: [ConfigService], + }, + RedisService, + ], + exports: [ + // Repositories + SystemAccountRepository, + PoolAccountRepository, + UserWalletRepository, + RegionRepository, + BlockchainRepository, + OutboxRepository, + // Services + KafkaProducerService, + KavaBlockchainService, + RedisService, + ClientsModule, + ], +}) +export class InfrastructureModule {} diff --git a/backend/services/mining-wallet-service/src/infrastructure/kafka/kafka-producer.service.ts b/backend/services/mining-wallet-service/src/infrastructure/kafka/kafka-producer.service.ts new file mode 100644 index 00000000..e64a8421 --- /dev/null +++ b/backend/services/mining-wallet-service/src/infrastructure/kafka/kafka-producer.service.ts @@ -0,0 +1,53 @@ +import { Injectable, Inject, OnModuleInit, Logger } from '@nestjs/common'; +import { ClientKafka } from '@nestjs/microservices'; +import { lastValueFrom } from 'rxjs'; + +export interface KafkaMessage { + key?: string; + value: any; + headers?: Record; +} + +@Injectable() +export class KafkaProducerService implements OnModuleInit { + private readonly logger = new Logger(KafkaProducerService.name); + + constructor( + @Inject('KAFKA_CLIENT') private readonly kafkaClient: ClientKafka, + ) {} + + async onModuleInit() { + await this.kafkaClient.connect(); + this.logger.log('Kafka producer connected'); + } + + async emit(topic: string, message: KafkaMessage): Promise { + try { + await lastValueFrom( + this.kafkaClient.emit(topic, { + key: message.key, + value: JSON.stringify(message.value), + headers: message.headers, + }), + ); + this.logger.debug(`Message emitted to topic ${topic}`); + } catch (error) { + this.logger.error(`Failed to emit message to topic ${topic}`, error); + throw error; + } + } + + async emitBatch(topic: string, messages: KafkaMessage[]): Promise { + try { + for (const message of messages) { + await this.emit(topic, message); + } + } catch (error) { + this.logger.error( + `Failed to emit batch messages to topic ${topic}`, + error, + ); + throw error; + } + } +} diff --git a/backend/services/mining-wallet-service/src/infrastructure/persistence/prisma/prisma.module.ts b/backend/services/mining-wallet-service/src/infrastructure/persistence/prisma/prisma.module.ts new file mode 100644 index 00000000..7207426f --- /dev/null +++ b/backend/services/mining-wallet-service/src/infrastructure/persistence/prisma/prisma.module.ts @@ -0,0 +1,9 @@ +import { Global, Module } from '@nestjs/common'; +import { PrismaService } from './prisma.service'; + +@Global() +@Module({ + providers: [PrismaService], + exports: [PrismaService], +}) +export class PrismaModule {} diff --git a/backend/services/mining-wallet-service/src/infrastructure/persistence/prisma/prisma.service.ts b/backend/services/mining-wallet-service/src/infrastructure/persistence/prisma/prisma.service.ts new file mode 100644 index 00000000..c0312a62 --- /dev/null +++ b/backend/services/mining-wallet-service/src/infrastructure/persistence/prisma/prisma.service.ts @@ -0,0 +1,25 @@ +import { Injectable, OnModuleInit, OnModuleDestroy, Logger } from '@nestjs/common'; +import { PrismaClient } from '@prisma/client'; + +@Injectable() +export class PrismaService extends PrismaClient implements OnModuleInit, OnModuleDestroy { + private readonly logger = new Logger(PrismaService.name); + + constructor() { + super({ + log: process.env.NODE_ENV === 'development' + ? ['query', 'info', 'warn', 'error'] + : ['error'], + }); + } + + async onModuleInit() { + await this.$connect(); + this.logger.log('Database connected'); + } + + async onModuleDestroy() { + await this.$disconnect(); + this.logger.log('Database disconnected'); + } +} diff --git a/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/blockchain.repository.ts b/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/blockchain.repository.ts new file mode 100644 index 00000000..4a434fa2 --- /dev/null +++ b/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/blockchain.repository.ts @@ -0,0 +1,399 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { PrismaService } from '../prisma/prisma.service'; +import { + WithdrawRequest, + DepositRecord, + DexSwapRecord, + BlockchainAddressBinding, + BlackHoleContract, + BurnToBlackHoleRecord, + WithdrawStatus, + AssetType, + CounterpartyType, + PoolAccountType, + Prisma, +} from '@prisma/client'; +import Decimal from 'decimal.js'; + +@Injectable() +export class BlockchainRepository { + private readonly logger = new Logger(BlockchainRepository.name); + + constructor(private readonly prisma: PrismaService) {} + + // ==================== Withdraw Requests ==================== + + async createWithdrawRequest(data: { + requestNo: string; + accountSequence: string; + assetType: AssetType; + amount: Decimal; + fee: Decimal; + toAddress: string; + }): Promise { + const netAmount = data.amount.minus(data.fee); + + return this.prisma.withdrawRequest.create({ + data: { + requestNo: data.requestNo, + accountSequence: data.accountSequence, + assetType: data.assetType, + amount: data.amount.toFixed(8), + fee: data.fee.toFixed(8), + netAmount: netAmount.toFixed(8), + toAddress: data.toAddress, + status: 'PENDING', + }, + }); + } + + async findWithdrawById(id: string): Promise { + return this.prisma.withdrawRequest.findUnique({ + where: { id }, + }); + } + + async findWithdrawByRequestNo(requestNo: string): Promise { + return this.prisma.withdrawRequest.findUnique({ + where: { requestNo }, + }); + } + + async findWithdrawsByAccount( + accountSequence: string, + options?: { + status?: WithdrawStatus; + limit?: number; + offset?: number; + }, + ): Promise<{ requests: WithdrawRequest[]; total: number }> { + const where: Prisma.WithdrawRequestWhereInput = { + accountSequence, + }; + + if (options?.status) { + where.status = options.status; + } + + const [requests, total] = await Promise.all([ + this.prisma.withdrawRequest.findMany({ + where, + orderBy: { createdAt: 'desc' }, + take: options?.limit || 50, + skip: options?.offset || 0, + }), + this.prisma.withdrawRequest.count({ where }), + ]); + + return { requests, total }; + } + + async updateWithdrawStatus( + id: string, + status: WithdrawStatus, + data?: { + txHash?: string; + blockNumber?: bigint; + confirmations?: number; + errorMessage?: string; + approvedBy?: string; + }, + ): Promise { + const updateData: Prisma.WithdrawRequestUpdateInput = { + status, + }; + + if (data?.txHash) updateData.txHash = data.txHash; + if (data?.blockNumber) updateData.blockNumber = data.blockNumber; + if (data?.confirmations) updateData.confirmations = data.confirmations; + if (data?.errorMessage) updateData.errorMessage = data.errorMessage; + if (data?.approvedBy) { + updateData.approvedBy = data.approvedBy; + updateData.approvedAt = new Date(); + } + if (status === 'COMPLETED') { + updateData.completedAt = new Date(); + } + + return this.prisma.withdrawRequest.update({ + where: { id }, + data: updateData, + }); + } + + // ==================== Deposit Records ==================== + + async createDepositRecord(data: { + txHash: string; + fromAddress: string; + toAddress: string; + assetType: AssetType; + amount: Decimal; + blockNumber: bigint; + confirmations?: number; + matchedAccountSeq?: string; + }): Promise { + return this.prisma.depositRecord.create({ + data: { + txHash: data.txHash, + fromAddress: data.fromAddress, + toAddress: data.toAddress, + assetType: data.assetType, + amount: data.amount.toFixed(8), + blockNumber: data.blockNumber, + confirmations: data.confirmations || 0, + matchedAccountSeq: data.matchedAccountSeq, + }, + }); + } + + async findDepositByTxHash(txHash: string): Promise { + return this.prisma.depositRecord.findUnique({ + where: { txHash }, + }); + } + + async findUnprocessedDeposits(limit: number = 100): Promise { + return this.prisma.depositRecord.findMany({ + where: { + isProcessed: false, + matchedAccountSeq: { not: null }, + }, + orderBy: { createdAt: 'asc' }, + take: limit, + }); + } + + async markDepositProcessed(id: string): Promise { + return this.prisma.depositRecord.update({ + where: { id }, + data: { + isProcessed: true, + processedAt: new Date(), + }, + }); + } + + async updateDepositConfirmations(txHash: string, confirmations: number): Promise { + return this.prisma.depositRecord.update({ + where: { txHash }, + data: { confirmations }, + }); + } + + // ==================== DEX Swap Records ==================== + + async createSwapRecord(data: { + swapNo: string; + accountSequence: string; + fromAsset: AssetType; + toAsset: AssetType; + fromAmount: Decimal; + toAmount: Decimal; + exchangeRate: Decimal; + slippage?: Decimal; + fee?: Decimal; + }): Promise { + return this.prisma.dexSwapRecord.create({ + data: { + swapNo: data.swapNo, + accountSequence: data.accountSequence, + fromAsset: data.fromAsset, + toAsset: data.toAsset, + fromAmount: data.fromAmount.toFixed(8), + toAmount: data.toAmount.toFixed(8), + exchangeRate: data.exchangeRate.toFixed(18), + slippage: data.slippage?.toFixed(4) || '0', + fee: data.fee?.toFixed(8) || '0', + status: 'PENDING', + }, + }); + } + + async findSwapById(id: string): Promise { + return this.prisma.dexSwapRecord.findUnique({ + where: { id }, + }); + } + + async findSwapBySwapNo(swapNo: string): Promise { + return this.prisma.dexSwapRecord.findUnique({ + where: { swapNo }, + }); + } + + async updateSwapStatus( + id: string, + status: WithdrawStatus, + data?: { + txHash?: string; + blockNumber?: bigint; + errorMessage?: string; + }, + ): Promise { + const updateData: Prisma.DexSwapRecordUpdateInput = { + status, + }; + + if (data?.txHash) updateData.txHash = data.txHash; + if (data?.blockNumber) updateData.blockNumber = data.blockNumber; + if (data?.errorMessage) updateData.errorMessage = data.errorMessage; + if (status === 'COMPLETED') { + updateData.completedAt = new Date(); + } + + return this.prisma.dexSwapRecord.update({ + where: { id }, + data: updateData, + }); + } + + // ==================== Address Binding ==================== + + async bindAddress(data: { + accountSequence: string; + kavaAddress: string; + }): Promise { + return this.prisma.blockchainAddressBinding.upsert({ + where: { accountSequence: data.accountSequence }, + create: { + accountSequence: data.accountSequence, + kavaAddress: data.kavaAddress, + }, + update: { + kavaAddress: data.kavaAddress, + isVerified: false, + verifiedAt: null, + verificationTxHash: null, + }, + }); + } + + async findAddressByAccount(accountSequence: string): Promise { + return this.prisma.blockchainAddressBinding.findUnique({ + where: { accountSequence }, + }); + } + + async findAccountByAddress(kavaAddress: string): Promise { + return this.prisma.blockchainAddressBinding.findUnique({ + where: { kavaAddress }, + }); + } + + async verifyAddress( + accountSequence: string, + verificationTxHash: string, + ): Promise { + return this.prisma.blockchainAddressBinding.update({ + where: { accountSequence }, + data: { + isVerified: true, + verifiedAt: new Date(), + verificationTxHash, + }, + }); + } + + // ==================== Black Hole Contract ==================== + + async createBlackHoleContract(data: { + contractAddress: string; + name: string; + targetBurn: Decimal; + }): Promise { + return this.prisma.blackHoleContract.create({ + data: { + contractAddress: data.contractAddress, + name: data.name, + targetBurn: data.targetBurn.toFixed(8), + remainingBurn: data.targetBurn.toFixed(8), + }, + }); + } + + async findBlackHoleContract(contractAddress: string): Promise { + return this.prisma.blackHoleContract.findUnique({ + where: { contractAddress }, + }); + } + + async getActiveBlackHoleContract(): Promise { + return this.prisma.blackHoleContract.findFirst({ + where: { isActive: true }, + }); + } + + async recordBurnToBlackHole(data: { + blackHoleId: string; + amount: Decimal; + sourceType: CounterpartyType; + sourceAccountSeq?: string; + sourceUserId?: string; + sourcePoolType?: PoolAccountType; + txHash?: string; + blockNumber?: bigint; + memo?: string; + }): Promise { + return this.prisma.$transaction(async (tx) => { + // 更新黑洞合约统计 + const blackHole = await tx.blackHoleContract.findUnique({ + where: { id: data.blackHoleId }, + }); + + if (!blackHole) { + throw new Error(`Black hole contract not found: ${data.blackHoleId}`); + } + + const newTotalBurned = new Decimal(blackHole.totalBurned.toString()).plus(data.amount); + const newRemainingBurn = new Decimal(blackHole.remainingBurn.toString()).minus(data.amount); + + await tx.blackHoleContract.update({ + where: { id: data.blackHoleId }, + data: { + totalBurned: newTotalBurned.toFixed(8), + remainingBurn: newRemainingBurn.greaterThan(0) ? newRemainingBurn.toFixed(8) : '0', + }, + }); + + // 创建销毁记录 + return tx.burnToBlackHoleRecord.create({ + data: { + blackHoleId: data.blackHoleId, + amount: data.amount.toFixed(8), + sourceType: data.sourceType, + sourceAccountSeq: data.sourceAccountSeq, + sourceUserId: data.sourceUserId, + sourcePoolType: data.sourcePoolType, + txHash: data.txHash, + blockNumber: data.blockNumber, + memo: data.memo, + }, + }); + }); + } + + async getBurnRecords( + blackHoleId: string, + options?: { + limit?: number; + offset?: number; + }, + ): Promise<{ records: BurnToBlackHoleRecord[]; total: number }> { + const where: Prisma.BurnToBlackHoleRecordWhereInput = { + blackHoleId, + }; + + const [records, total] = await Promise.all([ + this.prisma.burnToBlackHoleRecord.findMany({ + where, + orderBy: { createdAt: 'desc' }, + take: options?.limit || 50, + skip: options?.offset || 0, + }), + this.prisma.burnToBlackHoleRecord.count({ where }), + ]); + + return { records, total }; + } +} diff --git a/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/outbox.repository.ts b/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/outbox.repository.ts new file mode 100644 index 00000000..31d8be96 --- /dev/null +++ b/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/outbox.repository.ts @@ -0,0 +1,121 @@ +import { Injectable } from '@nestjs/common'; +import { PrismaService } from '../prisma/prisma.service'; +import { OutboxEvent, OutboxStatus } from '@prisma/client'; + +@Injectable() +export class OutboxRepository { + constructor(private readonly prisma: PrismaService) {} + + /** + * 创建 Outbox 事件 + */ + async create(data: { + aggregateType: string; + aggregateId: string; + eventType: string; + payload: any; + topic?: string; + key?: string; + }): Promise { + return this.prisma.outboxEvent.create({ + data: { + aggregateType: data.aggregateType, + aggregateId: data.aggregateId, + eventType: data.eventType, + payload: data.payload, + topic: data.topic || `mining-wallet.${data.eventType}`, + key: data.key || data.aggregateId, + status: OutboxStatus.PENDING, + }, + }); + } + + /** + * 获取待处理的事件(PENDING 状态且到达重试时间) + */ + async findPendingEvents(limit: number = 100): Promise { + return this.prisma.outboxEvent.findMany({ + where: { + status: OutboxStatus.PENDING, + OR: [{ nextRetryAt: null }, { nextRetryAt: { lte: new Date() } }], + }, + orderBy: { createdAt: 'asc' }, + take: limit, + }); + } + + /** + * 标记事件为已发布 + */ + async markAsPublished(id: string): Promise { + await this.prisma.outboxEvent.update({ + where: { id }, + data: { + status: OutboxStatus.PUBLISHED, + publishedAt: new Date(), + }, + }); + } + + /** + * 标记事件发布失败,计算下次重试时间(指数退避,最大3小时) + */ + async markAsFailed( + id: string, + error: string, + currentRetryCount: number, + maxRetries: number, + ): Promise { + const newRetryCount = currentRetryCount + 1; + const shouldFail = newRetryCount >= maxRetries; + + // 指数退避: 30s, 60s, 120s, 240s, 480s, 960s, 1920s, 3840s, 7680s, 10800s (最大3小时) + const baseDelayMs = 30000; // 30 seconds + const maxDelayMs = 3 * 60 * 60 * 1000; // 3 hours + const delayMs = Math.min( + baseDelayMs * Math.pow(2, newRetryCount - 1), + maxDelayMs, + ); + + await this.prisma.outboxEvent.update({ + where: { id }, + data: { + retryCount: newRetryCount, + lastError: error, + status: shouldFail ? OutboxStatus.FAILED : OutboxStatus.PENDING, + nextRetryAt: shouldFail ? null : new Date(Date.now() + delayMs), + }, + }); + } + + /** + * 删除已发布的旧事件 + */ + async deletePublished(before: Date): Promise { + const result = await this.prisma.outboxEvent.deleteMany({ + where: { + status: OutboxStatus.PUBLISHED, + publishedAt: { lt: before }, + }, + }); + return result.count; + } + + /** + * 重置长时间失败的事件(可选:给予重试机会) + */ + async resetFailedEvents(olderThan: Date): Promise { + const result = await this.prisma.outboxEvent.updateMany({ + where: { + status: OutboxStatus.FAILED, + createdAt: { gt: olderThan }, + }, + data: { + status: OutboxStatus.PENDING, + retryCount: 0, + nextRetryAt: new Date(), + }, + }); + return result.count; + } +} diff --git a/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/pool-account.repository.ts b/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/pool-account.repository.ts new file mode 100644 index 00000000..9fec58eb --- /dev/null +++ b/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/pool-account.repository.ts @@ -0,0 +1,318 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { PrismaService } from '../prisma/prisma.service'; +import { + PoolAccount, + PoolAccountTransaction, + PoolAccountType, + TransactionType, + CounterpartyType, + Prisma, +} from '@prisma/client'; +import Decimal from 'decimal.js'; + +export interface CreatePoolAccountInput { + poolType: PoolAccountType; + name: string; + targetBurn?: Decimal; + description?: string; +} + +export interface PoolTransactionInput { + poolAccountId: string; + poolType: PoolAccountType; + transactionType: TransactionType; + amount: Decimal; + balanceBefore: Decimal; + balanceAfter: Decimal; + counterpartyType?: CounterpartyType; + counterpartyAccountSeq?: string; + counterpartyUserId?: string; + counterpartySystemId?: string; + counterpartyPoolType?: PoolAccountType; + counterpartyAddress?: string; + referenceId?: string; + referenceType?: string; + txHash?: string; + memo?: string; + metadata?: any; +} + +@Injectable() +export class PoolAccountRepository { + private readonly logger = new Logger(PoolAccountRepository.name); + + constructor(private readonly prisma: PrismaService) {} + + async create(input: CreatePoolAccountInput): Promise { + return this.prisma.poolAccount.create({ + data: { + poolType: input.poolType, + name: input.name, + targetBurn: input.targetBurn?.toFixed(8), + remainingBurn: input.targetBurn?.toFixed(8), + description: input.description, + }, + }); + } + + async findByType(poolType: PoolAccountType): Promise { + return this.prisma.poolAccount.findUnique({ + where: { poolType }, + }); + } + + async findById(id: string): Promise { + return this.prisma.poolAccount.findUnique({ + where: { id }, + }); + } + + async findAll(): Promise { + return this.prisma.poolAccount.findMany({ + orderBy: { createdAt: 'asc' }, + }); + } + + /** + * 更新池余额并记录交易(原子操作) + */ + async updateBalanceWithTransaction( + poolType: PoolAccountType, + amount: Decimal, + transactionInput: Omit, + ): Promise<{ pool: PoolAccount; transaction: PoolAccountTransaction }> { + return this.prisma.$transaction(async (tx) => { + const pool = await tx.poolAccount.findUnique({ + where: { poolType }, + }); + + if (!pool) { + throw new Error(`Pool account not found: ${poolType}`); + } + + const currentBalance = new Decimal(pool.balance.toString()); + const newBalance = currentBalance.plus(amount); + + if (newBalance.lessThan(0)) { + throw new Error(`Insufficient pool balance: ${currentBalance} + ${amount} < 0`); + } + + // 更新池余额 + const updatedPool = await tx.poolAccount.update({ + where: { poolType }, + data: { + balance: newBalance.toFixed(8), + totalInflow: amount.greaterThan(0) + ? new Decimal(pool.totalInflow.toString()).plus(amount).toFixed(8) + : pool.totalInflow, + totalOutflow: amount.lessThan(0) + ? new Decimal(pool.totalOutflow.toString()).plus(amount.abs()).toFixed(8) + : pool.totalOutflow, + }, + }); + + // 记录交易 + const transaction = await tx.poolAccountTransaction.create({ + data: { + poolAccountId: pool.id, + poolType, + transactionType: transactionInput.transactionType, + amount: amount.toFixed(8), + balanceBefore: currentBalance.toFixed(8), + balanceAfter: newBalance.toFixed(8), + counterpartyType: transactionInput.counterpartyType, + counterpartyAccountSeq: transactionInput.counterpartyAccountSeq, + counterpartyUserId: transactionInput.counterpartyUserId, + counterpartySystemId: transactionInput.counterpartySystemId, + counterpartyPoolType: transactionInput.counterpartyPoolType, + counterpartyAddress: transactionInput.counterpartyAddress, + referenceId: transactionInput.referenceId, + referenceType: transactionInput.referenceType, + txHash: transactionInput.txHash, + memo: transactionInput.memo, + metadata: transactionInput.metadata, + }, + }); + + return { pool: updatedPool, transaction }; + }); + } + + /** + * 销毁到黑洞池(特殊操作) + */ + async burnToBlackHole( + fromPoolType: PoolAccountType, + amount: Decimal, + sourceInfo: { + counterpartyType?: CounterpartyType; + counterpartyAccountSeq?: string; + counterpartyUserId?: string; + memo?: string; + }, + ): Promise<{ + fromPool: PoolAccount; + blackHolePool: PoolAccount; + fromTransaction: PoolAccountTransaction; + blackHoleTransaction: PoolAccountTransaction; + }> { + return this.prisma.$transaction(async (tx) => { + // 从源池扣除 + const fromPool = await tx.poolAccount.findUnique({ + where: { poolType: fromPoolType }, + }); + + if (!fromPool) { + throw new Error(`Source pool not found: ${fromPoolType}`); + } + + const fromBalance = new Decimal(fromPool.balance.toString()); + if (fromBalance.lessThan(amount)) { + throw new Error(`Insufficient balance in ${fromPoolType}: ${fromBalance} < ${amount}`); + } + + const newFromBalance = fromBalance.minus(amount); + + const updatedFromPool = await tx.poolAccount.update({ + where: { poolType: fromPoolType }, + data: { + balance: newFromBalance.toFixed(8), + totalOutflow: new Decimal(fromPool.totalOutflow.toString()).plus(amount).toFixed(8), + }, + }); + + // 黑洞池增加 + const blackHolePool = await tx.poolAccount.findUnique({ + where: { poolType: 'BLACK_HOLE_POOL' }, + }); + + if (!blackHolePool) { + throw new Error('Black hole pool not found'); + } + + const blackHoleBalance = new Decimal(blackHolePool.balance.toString()); + const newBlackHoleBalance = blackHoleBalance.plus(amount); + const newRemainingBurn = blackHolePool.remainingBurn + ? new Decimal(blackHolePool.remainingBurn.toString()).minus(amount) + : null; + + const updatedBlackHolePool = await tx.poolAccount.update({ + where: { poolType: 'BLACK_HOLE_POOL' }, + data: { + balance: newBlackHoleBalance.toFixed(8), + totalInflow: new Decimal(blackHolePool.totalInflow.toString()).plus(amount).toFixed(8), + remainingBurn: newRemainingBurn?.toFixed(8), + }, + }); + + // 记录源池交易 + const fromTransaction = await tx.poolAccountTransaction.create({ + data: { + poolAccountId: fromPool.id, + poolType: fromPoolType, + transactionType: 'BURN', + amount: amount.negated().toFixed(8), + balanceBefore: fromBalance.toFixed(8), + balanceAfter: newFromBalance.toFixed(8), + counterpartyType: 'POOL', + counterpartyPoolType: 'BLACK_HOLE_POOL', + counterpartyAccountSeq: sourceInfo.counterpartyAccountSeq, + counterpartyUserId: sourceInfo.counterpartyUserId, + memo: sourceInfo.memo || `销毁到黑洞池, 数量${amount.toFixed(8)}`, + }, + }); + + // 记录黑洞池交易 + const blackHoleTransaction = await tx.poolAccountTransaction.create({ + data: { + poolAccountId: blackHolePool.id, + poolType: 'BLACK_HOLE_POOL', + transactionType: 'BURN', + amount: amount.toFixed(8), + balanceBefore: blackHoleBalance.toFixed(8), + balanceAfter: newBlackHoleBalance.toFixed(8), + counterpartyType: sourceInfo.counterpartyType || 'POOL', + counterpartyPoolType: fromPoolType, + counterpartyAccountSeq: sourceInfo.counterpartyAccountSeq, + counterpartyUserId: sourceInfo.counterpartyUserId, + memo: sourceInfo.memo || `从${fromPoolType}接收销毁, 数量${amount.toFixed(8)}`, + }, + }); + + return { + fromPool: updatedFromPool, + blackHolePool: updatedBlackHolePool, + fromTransaction, + blackHoleTransaction, + }; + }); + } + + async getTransactions( + poolType: PoolAccountType, + options?: { + transactionType?: TransactionType; + startDate?: Date; + endDate?: Date; + limit?: number; + offset?: number; + }, + ): Promise<{ transactions: PoolAccountTransaction[]; total: number }> { + const where: Prisma.PoolAccountTransactionWhereInput = { + poolType, + }; + + if (options?.transactionType) { + where.transactionType = options.transactionType; + } + if (options?.startDate || options?.endDate) { + where.createdAt = {}; + if (options.startDate) where.createdAt.gte = options.startDate; + if (options.endDate) where.createdAt.lte = options.endDate; + } + + const [transactions, total] = await Promise.all([ + this.prisma.poolAccountTransaction.findMany({ + where, + orderBy: { createdAt: 'desc' }, + take: options?.limit || 50, + skip: options?.offset || 0, + }), + this.prisma.poolAccountTransaction.count({ where }), + ]); + + return { transactions, total }; + } + + /** + * 获取所有池的统计信息 + */ + async getPoolStats(): Promise<{ + poolType: PoolAccountType; + balance: string; + totalInflow: string; + totalOutflow: string; + targetBurn?: string; + remainingBurn?: string; + }[]> { + const pools = await this.prisma.poolAccount.findMany({ + select: { + poolType: true, + balance: true, + totalInflow: true, + totalOutflow: true, + targetBurn: true, + remainingBurn: true, + }, + }); + + return pools.map((p) => ({ + poolType: p.poolType, + balance: p.balance.toString(), + totalInflow: p.totalInflow.toString(), + totalOutflow: p.totalOutflow.toString(), + targetBurn: p.targetBurn?.toString(), + remainingBurn: p.remainingBurn?.toString(), + })); + } +} diff --git a/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/region.repository.ts b/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/region.repository.ts new file mode 100644 index 00000000..a67c863e --- /dev/null +++ b/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/region.repository.ts @@ -0,0 +1,297 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { PrismaService } from '../prisma/prisma.service'; +import { Province, City, UserRegionMapping, Prisma } from '@prisma/client'; + +export interface CreateProvinceInput { + code: string; + name: string; +} + +export interface CreateCityInput { + provinceId: string; + code: string; + name: string; +} + +export interface AssignUserRegionInput { + accountSequence: string; + cityId: string; + assignedBy?: string; +} + +@Injectable() +export class RegionRepository { + private readonly logger = new Logger(RegionRepository.name); + + constructor(private readonly prisma: PrismaService) {} + + // ==================== Province Operations ==================== + + async createProvince(input: CreateProvinceInput): Promise { + return this.prisma.province.create({ + data: { + code: input.code, + name: input.name, + }, + }); + } + + async findProvinceById(id: string): Promise { + return this.prisma.province.findUnique({ + where: { id }, + include: { cities: true }, + }); + } + + async findProvinceByCode(code: string): Promise { + return this.prisma.province.findUnique({ + where: { code }, + include: { cities: true }, + }); + } + + async findAllProvinces(): Promise { + return this.prisma.province.findMany({ + where: { status: 'ACTIVE' }, + include: { cities: true }, + orderBy: { code: 'asc' }, + }); + } + + async updateProvinceStatus(id: string, status: string): Promise { + return this.prisma.province.update({ + where: { id }, + data: { status }, + }); + } + + // ==================== City Operations ==================== + + async createCity(input: CreateCityInput): Promise { + return this.prisma.city.create({ + data: { + provinceId: input.provinceId, + code: input.code, + name: input.name, + }, + include: { province: true }, + }); + } + + async findCityById(id: string): Promise { + return this.prisma.city.findUnique({ + where: { id }, + include: { province: true }, + }); + } + + async findCityByCode(code: string): Promise { + return this.prisma.city.findUnique({ + where: { code }, + include: { province: true }, + }); + } + + async findCitiesByProvince(provinceId: string): Promise { + return this.prisma.city.findMany({ + where: { + provinceId, + status: 'ACTIVE', + }, + include: { province: true }, + orderBy: { code: 'asc' }, + }); + } + + async findAllCities(): Promise { + return this.prisma.city.findMany({ + where: { status: 'ACTIVE' }, + include: { province: true }, + orderBy: [{ province: { code: 'asc' } }, { code: 'asc' }], + }); + } + + async updateCityStatus(id: string, status: string): Promise { + return this.prisma.city.update({ + where: { id }, + data: { status }, + }); + } + + // ==================== User Region Mapping Operations ==================== + + async assignUserToRegion(input: AssignUserRegionInput): Promise { + return this.prisma.userRegionMapping.upsert({ + where: { + accountSequence: input.accountSequence, + }, + create: { + accountSequence: input.accountSequence, + cityId: input.cityId, + assignedBy: input.assignedBy, + }, + update: { + cityId: input.cityId, + assignedBy: input.assignedBy, + assignedAt: new Date(), + }, + include: { + city: { + include: { province: true }, + }, + }, + }); + } + + async findUserRegion(accountSequence: string): Promise { + return this.prisma.userRegionMapping.findUnique({ + where: { accountSequence }, + include: { + city: { + include: { province: true }, + }, + }, + }); + } + + async findUsersByCity(cityId: string, options?: { + limit?: number; + offset?: number; + }): Promise<{ mappings: UserRegionMapping[]; total: number }> { + const [mappings, total] = await Promise.all([ + this.prisma.userRegionMapping.findMany({ + where: { cityId }, + include: { + city: { + include: { province: true }, + }, + }, + take: options?.limit || 50, + skip: options?.offset || 0, + orderBy: { assignedAt: 'desc' }, + }), + this.prisma.userRegionMapping.count({ where: { cityId } }), + ]); + + return { mappings, total }; + } + + async findUsersByProvince(provinceId: string, options?: { + limit?: number; + offset?: number; + }): Promise<{ mappings: UserRegionMapping[]; total: number }> { + const where: Prisma.UserRegionMappingWhereInput = { + city: { + provinceId, + }, + }; + + const [mappings, total] = await Promise.all([ + this.prisma.userRegionMapping.findMany({ + where, + include: { + city: { + include: { province: true }, + }, + }, + take: options?.limit || 50, + skip: options?.offset || 0, + orderBy: { assignedAt: 'desc' }, + }), + this.prisma.userRegionMapping.count({ where }), + ]); + + return { mappings, total }; + } + + async removeUserFromRegion(accountSequence: string): Promise { + await this.prisma.userRegionMapping.delete({ + where: { accountSequence }, + }).catch(() => { + // Ignore if not found + }); + } + + // ==================== Statistics ==================== + + async getRegionStats(): Promise<{ + provinces: { + id: string; + code: string; + name: string; + cityCount: number; + userCount: number; + }[]; + totalProvinces: number; + totalCities: number; + totalMappedUsers: number; + }> { + const provinces = await this.prisma.province.findMany({ + where: { status: 'ACTIVE' }, + include: { + cities: { + where: { status: 'ACTIVE' }, + include: { + _count: { + select: { userMappings: true }, + }, + }, + }, + }, + }); + + const provinceStats = provinces.map((p) => ({ + id: p.id, + code: p.code, + name: p.name, + cityCount: p.cities.length, + userCount: p.cities.reduce((sum, c) => sum + c._count.userMappings, 0), + })); + + const totalCities = await this.prisma.city.count({ where: { status: 'ACTIVE' } }); + const totalMappedUsers = await this.prisma.userRegionMapping.count(); + + return { + provinces: provinceStats, + totalProvinces: provinces.length, + totalCities, + totalMappedUsers, + }; + } + + /** + * 批量初始化区域数据 + */ + async bulkInitializeRegions( + regions: { provinceCode: string; provinceName: string; cities: { code: string; name: string }[] }[], + ): Promise { + await this.prisma.$transaction(async (tx) => { + for (const region of regions) { + const province = await tx.province.upsert({ + where: { code: region.provinceCode }, + create: { + code: region.provinceCode, + name: region.provinceName, + }, + update: { + name: region.provinceName, + }, + }); + + for (const city of region.cities) { + await tx.city.upsert({ + where: { code: city.code }, + create: { + provinceId: province.id, + code: city.code, + name: city.name, + }, + update: { + name: city.name, + }, + }); + } + } + }); + } +} diff --git a/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/system-account.repository.ts b/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/system-account.repository.ts new file mode 100644 index 00000000..0fda3639 --- /dev/null +++ b/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/system-account.repository.ts @@ -0,0 +1,329 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { PrismaService } from '../prisma/prisma.service'; +import { + SystemAccount, + SystemAccountTransaction, + SystemAccountType, + TransactionType, + AssetType, + CounterpartyType, + PoolAccountType, + Prisma, +} from '@prisma/client'; +import Decimal from 'decimal.js'; + +export interface CreateSystemAccountInput { + accountType: SystemAccountType; + name: string; + code: string; + provinceId?: string; + cityId?: string; + blockchainAddress?: string; + description?: string; +} + +export interface SystemAccountTransactionInput { + systemAccountId: string; + transactionType: TransactionType; + assetType: AssetType; + amount: Decimal; + balanceBefore: Decimal; + balanceAfter: Decimal; + counterpartyType?: CounterpartyType; + counterpartyAccountSeq?: string; + counterpartyUserId?: string; + counterpartySystemId?: string; + counterpartyPoolType?: PoolAccountType; + counterpartyAddress?: string; + referenceId?: string; + referenceType?: string; + txHash?: string; + memo?: string; + metadata?: any; +} + +@Injectable() +export class SystemAccountRepository { + private readonly logger = new Logger(SystemAccountRepository.name); + + constructor(private readonly prisma: PrismaService) {} + + async create(input: CreateSystemAccountInput): Promise { + return this.prisma.systemAccount.create({ + data: { + accountType: input.accountType, + name: input.name, + code: input.code, + provinceId: input.provinceId, + cityId: input.cityId, + blockchainAddress: input.blockchainAddress, + description: input.description, + }, + }); + } + + async findById(id: string): Promise { + return this.prisma.systemAccount.findUnique({ + where: { id }, + include: { + province: true, + city: true, + }, + }); + } + + async findByCode(code: string): Promise { + return this.prisma.systemAccount.findUnique({ + where: { code }, + }); + } + + async findByType(accountType: SystemAccountType): Promise { + return this.prisma.systemAccount.findMany({ + where: { accountType }, + include: { + province: true, + city: true, + }, + }); + } + + async findAll(): Promise { + return this.prisma.systemAccount.findMany({ + include: { + province: true, + city: true, + }, + orderBy: { createdAt: 'asc' }, + }); + } + + async findByRegion(provinceId?: string, cityId?: string): Promise { + const where: Prisma.SystemAccountWhereInput = {}; + if (provinceId) where.provinceId = provinceId; + if (cityId) where.cityId = cityId; + + return this.prisma.systemAccount.findMany({ + where, + include: { + province: true, + city: true, + }, + }); + } + + /** + * 更新余额并记录交易(原子操作) + */ + async updateBalanceWithTransaction( + systemAccountId: string, + assetType: AssetType, + amount: Decimal, + transactionInput: Omit, + ): Promise<{ account: SystemAccount; transaction: SystemAccountTransaction }> { + return this.prisma.$transaction(async (tx) => { + // 获取当前账户(加锁) + const account = await tx.systemAccount.findUnique({ + where: { id: systemAccountId }, + }); + + if (!account) { + throw new Error(`System account not found: ${systemAccountId}`); + } + + // 根据资产类型获取当前余额 + let currentBalance: Decimal; + let updateField: string; + + switch (assetType) { + case 'SHARE': + currentBalance = new Decimal(account.shareBalance.toString()); + updateField = 'shareBalance'; + break; + case 'USDT': + currentBalance = new Decimal(account.usdtBalance.toString()); + updateField = 'usdtBalance'; + break; + case 'GREEN_POINT': + currentBalance = new Decimal(account.greenPointBalance.toString()); + updateField = 'greenPointBalance'; + break; + default: + throw new Error(`Unsupported asset type: ${assetType}`); + } + + const newBalance = currentBalance.plus(amount); + + if (newBalance.lessThan(0)) { + throw new Error(`Insufficient balance for ${assetType}: ${currentBalance} + ${amount} < 0`); + } + + // 更新余额 + const updatedAccount = await tx.systemAccount.update({ + where: { id: systemAccountId }, + data: { + [updateField]: newBalance.toFixed(8), + totalInflow: amount.greaterThan(0) + ? new Decimal(account.totalInflow.toString()).plus(amount).toFixed(8) + : account.totalInflow, + totalOutflow: amount.lessThan(0) + ? new Decimal(account.totalOutflow.toString()).plus(amount.abs()).toFixed(8) + : account.totalOutflow, + }, + }); + + // 记录交易 + const transaction = await tx.systemAccountTransaction.create({ + data: { + systemAccountId, + transactionType: transactionInput.transactionType, + assetType, + amount: amount.toFixed(8), + balanceBefore: currentBalance.toFixed(8), + balanceAfter: newBalance.toFixed(8), + counterpartyType: transactionInput.counterpartyType, + counterpartyAccountSeq: transactionInput.counterpartyAccountSeq, + counterpartyUserId: transactionInput.counterpartyUserId, + counterpartySystemId: transactionInput.counterpartySystemId, + counterpartyPoolType: transactionInput.counterpartyPoolType, + counterpartyAddress: transactionInput.counterpartyAddress, + referenceId: transactionInput.referenceId, + referenceType: transactionInput.referenceType, + txHash: transactionInput.txHash, + memo: transactionInput.memo, + metadata: transactionInput.metadata, + }, + }); + + return { account: updatedAccount, transaction }; + }); + } + + /** + * 冻结/解冻余额 + */ + async freezeBalance( + systemAccountId: string, + assetType: AssetType, + amount: Decimal, + freeze: boolean, + transactionInput: Omit, + ): Promise<{ account: SystemAccount; transaction: SystemAccountTransaction }> { + return this.prisma.$transaction(async (tx) => { + const account = await tx.systemAccount.findUnique({ + where: { id: systemAccountId }, + }); + + if (!account) { + throw new Error(`System account not found: ${systemAccountId}`); + } + + let currentBalance: Decimal; + let currentFrozen: Decimal; + let balanceField: string; + let frozenField: string; + + switch (assetType) { + case 'SHARE': + currentBalance = new Decimal(account.shareBalance.toString()); + currentFrozen = new Decimal(account.frozenShare.toString()); + balanceField = 'shareBalance'; + frozenField = 'frozenShare'; + break; + case 'USDT': + currentBalance = new Decimal(account.usdtBalance.toString()); + currentFrozen = new Decimal(account.frozenUsdt.toString()); + balanceField = 'usdtBalance'; + frozenField = 'frozenUsdt'; + break; + default: + throw new Error(`Freeze not supported for asset type: ${assetType}`); + } + + let newBalance: Decimal; + let newFrozen: Decimal; + + if (freeze) { + if (currentBalance.lessThan(amount)) { + throw new Error(`Insufficient balance to freeze: ${currentBalance} < ${amount}`); + } + newBalance = currentBalance.minus(amount); + newFrozen = currentFrozen.plus(amount); + } else { + if (currentFrozen.lessThan(amount)) { + throw new Error(`Insufficient frozen balance to unfreeze: ${currentFrozen} < ${amount}`); + } + newBalance = currentBalance.plus(amount); + newFrozen = currentFrozen.minus(amount); + } + + const updatedAccount = await tx.systemAccount.update({ + where: { id: systemAccountId }, + data: { + [balanceField]: newBalance.toFixed(8), + [frozenField]: newFrozen.toFixed(8), + }, + }); + + const transaction = await tx.systemAccountTransaction.create({ + data: { + systemAccountId, + transactionType: freeze ? 'FREEZE' : 'UNFREEZE', + assetType, + amount: amount.toFixed(8), + balanceBefore: currentBalance.toFixed(8), + balanceAfter: newBalance.toFixed(8), + counterpartyType: transactionInput.counterpartyType, + counterpartyAccountSeq: transactionInput.counterpartyAccountSeq, + counterpartyUserId: transactionInput.counterpartyUserId, + referenceId: transactionInput.referenceId, + referenceType: transactionInput.referenceType, + memo: transactionInput.memo, + metadata: transactionInput.metadata, + }, + }); + + return { account: updatedAccount, transaction }; + }); + } + + async getTransactions( + systemAccountId: string, + options?: { + transactionType?: TransactionType; + assetType?: AssetType; + startDate?: Date; + endDate?: Date; + limit?: number; + offset?: number; + }, + ): Promise<{ transactions: SystemAccountTransaction[]; total: number }> { + const where: Prisma.SystemAccountTransactionWhereInput = { + systemAccountId, + }; + + if (options?.transactionType) { + where.transactionType = options.transactionType; + } + if (options?.assetType) { + where.assetType = options.assetType; + } + if (options?.startDate || options?.endDate) { + where.createdAt = {}; + if (options.startDate) where.createdAt.gte = options.startDate; + if (options.endDate) where.createdAt.lte = options.endDate; + } + + const [transactions, total] = await Promise.all([ + this.prisma.systemAccountTransaction.findMany({ + where, + orderBy: { createdAt: 'desc' }, + take: options?.limit || 50, + skip: options?.offset || 0, + }), + this.prisma.systemAccountTransaction.count({ where }), + ]); + + return { transactions, total }; + } +} diff --git a/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/user-wallet.repository.ts b/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/user-wallet.repository.ts new file mode 100644 index 00000000..08c977b5 --- /dev/null +++ b/backend/services/mining-wallet-service/src/infrastructure/persistence/repositories/user-wallet.repository.ts @@ -0,0 +1,484 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { PrismaService } from '../prisma/prisma.service'; +import { + UserWallet, + UserWalletTransaction, + UserWalletType, + TransactionType, + AssetType, + CounterpartyType, + PoolAccountType, + Prisma, +} from '@prisma/client'; +import Decimal from 'decimal.js'; + +export interface CreateUserWalletInput { + accountSequence: string; + walletType: UserWalletType; +} + +export interface UserWalletTransactionInput { + userWalletId: string; + accountSequence: string; + walletType: UserWalletType; + transactionType: TransactionType; + assetType: AssetType; + amount: Decimal; + balanceBefore: Decimal; + balanceAfter: Decimal; + counterpartyType?: CounterpartyType; + counterpartyAccountSeq?: string; + counterpartyUserId?: string; + counterpartySystemId?: string; + counterpartyPoolType?: PoolAccountType; + counterpartyAddress?: string; + referenceId?: string; + referenceType?: string; + txHash?: string; + memo?: string; + metadata?: any; +} + +@Injectable() +export class UserWalletRepository { + private readonly logger = new Logger(UserWalletRepository.name); + + constructor(private readonly prisma: PrismaService) {} + + /** + * 创建用户钱包(如果不存在) + */ + async createIfNotExists(input: CreateUserWalletInput): Promise { + return this.prisma.userWallet.upsert({ + where: { + accountSequence_walletType: { + accountSequence: input.accountSequence, + walletType: input.walletType, + }, + }, + create: { + accountSequence: input.accountSequence, + walletType: input.walletType, + }, + update: {}, + }); + } + + /** + * 为用户创建所有类型的钱包 + */ + async createAllWalletsForUser(accountSequence: string): Promise { + const walletTypes: UserWalletType[] = ['CONTRIBUTION', 'TOKEN_STORAGE', 'GREEN_POINTS']; + + return this.prisma.$transaction( + walletTypes.map((walletType) => + this.prisma.userWallet.upsert({ + where: { + accountSequence_walletType: { + accountSequence, + walletType, + }, + }, + create: { + accountSequence, + walletType, + }, + update: {}, + }), + ), + ); + } + + async findByAccountSequence(accountSequence: string): Promise { + return this.prisma.userWallet.findMany({ + where: { accountSequence }, + }); + } + + async findByAccountAndType( + accountSequence: string, + walletType: UserWalletType, + ): Promise { + return this.prisma.userWallet.findUnique({ + where: { + accountSequence_walletType: { + accountSequence, + walletType, + }, + }, + }); + } + + async findById(id: string): Promise { + return this.prisma.userWallet.findUnique({ + where: { id }, + }); + } + + /** + * 更新余额并记录交易(原子操作) + */ + async updateBalanceWithTransaction( + accountSequence: string, + walletType: UserWalletType, + assetType: AssetType, + amount: Decimal, + transactionInput: Omit< + UserWalletTransactionInput, + 'userWalletId' | 'accountSequence' | 'walletType' | 'assetType' | 'amount' | 'balanceBefore' | 'balanceAfter' + >, + ): Promise<{ wallet: UserWallet; transaction: UserWalletTransaction }> { + return this.prisma.$transaction(async (tx) => { + // 确保钱包存在 + let wallet = await tx.userWallet.findUnique({ + where: { + accountSequence_walletType: { + accountSequence, + walletType, + }, + }, + }); + + if (!wallet) { + wallet = await tx.userWallet.create({ + data: { + accountSequence, + walletType, + }, + }); + } + + const currentBalance = new Decimal(wallet.balance.toString()); + const newBalance = currentBalance.plus(amount); + + if (newBalance.lessThan(0)) { + throw new Error( + `Insufficient balance for user ${accountSequence} wallet ${walletType}: ${currentBalance} + ${amount} < 0`, + ); + } + + // 更新余额 + const updatedWallet = await tx.userWallet.update({ + where: { id: wallet.id }, + data: { + balance: newBalance.toFixed(8), + totalInflow: amount.greaterThan(0) + ? new Decimal(wallet.totalInflow.toString()).plus(amount).toFixed(8) + : wallet.totalInflow, + totalOutflow: amount.lessThan(0) + ? new Decimal(wallet.totalOutflow.toString()).plus(amount.abs()).toFixed(8) + : wallet.totalOutflow, + }, + }); + + // 记录交易 + const transaction = await tx.userWalletTransaction.create({ + data: { + userWalletId: wallet.id, + accountSequence, + walletType, + transactionType: transactionInput.transactionType, + assetType, + amount: amount.toFixed(8), + balanceBefore: currentBalance.toFixed(8), + balanceAfter: newBalance.toFixed(8), + counterpartyType: transactionInput.counterpartyType, + counterpartyAccountSeq: transactionInput.counterpartyAccountSeq, + counterpartyUserId: transactionInput.counterpartyUserId, + counterpartySystemId: transactionInput.counterpartySystemId, + counterpartyPoolType: transactionInput.counterpartyPoolType, + counterpartyAddress: transactionInput.counterpartyAddress, + referenceId: transactionInput.referenceId, + referenceType: transactionInput.referenceType, + txHash: transactionInput.txHash, + memo: transactionInput.memo, + metadata: transactionInput.metadata, + }, + }); + + return { wallet: updatedWallet, transaction }; + }); + } + + /** + * 冻结/解冻余额 + */ + async freezeBalance( + accountSequence: string, + walletType: UserWalletType, + assetType: AssetType, + amount: Decimal, + freeze: boolean, + transactionInput: Omit< + UserWalletTransactionInput, + 'userWalletId' | 'accountSequence' | 'walletType' | 'assetType' | 'amount' | 'balanceBefore' | 'balanceAfter' | 'transactionType' + >, + ): Promise<{ wallet: UserWallet; transaction: UserWalletTransaction }> { + return this.prisma.$transaction(async (tx) => { + const wallet = await tx.userWallet.findUnique({ + where: { + accountSequence_walletType: { + accountSequence, + walletType, + }, + }, + }); + + if (!wallet) { + throw new Error(`User wallet not found: ${accountSequence} - ${walletType}`); + } + + const currentBalance = new Decimal(wallet.balance.toString()); + const currentFrozen = new Decimal(wallet.frozenBalance.toString()); + + let newBalance: Decimal; + let newFrozen: Decimal; + + if (freeze) { + if (currentBalance.lessThan(amount)) { + throw new Error(`Insufficient balance to freeze: ${currentBalance} < ${amount}`); + } + newBalance = currentBalance.minus(amount); + newFrozen = currentFrozen.plus(amount); + } else { + if (currentFrozen.lessThan(amount)) { + throw new Error(`Insufficient frozen balance to unfreeze: ${currentFrozen} < ${amount}`); + } + newBalance = currentBalance.plus(amount); + newFrozen = currentFrozen.minus(amount); + } + + const updatedWallet = await tx.userWallet.update({ + where: { id: wallet.id }, + data: { + balance: newBalance.toFixed(8), + frozenBalance: newFrozen.toFixed(8), + }, + }); + + const transaction = await tx.userWalletTransaction.create({ + data: { + userWalletId: wallet.id, + accountSequence, + walletType, + transactionType: freeze ? 'FREEZE' : 'UNFREEZE', + assetType, + amount: amount.toFixed(8), + balanceBefore: currentBalance.toFixed(8), + balanceAfter: newBalance.toFixed(8), + counterpartyType: transactionInput.counterpartyType, + counterpartyAccountSeq: transactionInput.counterpartyAccountSeq, + counterpartyUserId: transactionInput.counterpartyUserId, + referenceId: transactionInput.referenceId, + referenceType: transactionInput.referenceType, + memo: transactionInput.memo, + metadata: transactionInput.metadata, + }, + }); + + return { wallet: updatedWallet, transaction }; + }); + } + + /** + * 用户间转账 + */ + async transferBetweenUsers( + fromAccountSeq: string, + toAccountSeq: string, + walletType: UserWalletType, + assetType: AssetType, + amount: Decimal, + referenceInfo: { + referenceId?: string; + referenceType?: string; + memo?: string; + }, + ): Promise<{ + fromWallet: UserWallet; + toWallet: UserWallet; + fromTransaction: UserWalletTransaction; + toTransaction: UserWalletTransaction; + }> { + return this.prisma.$transaction(async (tx) => { + // 获取发送方钱包 + const fromWallet = await tx.userWallet.findUnique({ + where: { + accountSequence_walletType: { + accountSequence: fromAccountSeq, + walletType, + }, + }, + }); + + if (!fromWallet) { + throw new Error(`From wallet not found: ${fromAccountSeq} - ${walletType}`); + } + + const fromBalance = new Decimal(fromWallet.balance.toString()); + if (fromBalance.lessThan(amount)) { + throw new Error(`Insufficient balance: ${fromBalance} < ${amount}`); + } + + // 确保接收方钱包存在 + let toWallet = await tx.userWallet.findUnique({ + where: { + accountSequence_walletType: { + accountSequence: toAccountSeq, + walletType, + }, + }, + }); + + if (!toWallet) { + toWallet = await tx.userWallet.create({ + data: { + accountSequence: toAccountSeq, + walletType, + }, + }); + } + + const toBalance = new Decimal(toWallet.balance.toString()); + + // 更新发送方 + const newFromBalance = fromBalance.minus(amount); + const updatedFromWallet = await tx.userWallet.update({ + where: { id: fromWallet.id }, + data: { + balance: newFromBalance.toFixed(8), + totalOutflow: new Decimal(fromWallet.totalOutflow.toString()).plus(amount).toFixed(8), + }, + }); + + // 更新接收方 + const newToBalance = toBalance.plus(amount); + const updatedToWallet = await tx.userWallet.update({ + where: { id: toWallet.id }, + data: { + balance: newToBalance.toFixed(8), + totalInflow: new Decimal(toWallet.totalInflow.toString()).plus(amount).toFixed(8), + }, + }); + + // 记录发送方交易 + const fromTransaction = await tx.userWalletTransaction.create({ + data: { + userWalletId: fromWallet.id, + accountSequence: fromAccountSeq, + walletType, + transactionType: 'TRANSFER_OUT', + assetType, + amount: amount.negated().toFixed(8), + balanceBefore: fromBalance.toFixed(8), + balanceAfter: newFromBalance.toFixed(8), + counterpartyType: 'USER', + counterpartyAccountSeq: toAccountSeq, + referenceId: referenceInfo.referenceId, + referenceType: referenceInfo.referenceType, + memo: referenceInfo.memo || `划转给用户[${toAccountSeq}], 数量${amount.toFixed(8)}`, + }, + }); + + // 记录接收方交易 + const toTransaction = await tx.userWalletTransaction.create({ + data: { + userWalletId: toWallet.id, + accountSequence: toAccountSeq, + walletType, + transactionType: 'TRANSFER_IN', + assetType, + amount: amount.toFixed(8), + balanceBefore: toBalance.toFixed(8), + balanceAfter: newToBalance.toFixed(8), + counterpartyType: 'USER', + counterpartyAccountSeq: fromAccountSeq, + referenceId: referenceInfo.referenceId, + referenceType: referenceInfo.referenceType, + memo: referenceInfo.memo || `收到用户[${fromAccountSeq}]划转, 数量${amount.toFixed(8)}`, + }, + }); + + return { + fromWallet: updatedFromWallet, + toWallet: updatedToWallet, + fromTransaction, + toTransaction, + }; + }); + } + + async getTransactions( + accountSequence: string, + options?: { + walletType?: UserWalletType; + transactionType?: TransactionType; + assetType?: AssetType; + startDate?: Date; + endDate?: Date; + limit?: number; + offset?: number; + }, + ): Promise<{ transactions: UserWalletTransaction[]; total: number }> { + const where: Prisma.UserWalletTransactionWhereInput = { + accountSequence, + }; + + if (options?.walletType) { + where.walletType = options.walletType; + } + if (options?.transactionType) { + where.transactionType = options.transactionType; + } + if (options?.assetType) { + where.assetType = options.assetType; + } + if (options?.startDate || options?.endDate) { + where.createdAt = {}; + if (options.startDate) where.createdAt.gte = options.startDate; + if (options.endDate) where.createdAt.lte = options.endDate; + } + + const [transactions, total] = await Promise.all([ + this.prisma.userWalletTransaction.findMany({ + where, + orderBy: { createdAt: 'desc' }, + take: options?.limit || 50, + skip: options?.offset || 0, + }), + this.prisma.userWalletTransaction.count({ where }), + ]); + + return { transactions, total }; + } + + /** + * 获取用户所有钱包余额汇总 + */ + async getUserWalletSummary(accountSequence: string): Promise<{ + walletType: UserWalletType; + balance: string; + frozenBalance: string; + totalInflow: string; + totalOutflow: string; + }[]> { + const wallets = await this.prisma.userWallet.findMany({ + where: { accountSequence }, + select: { + walletType: true, + balance: true, + frozenBalance: true, + totalInflow: true, + totalOutflow: true, + }, + }); + + return wallets.map((w) => ({ + walletType: w.walletType, + balance: w.balance.toString(), + frozenBalance: w.frozenBalance.toString(), + totalInflow: w.totalInflow.toString(), + totalOutflow: w.totalOutflow.toString(), + })); + } +} diff --git a/backend/services/mining-wallet-service/src/infrastructure/redis/redis.service.ts b/backend/services/mining-wallet-service/src/infrastructure/redis/redis.service.ts new file mode 100644 index 00000000..05ef4d24 --- /dev/null +++ b/backend/services/mining-wallet-service/src/infrastructure/redis/redis.service.ts @@ -0,0 +1,106 @@ +import { Injectable, Inject, OnModuleInit, OnModuleDestroy, Logger } from '@nestjs/common'; +import Redis from 'ioredis'; + +interface RedisOptions { + host: string; + port: number; + password?: string; + db?: number; +} + +@Injectable() +export class RedisService implements OnModuleInit, OnModuleDestroy { + private readonly logger = new Logger(RedisService.name); + private client: Redis; + + constructor(@Inject('REDIS_OPTIONS') private readonly options: RedisOptions) {} + + async onModuleInit() { + this.client = new Redis({ + host: this.options.host, + port: this.options.port, + password: this.options.password, + db: this.options.db ?? 15, + retryStrategy: (times) => Math.min(times * 50, 2000), + }); + + this.client.on('error', (err) => this.logger.error('Redis error', err)); + this.client.on('connect', () => this.logger.log('Connected to Redis')); + } + + async onModuleDestroy() { + await this.client.quit(); + } + + getClient(): Redis { + return this.client; + } + + 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 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); + } + + async acquireLock(lockKey: string, ttlSeconds: number = 30): Promise { + const lockValue = `${Date.now()}-${Math.random().toString(36).substring(7)}`; + const result = await this.client.set(lockKey, lockValue, 'EX', ttlSeconds, 'NX'); + return result === 'OK' ? lockValue : null; + } + + async releaseLock(lockKey: string, lockValue: string): Promise { + const script = ` + if redis.call("get", KEYS[1]) == ARGV[1] then + return redis.call("del", KEYS[1]) + else + return 0 + end + `; + const result = await this.client.eval(script, 1, lockKey, lockValue); + return result === 1; + } + + async incr(key: string): Promise { + return this.client.incr(key); + } + + async incrByFloat(key: string, increment: number): Promise { + return this.client.incrbyfloat(key, increment); + } + + async del(key: string): Promise { + return this.client.del(key); + } + + async hset(key: string, field: string, value: string): Promise { + return this.client.hset(key, field, value); + } + + async hget(key: string, field: string): Promise { + return this.client.hget(key, field); + } + + async hgetall(key: string): Promise> { + return this.client.hgetall(key); + } +} diff --git a/backend/services/mining-wallet-service/src/main.ts b/backend/services/mining-wallet-service/src/main.ts new file mode 100644 index 00000000..412661d7 --- /dev/null +++ b/backend/services/mining-wallet-service/src/main.ts @@ -0,0 +1,71 @@ +import { NestFactory } from '@nestjs/core'; +import { ValidationPipe } from '@nestjs/common'; +import { SwaggerModule, DocumentBuilder } from '@nestjs/swagger'; +import { MicroserviceOptions, Transport } from '@nestjs/microservices'; +import { AppModule } from './app.module'; + +async function bootstrap() { + const app = await NestFactory.create(AppModule); + + // Global validation pipe + app.useGlobalPipes( + new ValidationPipe({ + whitelist: true, + transform: true, + forbidNonWhitelisted: true, + }), + ); + + // CORS + app.enableCors({ + origin: process.env.CORS_ORIGIN || '*', + credentials: true, + }); + + // Global prefix + app.setGlobalPrefix('api/v1'); + + // Swagger documentation + if (process.env.SWAGGER_ENABLED !== 'false') { + const config = new DocumentBuilder() + .setTitle('Mining Wallet Service API') + .setDescription('挖矿钱包服务 API - 100% 独立的钱包管理系统') + .setVersion('1.0') + .addBearerAuth() + .addTag('System Accounts', '系统账户管理') + .addTag('Pool Accounts', '池账户管理') + .addTag('User Wallets', '用户钱包管理') + .addTag('Regions', '区域管理') + .addTag('Blockchain', 'KAVA区块链集成') + .addTag('Health', '健康检查') + .build(); + + const document = SwaggerModule.createDocument(app, config); + SwaggerModule.setup('api/docs', app, document); + } + + // Connect Kafka microservice + const kafkaBrokers = process.env.KAFKA_BROKERS || 'localhost:9092'; + app.connectMicroservice({ + transport: Transport.KAFKA, + options: { + client: { + clientId: 'mining-wallet-service', + brokers: kafkaBrokers.split(','), + }, + consumer: { + groupId: 'mining-wallet-service-group', + }, + }, + }); + + await app.startAllMicroservices(); + + const port = process.env.PORT || 3025; + await app.listen(port); + + console.log(`Mining Wallet Service is running on port ${port}`); + console.log(`Swagger docs: http://localhost:${port}/api/docs`); +} + +bootstrap(); diff --git a/backend/services/mining-wallet-service/src/shared/decorators/current-user.decorator.ts b/backend/services/mining-wallet-service/src/shared/decorators/current-user.decorator.ts new file mode 100644 index 00000000..d0be98f0 --- /dev/null +++ b/backend/services/mining-wallet-service/src/shared/decorators/current-user.decorator.ts @@ -0,0 +1,20 @@ +import { createParamDecorator, ExecutionContext } from '@nestjs/common'; + +export interface CurrentUserPayload { + userId: string; + accountSequence: string; + role?: string; +} + +export const CurrentUser = createParamDecorator( + (data: keyof CurrentUserPayload | undefined, ctx: ExecutionContext): CurrentUserPayload | string | undefined => { + const request = ctx.switchToHttp().getRequest(); + const user = request.user as CurrentUserPayload | undefined; + + if (data && user) { + return user[data]; + } + + return user; + }, +); diff --git a/backend/services/mining-wallet-service/src/shared/filters/domain-exception.filter.ts b/backend/services/mining-wallet-service/src/shared/filters/domain-exception.filter.ts new file mode 100644 index 00000000..8a67bea8 --- /dev/null +++ b/backend/services/mining-wallet-service/src/shared/filters/domain-exception.filter.ts @@ -0,0 +1,64 @@ +import { + ExceptionFilter, + Catch, + ArgumentsHost, + HttpStatus, + Logger, + HttpException, +} from '@nestjs/common'; +import { Request, Response } from 'express'; + +export class DomainException extends Error { + constructor( + message: string, + public readonly code: string, + public readonly httpStatus: HttpStatus = HttpStatus.BAD_REQUEST, + ) { + super(message); + this.name = 'DomainException'; + } +} + +@Catch() +export class DomainExceptionFilter implements ExceptionFilter { + private readonly logger = new Logger(DomainExceptionFilter.name); + + catch(exception: unknown, host: ArgumentsHost) { + const ctx = host.switchToHttp(); + const response = ctx.getResponse(); + const request = ctx.getRequest(); + + let status = HttpStatus.INTERNAL_SERVER_ERROR; + let code = 'INTERNAL_ERROR'; + let message = 'Internal server error'; + + if (exception instanceof DomainException) { + status = exception.httpStatus; + code = exception.code; + message = exception.message; + } else if (exception instanceof HttpException) { + status = exception.getStatus(); + const exceptionResponse = exception.getResponse(); + if (typeof exceptionResponse === 'object' && exceptionResponse !== null) { + message = (exceptionResponse as any).message || exception.message; + code = 'HTTP_ERROR'; + } else { + message = exception.message; + code = 'HTTP_ERROR'; + } + } else if (exception instanceof Error) { + message = exception.message; + this.logger.error(`Unhandled exception: ${exception.message}`, exception.stack); + } + + response.status(status).json({ + success: false, + error: { + code, + message: Array.isArray(message) ? message : [message], + }, + timestamp: new Date().toISOString(), + path: request.url, + }); + } +} diff --git a/backend/services/mining-wallet-service/src/shared/guards/jwt-auth.guard.ts b/backend/services/mining-wallet-service/src/shared/guards/jwt-auth.guard.ts new file mode 100644 index 00000000..e3a76559 --- /dev/null +++ b/backend/services/mining-wallet-service/src/shared/guards/jwt-auth.guard.ts @@ -0,0 +1,71 @@ +import { Injectable, CanActivate, ExecutionContext, UnauthorizedException, SetMetadata } from '@nestjs/common'; +import { Reflector } from '@nestjs/core'; +import { ConfigService } from '@nestjs/config'; +import * as jwt from 'jsonwebtoken'; + +export const IS_PUBLIC_KEY = 'isPublic'; +export const Public = () => SetMetadata(IS_PUBLIC_KEY, true); + +export const IS_ADMIN_KEY = 'isAdmin'; +export const AdminOnly = () => SetMetadata(IS_ADMIN_KEY, true); + +@Injectable() +export class JwtAuthGuard implements CanActivate { + constructor( + private reflector: Reflector, + private configService: ConfigService, + ) {} + + canActivate(context: ExecutionContext): boolean { + const isPublic = this.reflector.getAllAndOverride(IS_PUBLIC_KEY, [ + context.getHandler(), + context.getClass(), + ]); + + if (isPublic) { + return true; + } + + const request = context.switchToHttp().getRequest(); + const token = this.extractTokenFromHeader(request); + + if (!token) { + throw new UnauthorizedException('No token provided'); + } + + try { + const secret = this.configService.get('JWT_SECRET', 'default-secret'); + const payload = jwt.verify(token, secret) as any; + + request.user = { + userId: payload.sub, + accountSequence: payload.accountSequence, + role: payload.role, + }; + + // Check admin requirement + const isAdminOnly = this.reflector.getAllAndOverride(IS_ADMIN_KEY, [ + context.getHandler(), + context.getClass(), + ]); + + if (isAdminOnly && payload.role !== 'ADMIN' && payload.role !== 'SUPER_ADMIN') { + throw new UnauthorizedException('Admin access required'); + } + + return true; + } catch (error) { + if (error instanceof UnauthorizedException) { + throw error; + } + throw new UnauthorizedException('Invalid token'); + } + } + + private extractTokenFromHeader(request: any): string | null { + const authHeader = request.headers.authorization; + if (!authHeader) return null; + const [type, token] = authHeader.split(' '); + return type === 'Bearer' ? token : null; + } +} diff --git a/backend/services/mining-wallet-service/src/shared/interceptors/logging.interceptor.ts b/backend/services/mining-wallet-service/src/shared/interceptors/logging.interceptor.ts new file mode 100644 index 00000000..08cf5f73 --- /dev/null +++ b/backend/services/mining-wallet-service/src/shared/interceptors/logging.interceptor.ts @@ -0,0 +1,27 @@ +import { Injectable, NestInterceptor, ExecutionContext, CallHandler, Logger } from '@nestjs/common'; +import { Observable } from 'rxjs'; +import { tap } from 'rxjs/operators'; + +@Injectable() +export class LoggingInterceptor implements NestInterceptor { + private readonly logger = new Logger('HTTP'); + + intercept(context: ExecutionContext, next: CallHandler): Observable { + const request = context.switchToHttp().getRequest(); + const { method, url } = request; + const startTime = Date.now(); + + return next.handle().pipe( + tap({ + next: () => { + const responseTime = Date.now() - startTime; + this.logger.log(`${method} ${url} ${responseTime}ms`); + }, + error: (error) => { + const responseTime = Date.now() - startTime; + this.logger.error(`${method} ${url} ${responseTime}ms - Error: ${error.message}`); + }, + }), + ); + } +} diff --git a/backend/services/mining-wallet-service/src/shared/interceptors/transform.interceptor.ts b/backend/services/mining-wallet-service/src/shared/interceptors/transform.interceptor.ts new file mode 100644 index 00000000..c6205a4a --- /dev/null +++ b/backend/services/mining-wallet-service/src/shared/interceptors/transform.interceptor.ts @@ -0,0 +1,16 @@ +import { Injectable, NestInterceptor, ExecutionContext, CallHandler } from '@nestjs/common'; +import { Observable } from 'rxjs'; +import { map } from 'rxjs/operators'; + +@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/mining-wallet-service/tsconfig.json b/backend/services/mining-wallet-service/tsconfig.json new file mode 100644 index 00000000..bd3c3946 --- /dev/null +++ b/backend/services/mining-wallet-service/tsconfig.json @@ -0,0 +1,24 @@ +{ + "compilerOptions": { + "module": "commonjs", + "declaration": true, + "removeComments": true, + "emitDecoratorMetadata": true, + "experimentalDecorators": true, + "allowSyntheticDefaultImports": true, + "target": "ES2021", + "sourceMap": true, + "outDir": "./dist", + "baseUrl": "./", + "incremental": true, + "skipLibCheck": true, + "strictNullChecks": true, + "noImplicitAny": true, + "strictBindCallApply": true, + "forceConsistentCasingInFileNames": true, + "noFallthroughCasesInSwitch": true, + "paths": { + "@/*": ["src/*"] + } + } +} diff --git a/backend/services/trading-service/prisma/schema.prisma b/backend/services/trading-service/prisma/schema.prisma index 5670cb8b..0c06ce66 100644 --- a/backend/services/trading-service/prisma/schema.prisma +++ b/backend/services/trading-service/prisma/schema.prisma @@ -11,19 +11,19 @@ datasource db { // 用户交易账户 model TradingAccount { - id String @id @default(uuid()) - accountSequence String @unique - shareBalance Decimal @db.Decimal(30, 8) @default(0) // 积分股余额 - cashBalance Decimal @db.Decimal(30, 8) @default(0) // 现金余额 - frozenShares Decimal @db.Decimal(30, 8) @default(0) // 冻结积分股 - frozenCash Decimal @db.Decimal(30, 8) @default(0) // 冻结现金 - totalBought Decimal @db.Decimal(30, 8) @default(0) // 累计买入量 - totalSold Decimal @db.Decimal(30, 8) @default(0) // 累计卖出量 - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt + id String @id @default(uuid()) + accountSequence String @unique + shareBalance Decimal @default(0) @db.Decimal(30, 8) // 积分股余额 + cashBalance Decimal @default(0) @db.Decimal(30, 8) // 现金余额 + frozenShares Decimal @default(0) @db.Decimal(30, 8) // 冻结积分股 + frozenCash Decimal @default(0) @db.Decimal(30, 8) // 冻结现金 + totalBought Decimal @default(0) @db.Decimal(30, 8) // 累计买入量 + totalSold Decimal @default(0) @db.Decimal(30, 8) // 累计卖出量 + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt - orders Order[] - transactions TradingTransaction[] + orders Order[] + transactions TradingTransaction[] @@map("trading_accounts") } @@ -32,24 +32,24 @@ model TradingAccount { // 交易订单 model Order { - id String @id @default(uuid()) - orderNo String @unique // 订单号 + id String @id @default(uuid()) + orderNo String @unique // 订单号 accountSequence String - type String // BUY, SELL - status String // PENDING, PARTIAL, FILLED, CANCELLED - price Decimal @db.Decimal(30, 18) // 挂单价格 - quantity Decimal @db.Decimal(30, 8) // 订单数量 - filledQuantity Decimal @db.Decimal(30, 8) @default(0) // 已成交数量 - remainingQuantity Decimal @db.Decimal(30, 8) // 剩余数量 - averagePrice Decimal @db.Decimal(30, 18) @default(0) // 平均成交价 - totalAmount Decimal @db.Decimal(30, 8) @default(0) // 总成交金额 - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt + type String // BUY, SELL + status String // PENDING, PARTIAL, FILLED, CANCELLED + price Decimal @db.Decimal(30, 18) // 挂单价格 + quantity Decimal @db.Decimal(30, 8) // 订单数量 + filledQuantity Decimal @default(0) @db.Decimal(30, 8) // 已成交数量 + remainingQuantity Decimal @db.Decimal(30, 8) // 剩余数量 + averagePrice Decimal @default(0) @db.Decimal(30, 18) // 平均成交价 + totalAmount Decimal @default(0) @db.Decimal(30, 8) // 总成交金额 + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt cancelledAt DateTime? completedAt DateTime? - account TradingAccount @relation(fields: [accountSequence], references: [accountSequence]) - trades Trade[] + account TradingAccount @relation(fields: [accountSequence], references: [accountSequence]) + trades Trade[] @@index([accountSequence, status]) @@index([type, status, price]) @@ -59,18 +59,18 @@ model Order { // 成交记录 model Trade { - id String @id @default(uuid()) - tradeNo String @unique - buyOrderId String - sellOrderId String - buyerSequence String - sellerSequence String - price Decimal @db.Decimal(30, 18) - quantity Decimal @db.Decimal(30, 8) - amount Decimal @db.Decimal(30, 8) // price * quantity - createdAt DateTime @default(now()) + id String @id @default(uuid()) + tradeNo String @unique + buyOrderId String + sellOrderId String + buyerSequence String + sellerSequence String + price Decimal @db.Decimal(30, 18) + quantity Decimal @db.Decimal(30, 8) + amount Decimal @db.Decimal(30, 8) // price * quantity + createdAt DateTime @default(now()) - buyOrder Order @relation(fields: [buyOrderId], references: [id]) + buyOrder Order @relation(fields: [buyOrderId], references: [id]) @@index([buyerSequence]) @@index([sellerSequence]) @@ -81,41 +81,95 @@ model Trade { // ==================== 交易流水 ==================== model TradingTransaction { - id String @id @default(uuid()) + id String @id @default(uuid()) accountSequence String - type String // TRANSFER_IN, TRANSFER_OUT, BUY, SELL, FREEZE, UNFREEZE, DEPOSIT, WITHDRAW - assetType String // SHARE, CASH - amount Decimal @db.Decimal(30, 8) - balanceBefore Decimal @db.Decimal(30, 8) - balanceAfter Decimal @db.Decimal(30, 8) + type String // TRANSFER_IN, TRANSFER_OUT, BUY, SELL, FREEZE, UNFREEZE, DEPOSIT, WITHDRAW + assetType String // SHARE, CASH + amount Decimal @db.Decimal(30, 8) + balanceBefore Decimal @db.Decimal(30, 8) + balanceAfter Decimal @db.Decimal(30, 8) referenceId String? referenceType String? - description String? - createdAt DateTime @default(now()) - account TradingAccount @relation(fields: [accountSequence], references: [accountSequence]) + // 交易对手方信息(关键:用户ID和账户序列号) + counterpartyType String? @map("counterparty_type") // USER, POOL, SYSTEM + counterpartyAccountSeq String? @map("counterparty_account_seq") // 对手方账户序列号 + counterpartyUserId String? @map("counterparty_user_id") // 对手方用户ID + + // 详细备注(包含完整交易信息,格式: "卖出给用户[U123456], 价格0.5USDT") + memo String? @db.Text + description String? // 保留兼容旧字段 + createdAt DateTime @default(now()) + + account TradingAccount @relation(fields: [accountSequence], references: [accountSequence]) @@index([accountSequence, createdAt(sort: Desc)]) + @@index([counterpartyAccountSeq]) + @@index([counterpartyUserId]) @@map("trading_transactions") } // ==================== 流通池 ==================== -// 流通池 +// 流通池(交易所流通池) model CirculationPool { - id String @id @default(uuid()) - totalShares Decimal @db.Decimal(30, 8) @default(0) // 流通池中的积分股 - totalCash Decimal @db.Decimal(30, 8) @default(0) // 流通池中的现金(股池) - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt + id String @id @default(uuid()) + totalShares Decimal @default(0) @db.Decimal(30, 8) // 流通池中的积分股 + totalCash Decimal @default(0) @db.Decimal(30, 8) // 流通池中的现金(股池) + totalInflow Decimal @default(0) @db.Decimal(30, 8) // 累计流入 + totalOutflow Decimal @default(0) @db.Decimal(30, 8) // 累计流出 + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + transactions CirculationPoolTransaction[] @@map("circulation_pools") } -// 流通池变动记录 +// 流通池变动记录(包含交易对手方信息) +model CirculationPoolTransaction { + id String @id @default(uuid()) + poolId String @map("pool_id") + type String // SHARE_IN, SHARE_OUT, CASH_IN, CASH_OUT, TRADE_BUY, TRADE_SELL + assetType String // SHARE, CASH + amount Decimal @db.Decimal(30, 8) + balanceBefore Decimal @map("balance_before") @db.Decimal(30, 8) + balanceAfter Decimal @map("balance_after") @db.Decimal(30, 8) + + // 交易对手方信息(关键:用户ID和账户序列号) + counterpartyType String? @map("counterparty_type") // USER, POOL, SYSTEM, EXTERNAL + counterpartyAccountSeq String? @map("counterparty_account_seq") // 对手方账户序列号 + counterpartyUserId String? @map("counterparty_user_id") // 对手方用户ID + + // 关联信息 + referenceId String? @map("reference_id") // 关联业务ID(如订单ID、交易ID) + referenceType String? @map("reference_type") // 关联类型(ORDER, TRADE, TRANSFER) + + // 详细备注(包含完整交易信息) + // 格式示例: "用户[U123456]买入100股, 订单号ORD20240110001" + memo String? @db.Text + + // 扩展数据 + metadata Json? + + createdAt DateTime @default(now()) @map("created_at") + + pool CirculationPool @relation(fields: [poolId], references: [id]) + + @@index([poolId, createdAt(sort: Desc)]) + @@index([type, assetType]) + @@index([counterpartyAccountSeq]) + @@index([counterpartyUserId]) + @@index([referenceId]) + @@index([createdAt(sort: Desc)]) + @@map("circulation_pool_transactions") +} + +// 保留旧的PoolTransaction以兼容(标记为废弃,后续迁移后删除) +// @deprecated 使用 CirculationPoolTransaction 替代 model PoolTransaction { id String @id @default(uuid()) - type String // SHARE_IN, SHARE_OUT, CASH_IN, CASH_OUT + type String // SHARE_IN, SHARE_OUT, CASH_IN, CASH_OUT amount Decimal @db.Decimal(30, 8) referenceId String? description String? @@ -129,16 +183,16 @@ model PoolTransaction { // 分钟K线 model MinuteKLine { - id String @id @default(uuid()) - minute DateTime @unique - open Decimal @db.Decimal(30, 18) - high Decimal @db.Decimal(30, 18) - low Decimal @db.Decimal(30, 18) - close Decimal @db.Decimal(30, 18) - volume Decimal @db.Decimal(30, 8) // 成交量 - amount Decimal @db.Decimal(30, 8) // 成交额 - tradeCount Int @default(0) // 成交笔数 - createdAt DateTime @default(now()) + id String @id @default(uuid()) + minute DateTime @unique + open Decimal @db.Decimal(30, 18) + high Decimal @db.Decimal(30, 18) + low Decimal @db.Decimal(30, 18) + close Decimal @db.Decimal(30, 18) + volume Decimal @db.Decimal(30, 8) // 成交量 + amount Decimal @db.Decimal(30, 8) // 成交额 + tradeCount Int @default(0) // 成交笔数 + createdAt DateTime @default(now()) @@index([minute(sort: Desc)]) @@map("minute_klines") @@ -146,16 +200,16 @@ model MinuteKLine { // 小时K线 model HourKLine { - id String @id @default(uuid()) - hour DateTime @unique - open Decimal @db.Decimal(30, 18) - high Decimal @db.Decimal(30, 18) - low Decimal @db.Decimal(30, 18) - close Decimal @db.Decimal(30, 18) - volume Decimal @db.Decimal(30, 8) - amount Decimal @db.Decimal(30, 8) - tradeCount Int @default(0) - createdAt DateTime @default(now()) + id String @id @default(uuid()) + hour DateTime @unique + open Decimal @db.Decimal(30, 18) + high Decimal @db.Decimal(30, 18) + low Decimal @db.Decimal(30, 18) + close Decimal @db.Decimal(30, 18) + volume Decimal @db.Decimal(30, 8) + amount Decimal @db.Decimal(30, 8) + tradeCount Int @default(0) + createdAt DateTime @default(now()) @@index([hour(sort: Desc)]) @@map("hour_klines") @@ -163,16 +217,16 @@ model HourKLine { // 日K线 model DayKLine { - id String @id @default(uuid()) - date DateTime @unique @db.Date - open Decimal @db.Decimal(30, 18) - high Decimal @db.Decimal(30, 18) - low Decimal @db.Decimal(30, 18) - close Decimal @db.Decimal(30, 18) - volume Decimal @db.Decimal(30, 8) - amount Decimal @db.Decimal(30, 8) - tradeCount Int @default(0) - createdAt DateTime @default(now()) + id String @id @default(uuid()) + date DateTime @unique @db.Date + open Decimal @db.Decimal(30, 18) + high Decimal @db.Decimal(30, 18) + low Decimal @db.Decimal(30, 18) + close Decimal @db.Decimal(30, 18) + volume Decimal @db.Decimal(30, 8) + amount Decimal @db.Decimal(30, 8) + tradeCount Int @default(0) + createdAt DateTime @default(now()) @@index([date(sort: Desc)]) @@map("day_klines") @@ -182,16 +236,16 @@ model DayKLine { // 从挖矿账户划转记录 model TransferRecord { - id String @id @default(uuid()) - transferNo String @unique - accountSequence String - direction String // IN (从挖矿账户划入), OUT (划出到挖矿账户) - amount Decimal @db.Decimal(30, 8) - status String // PENDING, COMPLETED, FAILED - miningTxId String? // 挖矿服务的交易ID - errorMessage String? - createdAt DateTime @default(now()) - completedAt DateTime? + id String @id @default(uuid()) + transferNo String @unique + accountSequence String + direction String // IN (从挖矿账户划入), OUT (划出到挖矿账户) + amount Decimal @db.Decimal(30, 8) + status String // PENDING, COMPLETED, FAILED + miningTxId String? // 挖矿服务的交易ID + errorMessage String? + createdAt DateTime @default(now()) + completedAt DateTime? @@index([accountSequence]) @@index([status])