serverless-plugin-module-registry 1.0.4 → 1.0.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/service.ts"],"sourcesContent":["/*\n * serverless-plugin-module-registry\n * -------------------------------------------\n * A Serverless Framework plugin that scans module registry files,\n * discovers endpoints, and stores module feature mappings in DynamoDB\n * for cross-module discovery and access control.\n *\n * ✓ Compatible with Serverless Framework v3 & v4\n * ✓ Extends composer plugin functionality\n * ✓ Scans registry/ folders in each module\n * ✓ Creates/updates DynamoDB ModuleRegistry table\n */\n\nimport fs from 'fs'\nimport path from 'path'\nimport yaml from 'js-yaml'\nimport { glob } from 'glob'\nimport chalk from 'chalk'\n\ninterface ModuleRegistryConfig {\n tableName?: string\n region?: string\n skipDynamoDB?: boolean\n strict?: boolean\n}\n\ninterface RegistryFeature {\n name: string\n description: string\n version: string\n endpoints: string[]\n customPolicies?: any[]\n}\n\ninterface ModuleMetadata {\n name: string\n version: string\n description?: string\n}\n\ninterface RegistryModuleEntry {\n moduleName: string\n description: string\n version: string\n maintainer?: string\n tags?: string[]\n lastUpdated: string\n}\n\ninterface RegistryFeatureEntry {\n moduleName: string\n featureName: string\n description: string\n version: string\n endpoints: string[]\n customPolicies: any[]\n lastUpdated: string\n}\n\nclass ServerlessModuleRegistryPlugin {\n serverless: any\n options: Record<string, any>\n hooks: Record<string, () => Promise<void>>\n private config: ModuleRegistryConfig\n private servicePath: string\n private moduleEntries: RegistryModuleEntry[] = []\n private featureEntries: RegistryFeatureEntry[] = []\n\n constructor(serverless: any, options: Record<string, any>) {\n this.serverless = serverless\n this.options = options\n this.servicePath = this.serverless.config.servicePath || process.cwd()\n\n // Get plugin configuration with dynamic table name\n const custom = this.serverless.service.custom?.moduleRegistry || {}\n const defaultTableName = this.generateTableName()\n \n this.config = {\n tableName: custom.tableName || defaultTableName,\n region: custom.region || this.serverless.service.provider?.region || 'us-east-1',\n skipDynamoDB: custom.skipDynamoDB || false,\n strict: custom.strict || false,\n ...custom\n }\n\n // Register hooks\n this.hooks = {\n 'after:composer:initialize': this.processModuleRegistry.bind(this),\n 'after:aws:deploy:deploy:updateStack': this.ensureTableAndUpdateData.bind(this)\n }\n\n // Debug current configuration\n this.log(`📊 Module Registry plugin initialized:`)\n this.log(` Table: ${this.config.tableName}`)\n this.log(` Region: ${this.config.region}`)\n this.log(` Skip DynamoDB: ${this.config.skipDynamoDB}`)\n this.log(` Strict Mode: ${this.config.strict ? '🔒 ENABLED' : '✅ DISABLED'}`)\n this.log('')\n }\n\n /**\n * Generate table name following ARN prefix pattern\n */\n private generateTableName(): string {\n const service = this.serverless.service.service\n const provider = this.serverless.service.provider || {}\n const stage = provider.stage || 'dev'\n \n // Use same pattern as ARN prefixer: {service}-{stage}-module-registry\n return `${service}-${stage}-module-registry`\n }\n\n /**\n * Main processing function - scans all modules for registry files\n */\n private async processModuleRegistry(): Promise<void> {\n this.log('🔍 Scanning modules for registry definitions...')\n this.log(` Strict Mode: ${this.config.strict ? '🔒 ENABLED - Will FAIL deployment on missing registries' : '✅ DISABLED - Will gracefully skip missing registries'}`)\n\n const modulesDir = path.resolve(this.servicePath, 'serverless/modules')\n \n if (!fs.existsSync(modulesDir)) {\n this.log('⚠️ No modules directory found, skipping registry processing')\n return\n }\n\n // Get all module directories\n const moduleNames = fs.readdirSync(modulesDir)\n .filter(name => fs.statSync(path.join(modulesDir, name)).isDirectory())\n\n this.log(`Found ${moduleNames.length} modules: ${moduleNames.join(', ')}`)\n\n // Process each module\n for (const moduleName of moduleNames) {\n this.log(`\\n🔍 Checking module: ${moduleName}`)\n await this.processModule(moduleName, modulesDir)\n }\n\n this.log(`✅ Registry processing complete. Found ${this.moduleEntries.length} modules, ${this.featureEntries.length} features`)\n }\n\n /**\n * Process a single module's registry files\n */\n private async processModule(moduleName: string, modulesDir: string): Promise<void> {\n const moduleDir = path.join(modulesDir, moduleName)\n const registryDir = path.join(moduleDir, 'registry')\n\n if (!fs.existsSync(registryDir)) {\n if (this.config.strict) {\n const errorMessage = `❌ STRICT MODE VIOLATION: Module '${moduleName}' is missing required registry folder!\\n Expected: ${registryDir}\\n Fix: Create the registry folder or set strict: false in configuration`\n this.log('')\n this.log('🚨 DEPLOYMENT FAILED - STRICT MODE VIOLATION 🚨')\n this.log(errorMessage)\n this.log('')\n console.error(`[module-registry] ${errorMessage}`)\n throw new Error(`Module Registry Strict Mode Violation: Module '${moduleName}' missing registry folder`)\n }\n \n this.log(`📁 Module '${moduleName}' has no registry folder, skipping (strict mode disabled)`)\n return\n }\n\n this.log(`📋 Processing registry for module: ${moduleName}`)\n\n // Read module metadata\n const moduleMetadata = await this.readModuleMetadata(registryDir)\n\n // Create module entry\n const moduleEntry: RegistryModuleEntry = {\n moduleName,\n description: moduleMetadata.description || moduleMetadata.name,\n version: moduleMetadata.version,\n maintainer: (moduleMetadata as any).maintainer,\n tags: (moduleMetadata as any).tags,\n lastUpdated: new Date().toISOString()\n }\n this.moduleEntries.push(moduleEntry)\n this.log(` ✓ Module metadata: ${moduleEntry.version}`)\n\n // Read all feature files\n const featuresDir = path.join(registryDir, 'features')\n if (!fs.existsSync(featuresDir)) {\n if (this.config.strict) {\n const errorMessage = `❌ STRICT MODE VIOLATION: Module '${moduleName}' is missing required features directory!\\n Expected: ${featuresDir}\\n Fix: Create the features directory with at least one .yml file`\n this.log('')\n this.log('🚨 DEPLOYMENT FAILED - STRICT MODE VIOLATION 🚨')\n this.log(errorMessage)\n this.log('')\n console.error(`[module-registry] ${errorMessage}`)\n throw new Error(`Module Registry Strict Mode Violation: Module '${moduleName}' missing features directory`)\n }\n \n this.log(`⚠️ No features directory found in ${moduleName}/registry, skipping (strict mode disabled)`)\n return\n }\n\n // Find all feature YAML files\n const featureFiles = await glob(path.join(featuresDir, '*.{yml,yaml}').replace(/\\\\/g, '/'))\n \n this.log(` Found ${featureFiles.length} feature definitions`)\n\n // Process each feature file\n for (const featureFile of featureFiles) {\n await this.processFeatureFile(featureFile, moduleName, moduleMetadata)\n }\n }\n\n /**\n * Read module metadata from module.yml\n */\n private async readModuleMetadata(registryDir: string): Promise<ModuleMetadata> {\n const moduleFile = path.join(registryDir, 'module.yml')\n \n if (!fs.existsSync(moduleFile)) {\n return {\n name: 'unknown',\n version: '1.0.0',\n description: 'No module metadata available'\n }\n }\n\n try {\n const moduleData = yaml.load(fs.readFileSync(moduleFile, 'utf8')) as ModuleMetadata\n return moduleData\n } catch (error) {\n this.log(`⚠️ Error reading module metadata: ${(error as Error).message}`)\n return {\n name: 'unknown',\n version: '1.0.0', \n description: 'Error reading module metadata'\n }\n }\n }\n\n /**\n * Process a single feature file\n */\n private async processFeatureFile(featureFile: string, moduleName: string, moduleMetadata: ModuleMetadata): Promise<void> {\n const featureName = path.basename(featureFile, path.extname(featureFile))\n \n try {\n const featureData = yaml.load(fs.readFileSync(featureFile, 'utf8')) as RegistryFeature\n \n if (!featureData.name || !featureData.endpoints) {\n this.log(`⚠️ Invalid feature definition in ${featureFile}`)\n return\n }\n\n // Create registry entry\n const registryEntry: RegistryFeatureEntry = {\n moduleName,\n featureName,\n description: featureData.description || featureData.name,\n version: featureData.version || moduleMetadata.version,\n endpoints: featureData.endpoints || [],\n customPolicies: featureData.customPolicies || [],\n lastUpdated: new Date().toISOString()\n }\n\n this.featureEntries.push(registryEntry)\n \n this.log(` ✓ ${featureName}: ${registryEntry.endpoints.length} endpoints`)\n } catch (error) {\n this.log(`❌ Error processing feature ${featureFile}: ${(error as Error).message}`)\n }\n }\n\n /**\n * Ensure DynamoDB table exists using AWS SDK v3, then update registry data\n * This creates the table independently of any module's CloudFormation stack\n */\n private async ensureTableAndUpdateData(): Promise<void> {\n const totalEntries = this.moduleEntries.length + this.featureEntries.length\n if (this.config.skipDynamoDB || totalEntries === 0) {\n this.log('⏭️ Skipping DynamoDB operations')\n return\n }\n\n try {\n const { DynamoDBClient, CreateTableCommand, DescribeTableCommand, UpdateContinuousBackupsCommand, TagResourceCommand } = await import('@aws-sdk/client-dynamodb')\n \n const client = new DynamoDBClient({ region: this.config.region })\n const tableName = this.config.tableName!\n\n // First, check if table exists\n let tableExists = false\n try {\n await client.send(new DescribeTableCommand({ TableName: tableName }))\n tableExists = true\n this.log(`ℹ️ Table '${tableName}' already exists`)\n } catch (error: any) {\n if (error.name !== 'ResourceNotFoundException') {\n throw error\n }\n this.log(`📋 Table '${tableName}' does not exist, creating...`)\n }\n\n // Create table if it doesn't exist\n if (!tableExists) {\n const createTableParams = {\n TableName: tableName,\n BillingMode: 'PAY_PER_REQUEST' as const,\n AttributeDefinitions: [\n {\n AttributeName: 'pk',\n AttributeType: 'S' as const\n },\n {\n AttributeName: 'sk', \n AttributeType: 'S' as const\n },\n {\n AttributeName: 'gsi1pk',\n AttributeType: 'S' as const\n },\n {\n AttributeName: 'gsi1sk',\n AttributeType: 'S' as const\n }\n ],\n KeySchema: [\n {\n AttributeName: 'pk',\n KeyType: 'HASH' as const\n },\n {\n AttributeName: 'sk',\n KeyType: 'RANGE' as const\n }\n ],\n GlobalSecondaryIndexes: [\n {\n IndexName: 'GSI1',\n KeySchema: [\n {\n AttributeName: 'gsi1pk',\n KeyType: 'HASH' as const\n },\n {\n AttributeName: 'gsi1sk',\n KeyType: 'RANGE' as const\n }\n ],\n Projection: {\n ProjectionType: 'ALL' as const\n }\n }\n ]\n }\n\n await client.send(new CreateTableCommand(createTableParams))\n this.log(`✅ Created table '${tableName}'`)\n\n // Wait for table to be active before configuring additional features\n this.log('⏳ Waiting for table to be active...')\n let isActive = false\n while (!isActive) {\n await new Promise(resolve => setTimeout(resolve, 2000))\n const response = await client.send(new DescribeTableCommand({ TableName: tableName }))\n isActive = response.Table?.TableStatus === 'ACTIVE'\n }\n\n // Enable point-in-time recovery\n try {\n await client.send(new UpdateContinuousBackupsCommand({\n TableName: tableName,\n PointInTimeRecoverySpecification: {\n PointInTimeRecoveryEnabled: true\n }\n }))\n this.log('✅ Enabled point-in-time recovery')\n } catch (error) {\n this.log(`⚠️ Could not enable point-in-time recovery: ${(error as Error).message}`)\n }\n\n // Add tags\n try {\n // Get table ARN first\n const describeResponse = await client.send(new DescribeTableCommand({ TableName: tableName }))\n const tableArn = describeResponse.Table?.TableArn\n\n if (tableArn) {\n await client.send(new TagResourceCommand({\n ResourceArn: tableArn,\n Tags: [\n {\n Key: 'Module',\n Value: 'module-registry'\n },\n {\n Key: 'Purpose',\n Value: 'StateStore'\n }\n ]\n }))\n this.log('✅ Added compliance tags to table')\n }\n } catch (error) {\n this.log(`⚠️ Could not add tags: ${(error as Error).message}`)\n }\n }\n\n // Now update registry data\n await this.updateRegistryData()\n\n } catch (error) {\n this.log(`❌ Error ensuring table and updating data: ${(error as Error).message}`)\n // Don't fail deployment for registry operations\n }\n }\n\n /**\n * Update registry data in DynamoDB after deployment\n */\n private async updateRegistryData(): Promise<void> {\n const totalEntries = this.moduleEntries.length + this.featureEntries.length\n if (this.config.skipDynamoDB || totalEntries === 0) {\n this.log('⏭️ Skipping registry data update')\n return\n }\n\n this.log(`📝 Updating registry data in DynamoDB (${totalEntries} entries)...`)\n\n try {\n const { DynamoDBClient, BatchWriteItemCommand } = await import('@aws-sdk/client-dynamodb')\n const { marshall } = await import('@aws-sdk/util-dynamodb')\n \n const client = new DynamoDBClient({ region: this.config.region })\n\n // Combine all entries for batch processing\n const allItems: any[] = []\n\n // Add module entries\n for (const moduleEntry of this.moduleEntries) {\n allItems.push({\n pk: `MODULE#${moduleEntry.moduleName}`,\n sk: 'MODULE',\n gsi1pk: 'MODULES',\n gsi1sk: `MODULE#${moduleEntry.moduleName}`,\n itemType: 'module',\n ...moduleEntry\n })\n }\n\n // Add feature entries\n for (const featureEntry of this.featureEntries) {\n allItems.push({\n pk: `MODULE#${featureEntry.moduleName}`,\n sk: `FEATURE#${featureEntry.featureName}`,\n gsi1pk: `MODULE#${featureEntry.moduleName}`,\n gsi1sk: `FEATURE#${featureEntry.featureName}`,\n itemType: 'feature',\n ...featureEntry\n })\n }\n\n // Batch write all entries\n const batchSize = 25 // DynamoDB batch write limit\n const batches = this.chunkArray(allItems, batchSize)\n\n for (const batch of batches) {\n const putRequests = batch.map(item => ({\n PutRequest: {\n Item: marshall(item)\n }\n }))\n\n const command = new BatchWriteItemCommand({\n RequestItems: {\n [this.config.tableName!]: putRequests\n }\n })\n\n await client.send(command)\n }\n\n this.log(`✅ Registry data updated successfully: ${this.moduleEntries.length} modules, ${this.featureEntries.length} features`)\n } catch (error) {\n this.log(`❌ Error updating registry data: ${(error as Error).message}`)\n // Don't fail deployment for registry update errors\n }\n }\n\n /**\n * Utility function to chunk array into smaller arrays\n */\n private chunkArray<T>(array: T[], size: number): T[][] {\n const chunks: T[][] = []\n for (let i = 0; i < array.length; i += size) {\n chunks.push(array.slice(i, i + size))\n }\n return chunks\n }\n\n /**\n * Enhanced logging with color and prefix\n */\n private log(message: string): void {\n this.serverless.cli?.log?.(`${chalk.blue('[module-registry]')} ${message}`) ||\n console.log(`${chalk.blue('[module-registry]')} ${message}`)\n }\n}\n\n// Export service functions for direct import by handlers and other modules\nexport {\n listAllModules,\n getModuleFeatures, \n getFeatureDetails,\n getModuleMetadata,\n getAllEndpoints,\n createModuleRegistryLogger,\n type ModuleInfo,\n type FeatureInfo,\n type FeatureDetails,\n type EndpointInfo\n} from './service'\n\n// Export the plugin class as both default and named export for compatibility\nexport default ServerlessModuleRegistryPlugin\nexport { ServerlessModuleRegistryPlugin }\n\n// For CommonJS compatibility with Serverless Framework v4\n// @ts-ignore - Allow CommonJS export in ESM context for backward compatibility\nif (typeof module !== 'undefined' && module.exports) {\n module.exports = ServerlessModuleRegistryPlugin\n module.exports.default = ServerlessModuleRegistryPlugin\n}\n","/*\n * Module Registry Service Functions\n * -------------------------------------------\n * Service layer for querying module registry data from DynamoDB.\n * These functions can be imported directly from the plugin package.\n * \n * Usage:\n * import { listAllModules, getModuleFeatures } from 'serverless-plugin-module-registry'\n */\n\nimport { DynamoDBClient, QueryCommand, ScanCommand } from '@aws-sdk/client-dynamodb'\nimport { unmarshall } from '@aws-sdk/util-dynamodb'\n\n// Initialize DynamoDB client\nconst getDynamoClient = () => {\n return new DynamoDBClient({ \n region: process.env.AWS_REGION || 'us-east-1' \n })\n}\n\n// Configuration\nconst getTableName = () => {\n // Check for explicit table name first\n if (process.env.MODULE_REGISTRY_TABLE_NAME) {\n return process.env.MODULE_REGISTRY_TABLE_NAME\n }\n \n // Fallback to ARN prefix pattern (same as what the plugin uses)\n const service = process.env.SERVICE_NAME || 'api'\n const stage = process.env.STAGE || process.env.NODE_ENV || 'live'\n const defaultPrefix = `${service}-${stage}`\n \n return process.env.ARN_PREFIX \n ? `${process.env.ARN_PREFIX}-module-registry`\n : `${defaultPrefix}-module-registry`\n}\n\nconst GSI1_NAME = 'GSI1'\n\n// Types for better developer experience\nexport interface ModuleInfo {\n moduleName: string\n description: string\n version: string\n maintainer?: string\n tags: string[]\n lastUpdated: string\n}\n\nexport interface FeatureInfo {\n featureName: string\n description: string\n version: string\n endpoints: string[]\n customPolicies: any[]\n endpointCount: number\n lastUpdated: string\n}\n\nexport interface FeatureDetails extends FeatureInfo {\n moduleName: string\n policyCount: number\n}\n\nexport interface EndpointInfo {\n endpoint: string\n moduleName: string\n featureName: string\n featureDescription: string\n version: string\n}\n\n/**\n * Lists all deployed modules with basic metadata\n * @returns {Promise<ModuleInfo[]>} Array of module objects with clean data (no DDB internals)\n */\nexport const listAllModules = async (): Promise<ModuleInfo[]> => {\n const client = getDynamoClient()\n \n try {\n const command = new QueryCommand({\n TableName: getTableName(),\n IndexName: GSI1_NAME,\n KeyConditionExpression: 'gsi1pk = :pk',\n FilterExpression: 'itemType = :itemType',\n ExpressionAttributeValues: {\n ':pk': { S: 'MODULES' },\n ':itemType': { S: 'module' }\n }\n })\n\n const response = await client.send(command)\n \n if (!response.Items) {\n return []\n }\n\n // Transform DDB items to clean module objects\n const modules = response.Items.map(item => {\n const data = unmarshall(item)\n \n return {\n moduleName: data.moduleName,\n description: data.description,\n version: data.version,\n maintainer: data.maintainer,\n tags: data.tags || [],\n lastUpdated: data.lastUpdated\n }\n })\n\n return modules\n\n } catch (error) {\n throw new Error(`Failed to list modules: ${(error as Error).message}`)\n }\n}\n\n/**\n * Gets all features for a specific module\n * @param {string} moduleName - The module name (e.g., 'sign')\n * @returns {Promise<FeatureInfo[]>} Array of feature objects with endpoints\n */\nexport const getModuleFeatures = async (moduleName: string): Promise<FeatureInfo[]> => {\n if (!moduleName) {\n throw new Error('Module name is required')\n }\n\n const client = getDynamoClient()\n \n try {\n const command = new QueryCommand({\n TableName: getTableName(),\n KeyConditionExpression: 'pk = :pk AND begins_with(sk, :skPrefix)',\n FilterExpression: 'itemType = :itemType',\n ExpressionAttributeValues: {\n ':pk': { S: `MODULE#${moduleName}` },\n ':skPrefix': { S: 'FEATURE#' },\n ':itemType': { S: 'feature' }\n }\n })\n\n const response = await client.send(command)\n \n if (!response.Items) {\n return []\n }\n\n // Transform DDB items to clean feature objects\n const features = response.Items.map(item => {\n const data = unmarshall(item)\n \n return {\n featureName: data.featureName,\n description: data.description,\n version: data.version,\n endpoints: data.endpoints || [],\n customPolicies: data.customPolicies || [],\n endpointCount: (data.endpoints || []).length,\n lastUpdated: data.lastUpdated\n }\n })\n\n return features\n\n } catch (error) {\n throw new Error(`Failed to get module features: ${(error as Error).message}`)\n }\n}\n\n/**\n * Gets detailed information about a specific feature\n * @param {string} moduleName - The module name\n * @param {string} featureName - The feature name\n * @returns {Promise<FeatureDetails|null>} Feature details with endpoints and policies\n */\nexport const getFeatureDetails = async (moduleName: string, featureName: string): Promise<FeatureDetails | null> => {\n if (!moduleName) {\n throw new Error('Module name is required')\n }\n if (!featureName) {\n throw new Error('Feature name is required')\n }\n\n const client = getDynamoClient()\n \n try {\n const command = new QueryCommand({\n TableName: getTableName(),\n KeyConditionExpression: 'pk = :pk AND sk = :sk',\n ExpressionAttributeValues: {\n ':pk': { S: `MODULE#${moduleName}` },\n ':sk': { S: `FEATURE#${featureName}` }\n }\n })\n\n const response = await client.send(command)\n \n if (!response.Items || response.Items.length === 0) {\n return null\n }\n\n // Transform DDB item to clean feature object\n const data = unmarshall(response.Items[0])\n \n const featureDetails: FeatureDetails = {\n moduleName: data.moduleName,\n featureName: data.featureName,\n description: data.description,\n version: data.version,\n endpoints: data.endpoints || [],\n customPolicies: data.customPolicies || [],\n endpointCount: (data.endpoints || []).length,\n policyCount: (data.customPolicies || []).length,\n lastUpdated: data.lastUpdated\n }\n\n return featureDetails\n\n } catch (error) {\n throw new Error(`Failed to get feature details: ${(error as Error).message}`)\n }\n}\n\n/**\n * Gets module metadata (basic info without features)\n * @param {string} moduleName - The module name\n * @returns {Promise<ModuleInfo|null>} Module metadata or null if not found\n */\nexport const getModuleMetadata = async (moduleName: string): Promise<ModuleInfo | null> => {\n if (!moduleName) {\n throw new Error('Module name is required')\n }\n\n const client = getDynamoClient()\n \n try {\n const command = new QueryCommand({\n TableName: getTableName(),\n KeyConditionExpression: 'pk = :pk AND sk = :sk',\n ExpressionAttributeValues: {\n ':pk': { S: `MODULE#${moduleName}` },\n ':sk': { S: 'MODULE' }\n }\n })\n\n const response = await client.send(command)\n \n if (!response.Items || response.Items.length === 0) {\n return null\n }\n\n // Transform DDB item to clean module object\n const data = unmarshall(response.Items[0])\n \n const moduleMetadata: ModuleInfo = {\n moduleName: data.moduleName,\n description: data.description,\n version: data.version,\n maintainer: data.maintainer,\n tags: data.tags || [],\n lastUpdated: data.lastUpdated\n }\n\n return moduleMetadata\n\n } catch (error) {\n throw new Error(`Failed to get module metadata: ${(error as Error).message}`)\n }\n}\n\n/**\n * Gets endpoint summary across all modules for discovery\n * @returns {Promise<EndpointInfo[]>} Array of endpoints with module/feature context\n */\nexport const getAllEndpoints = async (): Promise<EndpointInfo[]> => {\n const client = getDynamoClient()\n \n try {\n const command = new ScanCommand({\n TableName: getTableName(),\n FilterExpression: 'itemType = :itemType',\n ExpressionAttributeValues: {\n ':itemType': { S: 'feature' }\n }\n })\n\n const response = await client.send(command)\n \n if (!response.Items) {\n return []\n }\n\n // Transform to flat endpoint list with context\n const endpoints: EndpointInfo[] = []\n \n for (const item of response.Items) {\n const data = unmarshall(item)\n \n if (data.endpoints && Array.isArray(data.endpoints)) {\n for (const endpoint of data.endpoints) {\n endpoints.push({\n endpoint,\n moduleName: data.moduleName,\n featureName: data.featureName,\n featureDescription: data.description,\n version: data.version\n })\n }\n }\n }\n\n return endpoints\n\n } catch (error) {\n throw new Error(`Failed to get all endpoints: ${(error as Error).message}`)\n }\n}\n\n/**\n * Utility function to create logger compatible with service-style.md\n * This can be used in handlers that import the service functions\n */\nexport const createModuleRegistryLogger = (context?: string) => {\n const logContext = context || 'module-registry'\n \n return {\n info: (message: string, data?: any) => {\n console.log(`[${logContext}] INFO: ${message}`, data ? JSON.stringify(data) : '')\n },\n error: (message: string, data?: any) => {\n console.error(`[${logContext}] ERROR: ${message}`, data ? JSON.stringify(data) : '')\n },\n warn: (message: string, data?: any) => {\n console.warn(`[${logContext}] WARN: ${message}`, data ? JSON.stringify(data) : '')\n }\n }\n}\n"],"mappings":"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","names":["path","fs","yaml","DynamoDBClient","chalk"]}
1
+ {"version":3,"sources":["../node_modules/balanced-match/index.js","../node_modules/glob/node_modules/brace-expansion/index.js","../src/index.ts","../src/utils/path-utils.ts","../src/core/config.ts","../src/core/dynamodb.ts","../src/core/policy-generator.ts","../src/core/registry-processor.ts","../node_modules/glob/node_modules/minimatch/src/index.ts","../node_modules/glob/node_modules/minimatch/src/assert-valid-pattern.ts","../node_modules/glob/node_modules/minimatch/src/brace-expressions.ts","../node_modules/glob/node_modules/minimatch/src/unescape.ts","../node_modules/glob/node_modules/minimatch/src/ast.ts","../node_modules/glob/node_modules/minimatch/src/escape.ts","../node_modules/glob/src/glob.ts","../node_modules/path-scurry/node_modules/lru-cache/src/index.ts","../node_modules/path-scurry/src/index.ts","../node_modules/minipass/src/index.ts","../node_modules/glob/src/pattern.ts","../node_modules/glob/src/ignore.ts","../node_modules/glob/src/processor.ts","../node_modules/glob/src/walker.ts","../node_modules/glob/src/has-magic.ts","../node_modules/glob/src/index.ts","../src/utils/feature-id.ts","../src/core/abac-role-scanner.ts","../src/commands/registry-generate.ts","../src/ai/generator.ts","../src/commands/package-generate.ts","../src/core/plugin-logger.ts","../src/core/abac-metadata-writer.ts","../src/core/plugin-orchestrator.ts","../src/core/commands.ts","../src/service.js"],"sourcesContent":["'use strict';\nmodule.exports = balanced;\nfunction balanced(a, b, str) {\n if (a instanceof RegExp) a = maybeMatch(a, str);\n if (b instanceof RegExp) b = maybeMatch(b, str);\n\n var r = range(a, b, str);\n\n return r && {\n start: r[0],\n end: r[1],\n pre: str.slice(0, r[0]),\n body: str.slice(r[0] + a.length, r[1]),\n post: str.slice(r[1] + b.length)\n };\n}\n\nfunction maybeMatch(reg, str) {\n var m = str.match(reg);\n return m ? m[0] : null;\n}\n\nbalanced.range = range;\nfunction range(a, b, str) {\n var begs, beg, left, right, result;\n var ai = str.indexOf(a);\n var bi = str.indexOf(b, ai + 1);\n var i = ai;\n\n if (ai >= 0 && bi > 0) {\n if(a===b) {\n return [ai, bi];\n }\n begs = [];\n left = str.length;\n\n while (i >= 0 && !result) {\n if (i == ai) {\n begs.push(i);\n ai = str.indexOf(a, i + 1);\n } else if (begs.length == 1) {\n result = [ begs.pop(), bi ];\n } else {\n beg = begs.pop();\n if (beg < left) {\n left = beg;\n right = bi;\n }\n\n bi = str.indexOf(b, i + 1);\n }\n\n i = ai < bi && ai >= 0 ? ai : bi;\n }\n\n if (begs.length) {\n result = [ left, right ];\n }\n }\n\n return result;\n}\n","var balanced = require('balanced-match');\n\nmodule.exports = expandTop;\n\nvar escSlash = '\\0SLASH'+Math.random()+'\\0';\nvar escOpen = '\\0OPEN'+Math.random()+'\\0';\nvar escClose = '\\0CLOSE'+Math.random()+'\\0';\nvar escComma = '\\0COMMA'+Math.random()+'\\0';\nvar escPeriod = '\\0PERIOD'+Math.random()+'\\0';\n\nfunction numeric(str) {\n return parseInt(str, 10) == str\n ? parseInt(str, 10)\n : str.charCodeAt(0);\n}\n\nfunction escapeBraces(str) {\n return str.split('\\\\\\\\').join(escSlash)\n .split('\\\\{').join(escOpen)\n .split('\\\\}').join(escClose)\n .split('\\\\,').join(escComma)\n .split('\\\\.').join(escPeriod);\n}\n\nfunction unescapeBraces(str) {\n return str.split(escSlash).join('\\\\')\n .split(escOpen).join('{')\n .split(escClose).join('}')\n .split(escComma).join(',')\n .split(escPeriod).join('.');\n}\n\n\n// Basically just str.split(\",\"), but handling cases\n// where we have nested braced sections, which should be\n// treated as individual members, like {a,{b,c},d}\nfunction parseCommaParts(str) {\n if (!str)\n return [''];\n\n var parts = [];\n var m = balanced('{', '}', str);\n\n if (!m)\n return str.split(',');\n\n var pre = m.pre;\n var body = m.body;\n var post = m.post;\n var p = pre.split(',');\n\n p[p.length-1] += '{' + body + '}';\n var postParts = parseCommaParts(post);\n if (post.length) {\n p[p.length-1] += postParts.shift();\n p.push.apply(p, postParts);\n }\n\n parts.push.apply(parts, p);\n\n return parts;\n}\n\nfunction expandTop(str) {\n if (!str)\n return [];\n\n // I don't know why Bash 4.3 does this, but it does.\n // Anything starting with {} will have the first two bytes preserved\n // but *only* at the top level, so {},a}b will not expand to anything,\n // but a{},b}c will be expanded to [a}c,abc].\n // One could argue that this is a bug in Bash, but since the goal of\n // this module is to match Bash's rules, we escape a leading {}\n if (str.substr(0, 2) === '{}') {\n str = '\\\\{\\\\}' + str.substr(2);\n }\n\n return expand(escapeBraces(str), true).map(unescapeBraces);\n}\n\nfunction embrace(str) {\n return '{' + str + '}';\n}\nfunction isPadded(el) {\n return /^-?0\\d/.test(el);\n}\n\nfunction lte(i, y) {\n return i <= y;\n}\nfunction gte(i, y) {\n return i >= y;\n}\n\nfunction expand(str, isTop) {\n var expansions = [];\n\n var m = balanced('{', '}', str);\n if (!m) return [str];\n\n // no need to expand pre, since it is guaranteed to be free of brace-sets\n var pre = m.pre;\n var post = m.post.length\n ? expand(m.post, false)\n : [''];\n\n if (/\\$$/.test(m.pre)) { \n for (var k = 0; k < post.length; k++) {\n var expansion = pre+ '{' + m.body + '}' + post[k];\n expansions.push(expansion);\n }\n } else {\n var isNumericSequence = /^-?\\d+\\.\\.-?\\d+(?:\\.\\.-?\\d+)?$/.test(m.body);\n var isAlphaSequence = /^[a-zA-Z]\\.\\.[a-zA-Z](?:\\.\\.-?\\d+)?$/.test(m.body);\n var isSequence = isNumericSequence || isAlphaSequence;\n var isOptions = m.body.indexOf(',') >= 0;\n if (!isSequence && !isOptions) {\n // {a},b}\n if (m.post.match(/,(?!,).*\\}/)) {\n str = m.pre + '{' + m.body + escClose + m.post;\n return expand(str);\n }\n return [str];\n }\n\n var n;\n if (isSequence) {\n n = m.body.split(/\\.\\./);\n } else {\n n = parseCommaParts(m.body);\n if (n.length === 1) {\n // x{{a,b}}y ==> x{a}y x{b}y\n n = expand(n[0], false).map(embrace);\n if (n.length === 1) {\n return post.map(function(p) {\n return m.pre + n[0] + p;\n });\n }\n }\n }\n\n // at this point, n is the parts, and we know it's not a comma set\n // with a single entry.\n var N;\n\n if (isSequence) {\n var x = numeric(n[0]);\n var y = numeric(n[1]);\n var width = Math.max(n[0].length, n[1].length)\n var incr = n.length == 3\n ? Math.abs(numeric(n[2]))\n : 1;\n var test = lte;\n var reverse = y < x;\n if (reverse) {\n incr *= -1;\n test = gte;\n }\n var pad = n.some(isPadded);\n\n N = [];\n\n for (var i = x; test(i, y); i += incr) {\n var c;\n if (isAlphaSequence) {\n c = String.fromCharCode(i);\n if (c === '\\\\')\n c = '';\n } else {\n c = String(i);\n if (pad) {\n var need = width - c.length;\n if (need > 0) {\n var z = new Array(need + 1).join('0');\n if (i < 0)\n c = '-' + z + c.slice(1);\n else\n c = z + c;\n }\n }\n }\n N.push(c);\n }\n } else {\n N = [];\n\n for (var j = 0; j < n.length; j++) {\n N.push.apply(N, expand(n[j], false));\n }\n }\n\n for (var j = 0; j < N.length; j++) {\n for (var k = 0; k < post.length; k++) {\n var expansion = pre + N[j] + post[k];\n if (!isTop || isSequence || expansion)\n expansions.push(expansion);\n }\n }\n }\n\n return expansions;\n}\n\n","/**\n * Serverless Module Registry Plugin\n * -------------------------------------------\n * A Serverless Framework plugin that scans module registry files,\n * discovers endpoints, and stores module feature mappings in DynamoDB\n * for cross-module discovery and access control.\n *\n * ✓ Compatible with Serverless Framework v2, v3 & v4\n * ✓ Independent and robust - includes variable resolution\n * ✓ Scans registry/ folders in each module\n * ✓ Creates/updates DynamoDB ModuleRegistry table\n * ✓ Supports CloudFormation intrinsic functions in YAML\n * ✓ Cross-version compatible variable resolution\n * ✓ nanoid-based feature ID generation (8-char opaque IDs) for IAM compliance\n * ✓ ABAC (Attribute-Based Access Control) with secure, non-descriptive identifiers\n * ✓ Auto-updating YAML files with missing feature IDs for persistence\n * ✓ Feature count enforcement (max 20 per module) for IAM policy compliance\n *\n * ## Configuration\n *\n * ```yaml\n * custom:\n * moduleRegistry:\n * maxFeaturesPerModule: 20 # Max features per module (default: 20)\n * tableName: ModuleRegistry # DynamoDB table name\n * region: us-east-1 # AWS region\n * policyPrefix: api # IAM policy prefix\n * ```\n *\n * ## Feature Limits\n *\n * **Why 20 features max per module?**\n * - AWS IAM managed policies have a 10,240 character limit\n * - Each feature adds ~322 characters to the policy\n * - 20 features = ~6,440 chars (safe margin for growth)\n * - Prevents IAM policy deployment failures\n * - Encourages good module architecture (single responsibility)\n *\n * **If you exceed 20 features:**\n * - Split your module: `users`, `users-admin`, `users-reports`\n * - Group related features with multiple endpoints\n * - Configure custom limit (not recommended, may cause failures)\n */\n\nimport fs from 'fs'\nimport path from 'path'\nimport { IAMClient, ListEntitiesForPolicyCommand, ListPoliciesCommand } from '@aws-sdk/client-iam'\nimport { STSClient, GetCallerIdentityCommand } from '@aws-sdk/client-sts'\nimport { APIGatewayClient, GetRestApisCommand } from '@aws-sdk/client-api-gateway'\nimport { normalizePath } from './utils/path-utils.js'\nimport { createLogger } from '@hyperdrive.bot/serverless-utils'\nimport { ConfigManager } from './core/config.js'\nimport { DynamoDBManager } from './core/dynamodb.js'\nimport { PolicyGenerator } from './core/policy-generator.js'\nimport { RegistryProcessor } from './core/registry-processor.js'\nimport { RegistryGenerateCommand } from './commands/registry-generate.js'\nimport { PackageGenerateCommand } from './commands/package-generate.js'\nimport { PluginLogger } from './core/plugin-logger.js'\nimport { PluginOrchestrator } from './core/plugin-orchestrator.js'\nimport { PLUGIN_COMMANDS, PLUGIN_HOOKS } from './core/commands.js'\nimport type {\n RegistryModuleEntry,\n RegistryFeatureEntry,\n AbacRoleMetadata,\n Logger,\n GenerateRegistryOptions,\n GeneratePackageOptions\n} from './types.js'\n\nclass ServerlessModuleRegistryPlugin {\n serverless: any\n options: Record<string, any>\n hooks: Record<string, () => Promise<void>>\n commands: Record<string, any>\n\n private configManager: ConfigManager\n private orchestrator: PluginOrchestrator | null = null\n private dynamoManager: DynamoDBManager | null = null\n private policyGenerator: PolicyGenerator | null = null\n private registryProcessor: RegistryProcessor | null = null\n private moduleEntries: RegistryModuleEntry[] = []\n private featureEntries: RegistryFeatureEntry[] = []\n private abacRoleMetadata: AbacRoleMetadata[] = []\n private pluginLogger: PluginLogger\n private logger: Logger\n private iamClient: IAMClient | null = null\n private stsClient: STSClient | null = null\n private apiGatewayClient: APIGatewayClient | null = null\n\n constructor(serverless: any, options: Record<string, any>) {\n this.serverless = serverless\n this.options = options\n\n // Initialize configuration manager\n this.configManager = new ConfigManager(serverless, options)\n\n // Initialize logger and plugin logger\n this.logger = createLogger(serverless, 'module-registry') as unknown as Logger\n this.pluginLogger = new PluginLogger(this.logger)\n\n // Register commands and hooks\n this.commands = PLUGIN_COMMANDS\n this.hooks = this.createHooks()\n\n // Log initial configuration\n this.pluginLogger.logInitialConfiguration(\n this.configManager.config,\n this.configManager.isModuleSpecificDeployment()\n )\n }\n\n private createHooks(): Record<string, () => Promise<void>> {\n const hooks: Record<string, () => Promise<void>> = {}\n\n Object.entries(PLUGIN_HOOKS).forEach(([hookName, methodName]) => {\n hooks[hookName] = (this as any)[methodName].bind(this)\n })\n\n return hooks\n }\n\n\n /**\n * Initialize dependent managers\n */\n private initializeManagers(): void {\n const config = this.configManager.config\n\n this.dynamoManager = new DynamoDBManager(config, this.logger)\n this.policyGenerator = new PolicyGenerator(config.policyPrefix!, this.logger)\n\n this.registryProcessor = new RegistryProcessor(\n this.serverless.config.servicePath || process.cwd(),\n this.configManager,\n this.logger,\n this.serverless\n )\n\n // Initialize orchestrator\n this.orchestrator = new PluginOrchestrator(this.serverless, config, this.logger)\n this.orchestrator.initializeManagers()\n\n // Initialize IAM, STS, and API Gateway clients for policy safety checks\n this.iamClient = new IAMClient({\n region: config.region\n })\n this.stsClient = new STSClient({\n region: config.region\n })\n this.apiGatewayClient = new APIGatewayClient({\n region: config.region\n })\n }\n\n /**\n * Process module registry with robust variable resolution (main entry point)\n */\n private async processModuleRegistryWithVariableResolution(): Promise<void> {\n this.pluginLogger.trackHookCall('before:package:initialize')\n\n this.logger.info('🚀 Starting module registry processing with variable resolution...')\n this.logger.info(` Serverless version: ${this.configManager.version}`)\n\n try {\n // First, resolve variables in the config\n await this.configManager.resolveVariables()\n this.configManager.validate()\n\n // Initialize managers after config resolution\n this.initializeManagers()\n\n // Debug resolved configuration\n const config = this.configManager.config\n this.logger.info('📊 Module Registry configuration (resolved):')\n this.logger.info(` Table: ${config.tableName}`)\n this.logger.info(` Region: ${config.region}`)\n this.logger.info(` Policy Prefix: ${config.policyPrefix}`)\n\n // Clean up obsolete files before generating new ones\n await this.cleanupObsoleteFiles()\n\n // Process module registry files and collect raw data\n const { moduleEntries, featureEntries, abacRoleMetadata } = await this.registryProcessor!.processModuleRegistry()\n this.moduleEntries = moduleEntries\n this.featureEntries = featureEntries\n this.abacRoleMetadata = abacRoleMetadata\n\n // Generate CloudFormation resources for managed policies (BEFORE deployment)\n await this.generatePolicyResources()\n\n // Generate service file for imports\n await this.generateServiceFile()\n\n this.logger.info('✅ Module registry processing complete with variable resolution')\n } catch (error) {\n this.logger.error('❌ Module registry processing failed', error)\n throw error\n }\n }\n\n /**\n * Check if a specific IAM policy is in use by any entities\n */\n private async isPolicyInUse(policyArn: string): Promise<{inUse: boolean, entities: string[]}> {\n if (!this.iamClient) {\n this.logger.warning('IAM client not initialized - skipping policy usage check')\n return { inUse: false, entities: [] }\n }\n\n try {\n const command = new ListEntitiesForPolicyCommand({\n PolicyArn: policyArn\n // EntityFilter defaults to all entity types when not specified\n })\n\n const response = await this.iamClient.send(command)\n \n const entities: string[] = []\n \n // Check all entity types\n if (response.PolicyUsers && response.PolicyUsers.length > 0) {\n entities.push(...response.PolicyUsers.map(user => `User: ${user.UserName}`))\n }\n if (response.PolicyGroups && response.PolicyGroups.length > 0) {\n entities.push(...response.PolicyGroups.map(group => `Group: ${group.GroupName}`))\n }\n if (response.PolicyRoles && response.PolicyRoles.length > 0) {\n entities.push(...response.PolicyRoles.map(role => `Role: ${role.RoleName}`))\n }\n\n return {\n inUse: entities.length > 0,\n entities\n }\n } catch (error: any) {\n // If policy doesn't exist, it's not in use\n if (error.name === 'NoSuchEntityException') {\n return { inUse: false, entities: [] }\n }\n \n this.logger.warning(`Could not check policy usage for ${policyArn}: ${error.message}`)\n // On error, assume it might be in use for safety\n return { inUse: true, entities: [`Error checking policy: ${error.message}`] }\n }\n }\n\n /**\n * Check all policies that would be affected by cleanup before proceeding\n */\n private async checkPolicySafetyBeforeCleanup(): Promise<{safe: boolean, policiesInUse: Array<{arn: string, entities: string[]}>}> {\n this.logger.info('🔍 Checking policy safety before cleanup...')\n \n const policiesInUse: Array<{arn: string, entities: string[]}> = []\n \n if (!this.iamClient) {\n this.logger.warning('IAM client not available - proceeding without policy safety checks')\n return { safe: true, policiesInUse: [] }\n }\n\n // Get list of policies that might be affected\n const policiesToCheck = await this.identifyPoliciesForSafetyCheck()\n \n if (policiesToCheck.length === 0) {\n this.logger.info('✅ No existing policies to check - cleanup is safe')\n return { safe: true, policiesInUse: [] }\n }\n\n this.logger.info(` Checking ${policiesToCheck.length} policies...`)\n\n // Check each policy\n for (const policyArn of policiesToCheck) {\n const { inUse, entities } = await this.isPolicyInUse(policyArn)\n \n if (inUse) {\n policiesInUse.push({ arn: policyArn, entities })\n this.logger.warning(` ⚠️ Policy in use: ${policyArn}`)\n entities.forEach(entity => this.logger.warning(` - Attached to: ${entity}`))\n } else {\n this.logger.info(` ✅ Policy safe to delete: ${policyArn}`)\n }\n }\n\n const safe = policiesInUse.length === 0\n\n if (!safe) {\n this.logger.error('🚨 SAFETY CHECK FAILED - Some policies are in use!')\n this.logger.error(' The following policies cannot be deleted:')\n policiesInUse.forEach(policy => {\n this.logger.error(` - ${policy.arn}`)\n policy.entities.forEach(entity => this.logger.error(` Attached to: ${entity}`))\n })\n this.logger.error('')\n this.logger.error(' To proceed safely:')\n this.logger.error(' 1. Detach policies from the entities listed above')\n this.logger.error(' 2. Or run without --force to skip policy cleanup')\n this.logger.error(' 3. Or set skipPolicies: true in configuration')\n } else {\n this.logger.info('✅ Policy safety check passed - all policies are safe to clean up')\n }\n\n return { safe, policiesInUse }\n }\n\n /**\n * Identify existing policies that would be affected by cleanup\n */\n private async identifyPoliciesForSafetyCheck(): Promise<string[]> {\n const policies: string[] = []\n \n // Get AWS account ID for policy ARN construction\n const accountId = await this.getAwsAccountId()\n if (!accountId) {\n this.logger.warning('Could not determine AWS account ID - skipping policy identification')\n return []\n }\n\n // Get policy prefix from config\n const policyPrefix = this.configManager.config.policyPrefix\n if (!policyPrefix) {\n this.logger.warning('No policy prefix configured - skipping policy identification')\n return []\n }\n\n // Method 1: Add policies from current feature entries\n // Generate policy names on-demand since they haven't been set yet\n const stage = this.serverless.service.provider?.stage || 'dev'\n for (const feature of this.featureEntries) {\n // Generate the policy name that would be created for this feature\n const policyName = this.policyGenerator!.generatePolicyName(\n feature.moduleName,\n feature.featureName,\n stage\n )\n const policyArn = `arn:aws:iam::${accountId}:policy/${policyName}`\n policies.push(policyArn)\n }\n\n // Method 2: Scan for ALL existing policies with our prefix pattern\n // This catches policies from previous generations that might still exist\n try {\n const existingPolicies = await this.scanExistingPoliciesByPrefix(policyPrefix, accountId)\n policies.push(...existingPolicies)\n \n this.logger.info(` Found ${existingPolicies.length} existing policies with prefix \"${policyPrefix}\"`)\n } catch (error: any) {\n this.logger.warning(`Could not scan for existing policies: ${error.message}`)\n this.logger.warning(' Will only check policies from current feature definitions')\n }\n\n // Remove duplicates and return\n return [...new Set(policies)]\n }\n\n /**\n * Scan for existing policies that match our naming prefix\n */\n private async scanExistingPoliciesByPrefix(policyPrefix: string, accountId: string): Promise<string[]> {\n if (!this.iamClient) {\n throw new Error('IAM client not initialized')\n }\n\n const matchingPolicies: string[] = []\n let marker: string | undefined\n\n try {\n do {\n const command = new ListPoliciesCommand({\n Scope: 'Local', // Only customer-managed policies\n OnlyAttached: false, // Include unattached policies too\n Marker: marker,\n MaxItems: 100\n })\n\n const response = await this.iamClient.send(command)\n\n if (response.Policies) {\n for (const policy of response.Policies) {\n if (policy.PolicyName && policy.Arn) {\n // Check if policy name starts with our prefix\n if (policy.PolicyName.startsWith(policyPrefix) && !policy.PolicyName.startsWith(`${policyPrefix}-lambda`)) {\n matchingPolicies.push(policy.Arn)\n this.logger.info(` 📋 Found policy: ${policy.PolicyName}`)\n }\n }\n }\n }\n\n marker = response.Marker\n } while (marker) // Continue if there are more pages\n\n } catch (error: any) {\n if (error.name === 'AccessDenied') {\n this.logger.warning('IAM ListPolicies permission denied - falling back to feature-only checking')\n this.logger.warning('Consider granting iam:ListPolicies permission for comprehensive safety checks')\n return []\n }\n throw error\n }\n\n return matchingPolicies\n }\n\n /**\n * Get AWS account ID for policy ARN construction\n */\n private async getAwsAccountId(): Promise<string | null> {\n try {\n // First, try environment variable for performance\n if (process.env.AWS_ACCOUNT_ID) {\n return process.env.AWS_ACCOUNT_ID\n }\n\n // Use STS GetCallerIdentity for reliable account ID retrieval\n if (this.stsClient) {\n try {\n const command = new GetCallerIdentityCommand({})\n const response = await this.stsClient.send(command)\n if (response.Account) {\n return response.Account\n }\n } catch (stsError: any) {\n this.logger.warning(`STS GetCallerIdentity failed: ${stsError.message}`)\n }\n }\n\n // Fallback: Try to get account ID from serverless provider config\n if (this.serverless?.service?.provider?.compiledCloudFormationTemplate?.Resources) {\n const template = this.serverless.service.provider.compiledCloudFormationTemplate\n // Account ID is often available in resource ARNs\n for (const resource of Object.values(template.Resources)) {\n const resourceStr = JSON.stringify(resource)\n const accountMatch = resourceStr.match(/\"arn:aws:iam::(\\d{12}):/)\n if (accountMatch) {\n return accountMatch[1]\n }\n }\n }\n\n this.logger.warning('Could not determine AWS account ID from any source')\n return null\n } catch (error) {\n this.logger.warning(`Could not determine AWS account ID: ${(error as Error).message}`)\n return null\n }\n }\n\n /**\n * Fetch API Gateway REST API by name pattern: ${stage}-${service}\n * Returns the API Gateway ID if found, null otherwise\n */\n private async getApiGatewayId(stage: string, service: string): Promise<string | null> {\n try {\n if (!this.apiGatewayClient) {\n this.logger.warning('API Gateway client not initialized')\n return null\n }\n\n const expectedName = `${stage}-${service}`\n this.logger.info(`🔍 Looking for API Gateway with name: ${expectedName}`)\n\n const command = new GetRestApisCommand({})\n const response = await this.apiGatewayClient.send(command)\n\n if (response.items) {\n for (const api of response.items) {\n console.log('api', api)\n if (api.name === expectedName && api.id) {\n this.logger.info(`✅ Found API Gateway: ${expectedName} (ID: ${api.id})`)\n return api.id\n }\n }\n }\n\n this.logger.info(`❌ API Gateway not found: ${expectedName}`)\n return null\n } catch (error) {\n this.logger.warning(`Could not fetch API Gateway ${stage}-${service}: ${(error as Error).message}`)\n return null\n }\n }\n\n /**\n * Clean up obsolete registry and service files when appropriate\n */\n private async cleanupObsoleteFiles(): Promise<void> {\n try {\n // Clean up obsolete service files when policies are skipped\n if (this.configManager.config.skipPolicies) {\n await this.cleanupPolicyGeneratedFiles()\n }\n\n // Clean up ALL registry files when force mode is enabled (clean slate approach)\n const forceMode = this.options.force || this.options.f\n if (forceMode) {\n // CRITICAL SAFETY CHECK: Verify no policies are in use before cleanup\n const safetyCheck = await this.checkPolicySafetyBeforeCleanup()\n \n if (!safetyCheck.safe) {\n this.logger.warning('🚨 FORCE MODE: Proceeding despite policies in use!')\n this.logger.warning(' Policy safety check failed but --force flag is enabled')\n this.logger.warning(' WARNING: This may cause issues with attached IAM entities')\n safetyCheck.policiesInUse.forEach(policy => {\n this.logger.warning(` - ${policy.arn}`)\n policy.entities.forEach(entity => this.logger.warning(` Attached to: ${entity}`))\n })\n }\n\n await this.cleanSlateRegistryFiles()\n }\n } catch (error) {\n this.logger.error('❌ Error during cleanup', error)\n throw error // Re-throw to stop deployment when safety checks fail\n }\n }\n\n /**\n * Clean up policy-generated service files when policies are disabled\n */\n private async cleanupPolicyGeneratedFiles(): Promise<void> {\n this.logger.info('🧹 Cleaning up policy-generated service files (policies disabled)...')\n \n // Safety check: Even when policies are disabled, check if any existing policies are in use\n const safetyCheck = await this.checkPolicySafetyBeforeCleanup()\n \n if (!safetyCheck.safe) {\n this.logger.warning('⚠️ WARNING: Some policies are still in use, but skipPolicies is enabled')\n this.logger.warning(' Policy-generated service files will not be cleaned up for safety')\n safetyCheck.policiesInUse.forEach(policy => {\n this.logger.warning(` - ${policy.arn} (used by: ${policy.entities.join(', ')})`)\n })\n return\n }\n \n const modulesDir = path.resolve(this.serverless.config.servicePath || process.cwd(), 'serverless/modules')\n if (!fs.existsSync(modulesDir)) return\n\n const moduleNames = fs.readdirSync(modulesDir)\n .filter(name => fs.statSync(path.join(modulesDir, name)).isDirectory())\n\n let cleanedFiles = 0\n\n for (const moduleName of moduleNames) {\n const moduleDir = path.join(modulesDir, moduleName)\n const generatedDir = path.join(moduleDir, 'generated')\n \n if (fs.existsSync(generatedDir)) {\n // Remove generated policy files\n const policyFiles = fs.readdirSync(generatedDir)\n .filter(file => file.includes('policy') || file.includes('iam'))\n \n for (const file of policyFiles) {\n const filePath = path.join(generatedDir, file)\n fs.unlinkSync(filePath)\n cleanedFiles++\n }\n \n // Remove generated directory if empty\n if (fs.readdirSync(generatedDir).length === 0) {\n fs.rmdirSync(generatedDir)\n }\n }\n }\n\n // Clean up module-registry service files\n const nodeModulesDir = path.join(this.serverless.config.servicePath || process.cwd(), 'node_modules')\n const moduleRegistryDir = path.join(nodeModulesDir, 'module-registry')\n \n if (fs.existsSync(moduleRegistryDir)) {\n const serviceFiles = ['service.js', 'service.d.ts', 'env.js', 'package.json']\n for (const file of serviceFiles) {\n const filePath = path.join(moduleRegistryDir, file)\n if (fs.existsSync(filePath)) {\n fs.unlinkSync(filePath)\n cleanedFiles++\n }\n }\n \n if (fs.readdirSync(moduleRegistryDir).length === 0) {\n fs.rmdirSync(moduleRegistryDir)\n }\n }\n\n if (cleanedFiles > 0) {\n this.logger.info(`✅ Cleaned up ${cleanedFiles} obsolete policy/service files`)\n } else {\n this.logger.info('✅ No obsolete policy/service files to clean up')\n }\n }\n\n /**\n * Clean up obsolete registry files specifically for registry generation command\n */\n private async cleanupObsoleteFilesForGeneration(force: boolean = false): Promise<void> {\n try {\n // Perform CRITICAL SAFETY CHECK: Verify no policies are in use before cleanup\n const safetyCheck = await this.checkPolicySafetyBeforeCleanup()\n \n if (!safetyCheck.safe) {\n if (force) {\n this.logger.warning('🚨 FORCE MODE: Proceeding despite policies in use!')\n this.logger.warning(' Policy safety check failed but --force flag is enabled')\n this.logger.warning(' WARNING: This may cause issues with attached IAM entities')\n safetyCheck.policiesInUse.forEach(policy => {\n this.logger.warning(` - ${policy.arn}`)\n policy.entities.forEach(entity => this.logger.warning(` Attached to: ${entity}`))\n })\n } else {\n this.logger.error('🚨 CLEANUP ABORTED - Policies are in use!')\n this.logger.error(' Registry generation cannot proceed while policies are attached to IAM entities.')\n this.logger.error(' Either detach the policies or run without --force flag.')\n throw new Error('Policy safety check failed: Cannot delete registry files while policies are in use')\n }\n }\n\n // If safe or force is enabled, proceed with targeted cleanup for the specific module\n await this.cleanSlateRegistryFiles()\n } catch (error) {\n this.logger.error('❌ Error during registry generation cleanup', error)\n throw error // Re-throw to stop generation when safety checks fail\n }\n }\n\n /**\n * Clean slate approach: Delete ALL existing registry files when --force is used\n */\n private async cleanSlateRegistryFiles(): Promise<void> {\n this.logger.info('🧹 Clean slate: Removing ALL existing registry files (--force mode)...')\n \n const modulesDir = path.resolve(this.serverless.config.servicePath || process.cwd(), 'serverless/modules')\n if (!fs.existsSync(modulesDir)) return\n\n // Get list of modules to process based on deployment mode\n const targetModuleName = this.configManager.isModuleSpecificDeployment() \n ? this.configManager.extractModuleName() \n : null\n\n const moduleNames = targetModuleName \n ? [targetModuleName].filter(name => fs.existsSync(path.join(modulesDir, name)))\n : fs.readdirSync(modulesDir).filter(name => fs.statSync(path.join(modulesDir, name)).isDirectory())\n\n let cleanedFiles = 0\n let cleanedDirectories = 0\n\n for (const moduleName of moduleNames) {\n const registryDir = path.join(modulesDir, moduleName, 'registry')\n \n if (!fs.existsSync(registryDir)) continue\n\n this.logger.info(` 🗑️ Cleaning module: ${moduleName}`)\n\n // Remove the entire registry directory and recreate it\n try {\n // First, recursively remove all files in registry directory\n const removeDirectory = (dirPath: string) => {\n if (!fs.existsSync(dirPath)) return\n\n const files = fs.readdirSync(dirPath)\n for (const file of files) {\n const fullPath = path.join(dirPath, file)\n const stat = fs.statSync(fullPath)\n \n if (stat.isDirectory()) {\n removeDirectory(fullPath)\n } else {\n fs.unlinkSync(fullPath)\n cleanedFiles++\n }\n }\n \n // Remove empty directory\n try {\n fs.rmdirSync(dirPath)\n if (dirPath !== registryDir) cleanedDirectories++\n } catch (e) {\n // Directory might not be empty, that's okay\n }\n }\n\n // Remove everything inside registry directory\n const registryContents = fs.readdirSync(registryDir)\n for (const item of registryContents) {\n const itemPath = path.join(registryDir, item)\n const stat = fs.statSync(itemPath)\n \n if (stat.isDirectory()) {\n removeDirectory(itemPath)\n } else {\n fs.unlinkSync(itemPath)\n cleanedFiles++\n }\n }\n\n this.logger.info(` ✅ Cleaned registry directory for ${moduleName}`)\n } catch (error) {\n this.logger.warning(` ⚠️ Error cleaning ${moduleName}: ${(error as Error).message}`)\n }\n }\n\n if (cleanedFiles > 0 || cleanedDirectories > 0) {\n this.logger.info(`✅ Clean slate complete: removed ${cleanedFiles} files and ${cleanedDirectories} directories`)\n this.logger.info(' 📝 Fresh registry files will be generated based on current configuration')\n } else {\n this.logger.info('✅ No existing registry files to clean up')\n }\n }\n\n /**\n * Generate CloudFormation resources for ABAC module-level policies\n */\n private async generatePolicyResources(): Promise<void> {\n if (this.configManager.config.skipPolicies || this.featureEntries.length === 0) {\n this.logger.info('⏭️ Skipping policy resource generation')\n return\n }\n\n try {\n const stage = this.serverless.service.provider?.stage || 'dev'\n const resources = this.serverless.service.resources || { Resources: {} }\n\n // Ensure Resources section exists\n if (!resources.Resources) {\n resources.Resources = {}\n }\n\n // Group features by module for ABAC approach\n const featuresByModule = this.groupFeaturesByModule(this.featureEntries)\n \n // Generate ABAC module-level policy resources instead of individual feature policies\n const abacPolicyResources = await this.generateAbacModulePolicies(featuresByModule, stage)\n\n // Add resources to CloudFormation template\n Object.assign(resources.Resources, abacPolicyResources)\n\n // Update the serverless service with the new resources\n this.serverless.service.resources = resources\n\n this.logger.info(`✅ Generated ${Object.keys(abacPolicyResources).length} ABAC module-level policy resources`)\n this.logger.info(` 📊 Policy efficiency: ${Object.keys(abacPolicyResources).length} module policies vs ${this.featureEntries.length} feature policies (${Math.round((1 - Object.keys(abacPolicyResources).length / this.featureEntries.length) * 100)}% reduction)`)\n } catch (error) {\n this.logger.error('❌ Error generating ABAC policy resources', error)\n this.logger.error('🚨 ABAC policy generation failure is critical - deployment cannot continue safely')\n throw error // Re-throw to interrupt deployment\n }\n }\n\n /**\n * Group features by module for ABAC policy generation\n */\n private groupFeaturesByModule(featureEntries: RegistryFeatureEntry[]): Record<string, RegistryFeatureEntry[]> {\n const featuresByModule: Record<string, RegistryFeatureEntry[]> = {}\n \n for (const feature of featureEntries) {\n if (!featuresByModule[feature.moduleName]) {\n featuresByModule[feature.moduleName] = []\n }\n featuresByModule[feature.moduleName].push(feature)\n }\n \n this.logger.info(`📊 Grouped ${featureEntries.length} features into ${Object.keys(featuresByModule).length} modules for ABAC policies`)\n Object.entries(featuresByModule).forEach(([module, features]) => {\n this.logger.info(` - ${module}: ${features.length} features`)\n })\n \n return featuresByModule\n }\n\n /**\n * Generate ABAC module-level policies using feature data from registry YAML files\n */\n private async generateAbacModulePolicies(featuresByModule: Record<string, RegistryFeatureEntry[]>, stage: string): Promise<Record<string, any>> {\n const policyResources: Record<string, any> = {}\n \n // Get AWS account ID for policy ARN generation\n const accountId = await this.getAwsAccountId()\n if (!accountId) {\n throw new Error('Cannot generate ABAC policies without AWS account ID')\n }\n \n const region = this.configManager.config.region\n const serviceName = this.serverless.service.service\n \n for (const [moduleName, moduleFeatures] of Object.entries(featuresByModule)) {\n try {\n this.logger.info(`🔐 Generating ABAC policy for module: ${moduleName}`)\n \n if (!moduleFeatures || moduleFeatures.length === 0) {\n this.logger.warning(`No features found for module ${moduleName}`)\n continue\n }\n \n // Try to fetch the API Gateway ID for this module\n const apiGatewayId = await this.getApiGatewayId(stage, serviceName)\n console.log('apiGatewayId', apiGatewayId, stage, serviceName)\n // Generate policy statements for each feature\n const statements = moduleFeatures.map(feature => {\n const featureNameSafeId = feature.featureName.replace(/[^a-zA-Z0-9]/g, '')\n \n // Collect all actions (API Gateway + custom policies)\n const actions = ['execute-api:Invoke']\n const resources: string[] = []\n \n // Add API Gateway resources from endpoints\n if (feature.endpoints && feature.endpoints.length > 0) {\n feature.endpoints.forEach(endpoint => {\n resources.push(this.endpointToArn(endpoint, '*', '*', apiGatewayId ? apiGatewayId : '#{ApiGatewayRestApi}', stage))\n })\n }\n \n // Add custom policy actions and resources\n if (feature.customPolicies && feature.customPolicies.length > 0) {\n feature.customPolicies.forEach(policy => {\n if (policy.Action) {\n const policyActions = Array.isArray(policy.Action) ? policy.Action : [policy.Action]\n actions.push(...policyActions)\n }\n if (policy.Resource) {\n const policyResources = Array.isArray(policy.Resource) ? policy.Resource : [policy.Resource]\n resources.push(...policyResources)\n }\n })\n }\n \n // Remove duplicates\n const uniqueActions = [...new Set(actions)]\n const uniqueResources = [...new Set(resources)]\n \n // If no resources defined, default to all\n if (uniqueResources.length === 0) {\n uniqueResources.push('*')\n }\n \n // Use feature ID for ABAC condition (fallback to feature name for backward compatibility)\n const featureIdentifier = feature.featureId || feature.featureName\n \n return {\n Sid: `${featureNameSafeId}`,\n Effect: 'Allow',\n Action: uniqueActions,\n Resource: uniqueResources,\n Condition: this.createAbacCondition(moduleName, featureIdentifier)\n }\n })\n \n // Create CloudFormation resource for this module policy\n const policyResourceName = `${this.toPascalCase(moduleName)}AbacPolicy`\n \n policyResources[policyResourceName] = {\n Type: 'AWS::IAM::ManagedPolicy',\n Properties: {\n ManagedPolicyName: `${this.configManager.config.policyPrefix}-abac`,\n Description: `ABAC-enabled policy for ${moduleName} module (${moduleFeatures.length} features)`,\n PolicyDocument: {\n Version: '2012-10-17',\n Statement: statements.map(statement => ({\n ...statement,\n Resource: statement.Resource.map((resource: string) => \n resource.includes('#{ApiGatewayRestApi}') \n ? { 'Fn::Sub': resource.replace('#{ApiGatewayRestApi}', '${ApiGatewayRestApi}') }\n : resource\n )\n }))\n },\n Path: '/'\n }\n }\n \n this.logger.info(` ✅ Created ABAC policy: ${policyResourceName} (${moduleFeatures.length} features)`)\n \n } catch (error) {\n this.logger.error(`❌ Failed to generate ABAC policy for module ${moduleName}:`, error)\n throw error\n }\n }\n \n return policyResources\n }\n\n /**\n * Converts an endpoint string to API Gateway ARN format\n * @param endpoint - Endpoint in format \"METHOD - /path/pattern\"\n * @param region - AWS region (defaults to *)\n * @param accountId - AWS account ID (defaults to *)\n * @param apiId - API Gateway ID or Serverless Framework reference (defaults to *)\n * @param stage - Deployment stage (defaults to *)\n * @returns API Gateway ARN with Serverless Framework reference for apiId\n */\n private endpointToArn(endpoint: string, region = '*', accountId = '*', apiId = '*', stage = '*'): string {\n // Parse endpoint format: \"POST - /admin/tenants\" or \"POST - admin/tenants\"\n const parts = endpoint.split(' - ')\n if (parts.length !== 2) {\n throw new Error(`Invalid endpoint format: ${endpoint}. Expected \"METHOD - path\"`)\n }\n \n const method = parts[0].trim().toUpperCase()\n let path = parts[1].trim()\n \n // Normalize path to ensure it starts with /\n path = normalizePath(path)\n \n // Convert path parameters from {param} to * for ARN matching\n const arnPath = path.replace(/\\{[^}]+\\}/g, '*')\n \n return `arn:aws:execute-api:${region}:${accountId}:${apiId}/${stage}/${method}${arnPath}`\n }\n\n /**\n * Creates an ABAC condition for feature access\n * @param moduleName - Module name (e.g., 'tenants')\n * @param featureId - Short feature ID (e.g., 'create-tenant-a1b2c3d4')\n * @returns IAM condition object\n */\n private createAbacCondition(moduleName: string, featureId: string): Record<string, any> {\n return {\n StringLike: {\n [`aws:PrincipalTag/${moduleName}Features`]: `*${featureId}*`\n }\n }\n }\n\n /**\n * Convert string to PascalCase for CloudFormation resource names\n */\n private toPascalCase(str: string): string {\n return str\n .split(/[-_\\s]+/)\n .map(word => word.charAt(0).toUpperCase() + word.slice(1).toLowerCase())\n .join('')\n }\n\n /**\n * Generate service package for imports\n */\n private async generateServiceFile(): Promise<void> {\n try {\n const packageCommand = new PackageGenerateCommand(this.configManager.config, this.logger)\n await packageCommand.execute({ force: true })\n } catch (error) {\n this.logger.error('❌ Error generating service file', error)\n this.logger.error('🚨 Service file generation failure - deployment cannot continue')\n throw error // Re-throw to interrupt deployment\n }\n }\n\n /**\n * Resolve CloudFormation policy ARNs to actual ARNs before storing in DynamoDB\n */\n private async resolvePolicyArns(): Promise<void> {\n if (this.featureEntries.length === 0) {\n return\n }\n\n // Get AWS account ID for ARN resolution\n const accountId = await this.getAwsAccountId()\n if (!accountId) {\n this.logger.warning('Could not determine AWS account ID - policy ARNs will not be resolved')\n return\n }\n\n this.logger.info(`🔧 Resolving ${this.featureEntries.length} policy ARNs with account ID: ${accountId}`)\n\n let resolvedCount = 0\n for (const feature of this.featureEntries) {\n if (feature.policyArn && typeof feature.policyArn === 'object') {\n // This is a CloudFormation intrinsic function - resolve it\n if (feature.policyArn['Fn::Sub'] && feature.policyName) {\n const resolvedArn = `arn:aws:iam::${accountId}:policy/${feature.policyName}`\n feature.policyArn = resolvedArn\n resolvedCount++\n this.logger.info(` ✅ Resolved: ${feature.featureName} → ${resolvedArn}`)\n }\n }\n }\n\n this.logger.info(`✅ Resolved ${resolvedCount} CloudFormation policy ARNs to actual ARNs`)\n }\n\n /**\n * Discover API Gateway RestApiId from CloudFormation stack or AWS API Gateway\n */\n private async discoverApiGatewayId(stackName: string, serviceName: string, region: string): Promise<string | undefined> {\n try {\n // Primary method: Query CloudFormation stack for API Gateway resources\n const { CloudFormationClient, DescribeStackResourcesCommand } = await import('@aws-sdk/client-cloudformation')\n const cfnClient = new CloudFormationClient({ region })\n\n try {\n const describeCommand = new DescribeStackResourcesCommand({\n StackName: stackName\n })\n\n const response = await cfnClient.send(describeCommand)\n\n // Find API Gateway REST API resource\n const apiResource = response.StackResources?.find(\n (resource: any) => resource.ResourceType === 'AWS::ApiGateway::RestApi'\n )\n\n if (apiResource?.PhysicalResourceId) {\n this.logger.info(`✅ Discovered API Gateway ID from CloudFormation: ${apiResource.PhysicalResourceId}`)\n return apiResource.PhysicalResourceId\n }\n } catch (cfnError: any) {\n this.logger.warning(`CloudFormation API Gateway lookup failed: ${cfnError.message}`)\n }\n\n // Fallback method: Use API Gateway client to list and match by name\n if (!this.apiGatewayClient) {\n this.apiGatewayClient = new APIGatewayClient({ region })\n }\n\n try {\n const apisCommand = new GetRestApisCommand({})\n const apisResponse = await this.apiGatewayClient.send(apisCommand)\n\n // Match by service name (common pattern in Serverless Framework)\n const matchingApi = apisResponse.items?.find(\n api => api.name === serviceName || api.name?.includes(serviceName)\n )\n\n if (matchingApi?.id) {\n this.logger.info(`✅ Discovered API Gateway ID from API Gateway service: ${matchingApi.id}`)\n return matchingApi.id\n }\n } catch (apiError: any) {\n this.logger.warning(`API Gateway service lookup failed: ${apiError.message}`)\n }\n\n // Graceful failure: log warning and return undefined\n this.logger.warning('Could not discover API Gateway ID. Module will be registered without apiGatewayId.')\n return undefined\n\n } catch (error: any) {\n this.logger.warning(`Error discovering API Gateway ID: ${error.message}`)\n return undefined\n }\n }\n\n /**\n * Ensure DynamoDB table exists using AWS SDK v3, then update registry data\n */\n private async ensureTableAndUpdateData(): Promise<void> {\n this.pluginLogger.trackHookCall('after:aws:deploy:deploy:updateStack')\n\n const totalEntries = this.moduleEntries.length + this.featureEntries.length\n if (this.configManager.config.skipDynamoDB || totalEntries === 0) {\n this.logger.info('⏭️ Skipping DynamoDB operations')\n return\n }\n\n try {\n // Discover API Gateway ID and update module entries\n const stackName = this.serverless.service.provider.stackName ||\n `${this.serverless.service.service}-${this.serverless.service.provider.stage}`\n const serviceName = this.serverless.service.service\n const region = this.configManager.config.region!\n\n const apiGatewayId = await this.discoverApiGatewayId(stackName, serviceName, region)\n\n // Update all module entries with the discovered API Gateway ID\n if (apiGatewayId) {\n this.moduleEntries = this.moduleEntries.map(entry => ({\n ...entry,\n apiGatewayId\n }))\n }\n\n // Resolve CloudFormation policy ARNs to actual ARNs before storing in DynamoDB\n await this.resolvePolicyArns()\n\n // Update registry data in DynamoDB\n await this.dynamoManager!.updateRegistryData(this.moduleEntries, this.featureEntries)\n\n // Write ABAC metadata to DynamoDB (if any exists)\n if (this.orchestrator && this.abacRoleMetadata && this.abacRoleMetadata.length > 0) {\n await this.orchestrator.writeAbacMetadata(this.abacRoleMetadata)\n }\n\n // Display summary\n this.pluginLogger.displayRegistrySummary(this.configManager.config, this.moduleEntries, this.featureEntries)\n } catch (error) {\n this.logger.error('❌ Error ensuring table and updating data', error)\n\n // Still show the registry summary even when table operations fail\n if (totalEntries > 0) {\n this.pluginLogger.displayRegistrySummary(this.configManager.config, this.moduleEntries, this.featureEntries)\n }\n\n // Don't fail deployment for registry operations\n }\n }\n\n\n /**\n * Handler for the registryGenerate command\n */\n private async registryGenerateHandler(): Promise<void> {\n this.pluginLogger.trackHookCall('registryGenerate:generate')\n\n const options: GenerateRegistryOptions = {\n module: this.options.module || this.options.m,\n force: this.options.force || this.options.f || false\n }\n\n try {\n // If force mode is enabled, we need to perform policy safety checks and cleanup\n if (options.force) {\n this.logger.info('🔧 Force mode enabled - performing safety checks and cleanup...')\n \n // Resolve variables in the config first\n await this.configManager.resolveVariables()\n this.configManager.validate()\n\n // Initialize managers for policy safety checks\n this.initializeManagers()\n\n // Read existing registry files to check for policies that might be in use\n // Temporarily disable strict mode for generation to avoid validation errors on missing directories\n if (this.registryProcessor) {\n const originalStrict = this.configManager.config.strict\n try {\n // Temporarily set strict mode to false during generation\n ;(this.configManager as any)._config.strict = false\n const { featureEntries } = await this.registryProcessor.processModuleRegistry()\n this.featureEntries = featureEntries\n } finally {\n // Restore original strict mode setting\n ;(this.configManager as any)._config.strict = originalStrict\n }\n }\n\n // Perform policy safety checks and cleanup\n await this.cleanupObsoleteFilesForGeneration(options.force)\n }\n\n const servicePath = this.serverless.config.servicePath || process.cwd()\n const generateCommand = new RegistryGenerateCommand(servicePath, this.logger)\n await generateCommand.execute(options)\n } catch (error) {\n this.logger.error('Registry generation failed', error)\n throw error\n }\n }\n\n /**\n * Handler for the registryGeneratePackage command\n */\n private async generateServicePackageCommand(): Promise<void> {\n const options: GeneratePackageOptions = {\n force: this.options.force || this.options.f || false,\n verbose: this.options.verbose || this.options.v || false\n }\n\n try {\n // Resolve variables in the config first\n await this.configManager.resolveVariables()\n\n const packageCommand = new PackageGenerateCommand(this.configManager.config, this.logger)\n await packageCommand.execute(options)\n } catch (error: any) {\n this.logger.error(`Failed to generate service package: ${error.message}`)\n throw error\n }\n }\n}\n\n// Export service functions for direct import by handlers and other modules\nexport {\n listAllModules,\n getModuleFeatures,\n getFeatureDetails,\n getFeatureById,\n getModuleMetadata,\n getAllEndpoints,\n getNonCustomEndpoints,\n createCustomFeature,\n // ABAC (Attribute-Based Access Control) functions\n generateModuleAbacPolicy,\n generateAllModuleAbacPolicies,\n generateAbacTags,\n getRequiredModulePolicyArns,\n createModuleRegistryLogger\n} from './service.js'\n\n// Export AI Registry Generator for ABAC-optimized registry generation\nexport { AIRegistryGenerator } from './ai/generator.js'\n\n// Export createModuleRegistryLogger separately due to module resolution\n// export { createModuleRegistryLogger } from './service.js'\n\n// Export types (commented out due to build issues)\n// export type {\n// ModuleInfo,\n// FeatureInfo,\n// FeatureDetails,\n// EndpointInfo\n// } from './types.js'\n\n// Export the plugin class as both default and named export for compatibility\nexport default ServerlessModuleRegistryPlugin\nexport { ServerlessModuleRegistryPlugin }\n\n// For CommonJS compatibility with Serverless Framework v4\n// @ts-ignore - Allow CommonJS export in ESM context for backward compatibility\ndeclare const module: any\nif (typeof module !== 'undefined' && module.exports) {\n // Preserve existing named exports while adding main export\n const existingExports = module.exports\n module.exports = ServerlessModuleRegistryPlugin\n module.exports.default = ServerlessModuleRegistryPlugin\n // Restore named exports\n if (existingExports && typeof existingExports === 'object') {\n Object.assign(module.exports, existingExports)\n }\n}","/**\n * Path utilities for module registry\n */\n\n/**\n * Normalize a path to ensure it starts with a forward slash\n * @param path - The path to normalize (e.g., \"users\" or \"/users\")\n * @returns Normalized path starting with \"/\" (e.g., \"/users\")\n */\nexport function normalizePath(path: string): string {\n if (!path) {\n return '/'\n }\n \n // Trim whitespace\n const trimmedPath = path.trim()\n \n // Ensure path starts with /\n return trimmedPath.startsWith('/') ? trimmedPath : `/${trimmedPath}`\n}\n\n/**\n * Parse an endpoint string and return normalized method and path\n * @param endpoint - Endpoint in format \"METHOD - path\" or \"METHOD - /path\"\n * @returns Object with method and normalized path, or null if invalid\n */\nexport function parseEndpoint(endpoint: string): { method: string; path: string } | null {\n if (!endpoint || typeof endpoint !== 'string') {\n return null\n }\n\n // Split on the first occurrence of ' - ' to handle paths that might contain dashes\n const dashIndex = endpoint.indexOf(' - ')\n if (dashIndex === -1) {\n return null\n }\n\n const method = endpoint.substring(0, dashIndex).trim().toUpperCase()\n const rawPath = endpoint.substring(dashIndex + 3).trim() // +3 for ' - '\n \n if (!method || !rawPath) {\n return null\n }\n\n return {\n method,\n path: normalizePath(rawPath)\n }\n}\n\n/**\n * Replace path parameters like {id}, {userId} with wildcards for ARN patterns\n * @param path - Path with parameters (e.g., \"/users/{id}/orders/{orderId}\")\n * @returns Path with wildcards (e.g., \"/users/star/orders/star\")\n */\nexport function replacePathParametersWithWildcards(path: string): string {\n return path.replace(/\\{[^}]+\\}/g, '*')\n}\n\n/**\n * Convert an endpoint to an API Gateway ARN\n * @param endpoint - Endpoint in format \"METHOD - path\"\n * @param region - AWS region (default: '*')\n * @param accountId - AWS account ID (default: '*')\n * @param apiId - API Gateway ID (default: '*')\n * @param stage - API Gateway stage (default: '*')\n * @returns API Gateway ARN or null if endpoint is invalid\n */\nexport function endpointToArn(\n endpoint: string,\n region: string = '*',\n accountId: string = '*',\n apiId: string = '*',\n stage: string = '*'\n): string | null {\n const parsed = parseEndpoint(endpoint)\n if (!parsed) {\n return null\n }\n\n const { method, path } = parsed\n const arnPath = replacePathParametersWithWildcards(path)\n \n return `arn:aws:execute-api:${region}:${accountId}:${apiId}/${stage}/${method}${arnPath}`\n}\n\n/**\n * Validate that a path is properly formatted\n * @param path - Path to validate\n * @returns True if path is valid, false otherwise\n */\nexport function isValidPath(path: string): boolean {\n if (!path || typeof path !== 'string') {\n return false\n }\n\n const trimmedPath = path.trim()\n \n // Must not be empty after trimming\n if (!trimmedPath) {\n return false\n }\n\n // Should not contain invalid characters for API Gateway paths\n // Allow alphanumeric, hyphens, underscores, slashes, curly braces, and dots\n const validPathRegex = /^[a-zA-Z0-9\\-_/.{}]+$/\n \n return validPathRegex.test(trimmedPath)\n}\n\n/**\n * Validate an endpoint string format\n * @param endpoint - Endpoint to validate\n * @returns True if endpoint is valid, false otherwise\n */\nexport function isValidEndpoint(endpoint: string): boolean {\n const parsed = parseEndpoint(endpoint)\n if (!parsed) {\n return false\n }\n\n const { method, path } = parsed\n \n // Validate HTTP method\n const validMethods = ['GET', 'POST', 'PUT', 'DELETE', 'PATCH', 'HEAD', 'OPTIONS']\n if (!validMethods.includes(method)) {\n return false\n }\n\n // Validate path\n return isValidPath(path)\n}\n","/**\n * Configuration Management for Module Registry Plugin\n */\n\nimport type { ModuleRegistryConfig, ServerlessVersion } from '../types.js'\nimport { detectServerlessVersion, resolveVarsSafely } from '@hyperdrive.bot/serverless-utils'\n\nexport class ConfigManager {\n private serverless: any\n private options: Record<string, any>\n private serverlessVersion: ServerlessVersion\n private _config: ModuleRegistryConfig\n\n constructor(serverless: any, options: Record<string, any>) {\n this.serverless = serverless\n this.options = options\n this.serverlessVersion = detectServerlessVersion(serverless)\n this._config = this.buildInitialConfig()\n }\n\n private buildInitialConfig(): ModuleRegistryConfig {\n const custom = this.serverless.service.custom?.moduleRegistry || {}\n const defaultTableName = this.generateTableName()\n\n // Determine default policy prefix\n const defaultPolicyPrefix = this.serverless.service.service?.includes('${')\n ? 'api' // Fallback if service name contains unresolved variables\n : this.serverless.service.service || 'api'\n\n return {\n tableName: custom.tableName || defaultTableName,\n region: custom.region || this.serverless.service.provider?.region || 'us-east-1',\n skipDynamoDB: custom.skipDynamoDB || false,\n skipPolicies: custom.skipPolicies || false,\n strict: custom.strict || false,\n policyPrefix: custom.policyPrefix || defaultPolicyPrefix,\n generateService: custom.generateService !== false, // Enabled by default\n servicePath: custom.servicePath || 'service.js',\n force: this.options.force || false,\n maxFeaturesPerModule: custom.maxFeaturesPerModule || 20, // Default to 20 features max\n ...custom\n }\n }\n\n private generateTableName(): string {\n let service = this.serverless.service.service || 'api'\n const provider = this.serverless.service.provider || {}\n const stage = provider.stage || 'dev'\n const region = provider.region || 'us-east-1'\n\n // Clean service name of any unresolved variables\n if (service.includes('${')) {\n service = 'api'\n }\n\n // Use pattern: {service}-{stage}-module-registry-{region}\n return `${service}-${stage}-module-registry-${region}`\n }\n\n /**\n * Resolve variables in all config properties\n */\n async resolveVariables(): Promise<void> {\n try {\n const resolvedConfig = await resolveVarsSafely(this.serverless, this._config as any)\n this._config = { ...this._config, ...resolvedConfig }\n\n // Check for any remaining unresolved variables\n const unresolvedVars: string[] = []\n Object.keys(this._config).forEach(key => {\n const value = (this._config as any)[key]\n if (typeof value === 'string' && value.includes('${')) {\n unresolvedVars.push(`${key}: ${value}`)\n }\n })\n\n if (unresolvedVars.length > 0) {\n throw new Error(`Unresolved variables in configuration: ${unresolvedVars.join(', ')}`)\n }\n } catch (error) {\n throw new Error(`Configuration variable resolution failed: ${(error as Error).message}`)\n }\n }\n\n /**\n * Get the current configuration\n */\n get config(): Readonly<ModuleRegistryConfig> {\n return Object.freeze({ ...this._config })\n }\n\n /**\n * Get serverless version\n */\n get version(): ServerlessVersion {\n return this.serverlessVersion\n }\n\n /**\n * Check if we're in a module-specific deployment\n */\n isModuleSpecificDeployment(): boolean {\n const moduleOption = this.options.module || this.options.m\n return !!moduleOption\n }\n\n /**\n * Extract module name from --module parameter\n */\n extractModuleName(): string | null {\n const moduleOption = this.options.module || this.options.m\n const moduleParam = Array.isArray(moduleOption) ? moduleOption[0] : moduleOption\n return moduleParam || null\n }\n\n /**\n * Get strict mode setting\n */\n get strict(): boolean {\n return this.config.strict || false\n }\n\n /**\n * Validate configuration\n */\n validate(): void {\n const required = ['tableName', 'region']\n const missing = required.filter(key => !this._config[key as keyof ModuleRegistryConfig])\n\n if (missing.length > 0) {\n throw new Error(`Missing required configuration: ${missing.join(', ')}`)\n }\n\n // Validate AWS region format\n const region = this._config.region!\n if (!/^[a-z0-9-]+$/.test(region)) {\n throw new Error(`Invalid AWS region format: ${region}`)\n }\n\n // Validate table name\n const tableName = this._config.tableName!\n if (tableName.length > 255 || !/^[a-zA-Z0-9._-]+$/.test(tableName)) {\n throw new Error(`Invalid DynamoDB table name: ${tableName}`)\n }\n }\n}","/**\n * DynamoDB Operations for Module Registry\n */\n\nimport {\n DynamoDBClient,\n CreateTableCommand,\n DescribeTableCommand,\n UpdateContinuousBackupsCommand,\n TagResourceCommand,\n QueryCommand,\n BatchWriteItemCommand,\n type BatchWriteItemCommandInput\n} from '@aws-sdk/client-dynamodb'\nimport { marshall, unmarshall } from '@aws-sdk/util-dynamodb'\nimport { chunkArray } from '@hyperdrive.bot/serverless-utils'\nimport type {\n ModuleRegistryConfig,\n RegistryModuleEntry,\n RegistryFeatureEntry,\n Logger,\n DynamoDBItem,\n BatchOperation\n} from '../types.js'\n\nexport class DynamoDBManager {\n private client: DynamoDBClient\n private config: ModuleRegistryConfig\n private logger: Logger\n\n constructor(config: ModuleRegistryConfig, logger: Logger) {\n this.config = config\n this.logger = logger\n this.client = new DynamoDBClient({\n region: config.region,\n maxAttempts: 3,\n retryMode: 'adaptive'\n })\n }\n\n /**\n * Ensure DynamoDB table exists and is properly configured\n */\n async ensureTable(): Promise<void> {\n const tableName = this.config.tableName!\n\n // Check if table exists\n let tableExists = false\n try {\n await this.client.send(new DescribeTableCommand({ TableName: tableName }))\n tableExists = true\n this.logger.info(`Table '${tableName}' already exists`)\n } catch (error: any) {\n if (error.name !== 'ResourceNotFoundException') {\n throw error\n }\n this.logger.info(`Table '${tableName}' does not exist, creating...`)\n }\n\n if (!tableExists) {\n await this.createTable(tableName)\n }\n }\n\n /**\n * Create DynamoDB table with proper configuration\n */\n private async createTable(tableName: string): Promise<void> {\n const createTableParams = {\n TableName: tableName,\n BillingMode: 'PAY_PER_REQUEST' as const,\n AttributeDefinitions: [\n { AttributeName: 'pk', AttributeType: 'S' as const },\n { AttributeName: 'sk', AttributeType: 'S' as const },\n { AttributeName: 'gsi1pk', AttributeType: 'S' as const },\n { AttributeName: 'gsi1sk', AttributeType: 'S' as const },\n { AttributeName: 'gsi2pk', AttributeType: 'S' as const },\n { AttributeName: 'gsi2sk', AttributeType: 'S' as const }\n ],\n KeySchema: [\n { AttributeName: 'pk', KeyType: 'HASH' as const },\n { AttributeName: 'sk', KeyType: 'RANGE' as const }\n ],\n GlobalSecondaryIndexes: [\n {\n IndexName: 'GSI1',\n KeySchema: [\n { AttributeName: 'gsi1pk', KeyType: 'HASH' as const },\n { AttributeName: 'gsi1sk', KeyType: 'RANGE' as const }\n ],\n Projection: { ProjectionType: 'ALL' as const }\n },\n {\n IndexName: 'GSI2',\n KeySchema: [\n { AttributeName: 'gsi2pk', KeyType: 'HASH' as const },\n { AttributeName: 'gsi2sk', KeyType: 'RANGE' as const }\n ],\n Projection: { ProjectionType: 'ALL' as const }\n }\n ]\n }\n\n await this.client.send(new CreateTableCommand(createTableParams))\n this.logger.info(`Created table '${tableName}'`)\n\n // Wait for table to be active\n await this.waitForTableActive(tableName)\n\n // Configure additional features\n await this.configureTableFeatures(tableName)\n }\n\n /**\n * Wait for table to become active\n */\n private async waitForTableActive(tableName: string): Promise<void> {\n this.logger.info('Waiting for table to be active...')\n let isActive = false\n while (!isActive) {\n await new Promise(resolve => setTimeout(resolve, 2000))\n const response = await this.client.send(new DescribeTableCommand({ TableName: tableName }))\n isActive = response.Table?.TableStatus === 'ACTIVE'\n }\n }\n\n /**\n * Configure table features (point-in-time recovery, tags)\n */\n private async configureTableFeatures(tableName: string): Promise<void> {\n // Enable point-in-time recovery\n try {\n await this.client.send(new UpdateContinuousBackupsCommand({\n TableName: tableName,\n PointInTimeRecoverySpecification: {\n PointInTimeRecoveryEnabled: true\n }\n }))\n this.logger.info('Enabled point-in-time recovery')\n } catch (error) {\n this.logger.warning(`Could not enable point-in-time recovery: ${(error as Error).message}`)\n }\n\n // Add compliance tags\n try {\n const describeResponse = await this.client.send(new DescribeTableCommand({ TableName: tableName }))\n const tableArn = describeResponse.Table?.TableArn\n\n if (tableArn) {\n await this.client.send(new TagResourceCommand({\n ResourceArn: tableArn,\n Tags: [\n { Key: 'Module', Value: 'module-registry' },\n { Key: 'Purpose', Value: 'StateStore' }\n ]\n }))\n this.logger.info('Added compliance tags to table')\n }\n } catch (error) {\n this.logger.warning(`Could not add tags: ${(error as Error).message}`)\n }\n }\n\n /**\n * Query existing entries for a module\n */\n async queryModuleEntries(moduleName: string): Promise<any[]> {\n const command = new QueryCommand({\n TableName: this.config.tableName!,\n KeyConditionExpression: 'pk = :pk',\n ExpressionAttributeValues: {\n ':pk': { S: `MODULE#${moduleName}` }\n }\n })\n\n const response = await this.client.send(command)\n return response.Items ? response.Items.map(item => unmarshall(item)) : []\n }\n\n /**\n * Query feature by featureId using GSI2\n */\n async queryFeatureById(featureId: string): Promise<any | null> {\n const command = new QueryCommand({\n TableName: this.config.tableName!,\n IndexName: 'GSI2',\n KeyConditionExpression: 'gsi2pk = :gsi2pk AND gsi2sk = :gsi2sk',\n ExpressionAttributeValues: {\n ':gsi2pk': { S: 'FEATURES' },\n ':gsi2sk': { S: `FEATURE#${featureId}` }\n }\n })\n\n const response = await this.client.send(command)\n const items = response.Items ? response.Items.map(item => unmarshall(item)) : []\n return items.length > 0 ? items[0] : null\n }\n\n /**\n * Query ABAC role entries\n * @param roleName - Optional role name. If provided, queries specific role. Otherwise, scans all roles.\n */\n async queryAbacRoleEntries(roleName?: string): Promise<any[]> {\n if (roleName) {\n // Query specific role\n const command = new QueryCommand({\n TableName: this.config.tableName!,\n KeyConditionExpression: 'pk = :pk',\n ExpressionAttributeValues: {\n ':pk': { S: `ROLE#${roleName}` }\n }\n })\n\n const response = await this.client.send(command)\n return response.Items ? response.Items.map(item => unmarshall(item)) : []\n } else {\n // Scan all roles (using GSI1 for better performance)\n const command = new QueryCommand({\n TableName: this.config.tableName!,\n IndexName: 'GSI1',\n KeyConditionExpression: 'gsi1pk = :gsi1pk',\n ExpressionAttributeValues: {\n ':gsi1pk': { S: 'ROLES' }\n }\n })\n\n const response = await this.client.send(command)\n return response.Items ? response.Items.map(item => unmarshall(item)) : []\n }\n }\n\n /**\n * Batch write operations to DynamoDB\n */\n async batchWrite(operations: BatchOperation[]): Promise<void> {\n if (operations.length === 0) return\n\n const batches = chunkArray(operations, 25)\n\n for (const batch of batches) {\n const requestItems: BatchWriteItemCommandInput['RequestItems'] = {\n [this.config.tableName!]: batch.map(op => {\n if (op.operation === 'put') {\n return { PutRequest: { Item: marshall(op.item) } }\n } else {\n return { DeleteRequest: { Key: marshall({ pk: op.item.pk, sk: op.item.sk }) } }\n }\n })\n }\n\n await this.client.send(new BatchWriteItemCommand({ RequestItems: requestItems }))\n }\n }\n\n /**\n * Convert module entries to DynamoDB items\n */\n moduleEntriesToItems(moduleEntries: RegistryModuleEntry[]): DynamoDBItem[] {\n return moduleEntries.map(entry => ({\n pk: `MODULE#${entry.moduleName}`,\n sk: 'MODULE',\n gsi1pk: 'MODULES',\n gsi1sk: `MODULE#${entry.moduleName}`,\n itemType: 'module' as const,\n ...entry\n }))\n }\n\n /**\n * Convert feature entries to DynamoDB items\n */\n featureEntriesToItems(featureEntries: RegistryFeatureEntry[]): DynamoDBItem[] {\n return featureEntries.map(entry => ({\n pk: `MODULE#${entry.moduleName}`,\n sk: `FEATURE#${entry.featureId}`,\n gsi1pk: `MODULE#${entry.moduleName}`,\n gsi1sk: `FEATURE#${entry.featureId}`,\n gsi2pk: 'FEATURES',\n gsi2sk: `FEATURE#${entry.featureId}`,\n itemType: 'feature' as const,\n ...entry\n }))\n }\n\n /**\n * Clean up stale entries for given modules\n */\n async cleanupStaleEntries(\n moduleNames: string[],\n currentModules: RegistryModuleEntry[],\n currentFeatures: RegistryFeatureEntry[]\n ): Promise<void> {\n const itemsToDelete: DynamoDBItem[] = []\n\n for (const moduleName of moduleNames) {\n const existingItems = await this.queryModuleEntries(moduleName)\n\n for (const existingItem of existingItems) {\n const isStillPresent = existingItem.itemType === 'module'\n ? currentModules.some(m => m.moduleName === existingItem.moduleName)\n : currentFeatures.some(f =>\n f.moduleName === existingItem.moduleName &&\n f.featureName === existingItem.featureName\n )\n\n if (!isStillPresent) {\n itemsToDelete.push({\n pk: existingItem.pk,\n sk: existingItem.sk,\n itemType: existingItem.itemType\n } as DynamoDBItem)\n }\n }\n }\n\n if (itemsToDelete.length > 0) {\n this.logger.info(`Cleaning up ${itemsToDelete.length} stale entries`)\n const deleteOperations: BatchOperation[] = itemsToDelete.map(item => ({\n operation: 'delete',\n item\n }))\n await this.batchWrite(deleteOperations)\n }\n }\n\n /**\n * Update registry data in DynamoDB\n */\n async updateRegistryData(\n moduleEntries: RegistryModuleEntry[],\n featureEntries: RegistryFeatureEntry[]\n ): Promise<void> {\n if (this.config.skipDynamoDB) {\n this.logger.info('Skipping DynamoDB operations (skipDynamoDB: true)')\n return\n }\n\n const totalEntries = moduleEntries.length + featureEntries.length\n if (totalEntries === 0) {\n this.logger.info('No registry entries to update')\n return\n }\n\n this.logger.info(`Updating registry data in DynamoDB (${totalEntries} entries)`)\n\n try {\n await this.ensureTable()\n\n // Clean up stale entries\n const moduleNames = [...new Set(moduleEntries.map(m => m.moduleName))]\n await this.cleanupStaleEntries(moduleNames, moduleEntries, featureEntries)\n\n // Prepare new entries\n const allItems = [\n ...this.moduleEntriesToItems(moduleEntries),\n ...this.featureEntriesToItems(featureEntries)\n ]\n\n // Batch write new entries\n if (allItems.length > 0) {\n this.logger.info(`Writing ${allItems.length} entries to DynamoDB`)\n const putOperations: BatchOperation[] = allItems.map(item => ({\n operation: 'put',\n item\n }))\n await this.batchWrite(putOperations)\n }\n\n this.logger.info('Registry data updated successfully')\n } catch (error) {\n this.logger.error('Error updating registry data', error)\n throw error\n }\n }\n}","/**\n * Policy Generation for Module Registry\n */\n\nimport type {\n RegistryFeatureEntry,\n PolicyDocument,\n PolicyStatement,\n CloudFormationPolicyResource,\n Logger\n} from '../types.js'\n\n/**\n * Parse an endpoint string and return normalized method and path\n * @param endpoint - Endpoint in format \"METHOD - path\" or \"METHOD - /path\"\n * @returns Object with method and normalized path, or null if invalid\n */\nfunction parseEndpoint(endpoint: string): { method: string; path: string } | null {\n if (!endpoint || typeof endpoint !== 'string') {\n return null\n }\n\n // Split on the first occurrence of ' - ' to handle paths that might contain dashes\n const dashIndex = endpoint.indexOf(' - ')\n if (dashIndex === -1) {\n return null\n }\n\n const method = endpoint.substring(0, dashIndex).trim().toUpperCase()\n let rawPath = endpoint.substring(dashIndex + 3).trim() // +3 for ' - '\n \n if (!method || !rawPath) {\n return null\n }\n\n // Normalize path to ensure it starts with /\n if (!rawPath.startsWith('/')) {\n rawPath = `/${rawPath}`\n }\n\n return {\n method,\n path: rawPath\n }\n}\n\n/**\n * Replace path parameters like {id}, {userId} with wildcards for ARN patterns\n * @param path - Path with parameters (e.g., \"/users/{id}/orders/{orderId}\")\n * @returns Path with wildcards (e.g., \"/users/star/orders/star\")\n */\nfunction replacePathParametersWithWildcards(path: string): string {\n return path.replace(/\\{[^}]+\\}/g, '*')\n}\n\nexport class PolicyGenerator {\n private policyPrefix: string\n private logger: Logger\n\n constructor(policyPrefix: string, logger: Logger) {\n this.policyPrefix = policyPrefix\n this.logger = logger\n }\n\n /**\n * Generate CloudFormation policy resources for all features\n */\n generatePolicyResources(\n featureEntries: RegistryFeatureEntry[],\n stage: string\n ): Record<string, CloudFormationPolicyResource> {\n if (featureEntries.length === 0) {\n this.logger.info('No features to generate policies for')\n return {}\n }\n\n this.logger.info(`Generating CloudFormation policy resources for ${featureEntries.length} features`)\n\n const resources: Record<string, CloudFormationPolicyResource> = {}\n let generatedPolicies = 0\n\n for (const feature of featureEntries) {\n try {\n const policyName = this.generatePolicyName(feature.moduleName, feature.featureName, stage)\n const resourceName = this.generateCloudFormationResourceName(feature.moduleName, feature.featureName)\n\n this.logger.info(`Generating resource: ${resourceName} (${policyName})`)\n\n // Generate policy document from endpoints\n const policyDocument = this.generatePolicyDocument(feature.endpoints, feature.customPolicies)\n\n // Create CloudFormation IAM Policy resource\n const policyResource: CloudFormationPolicyResource = {\n Type: 'AWS::IAM::ManagedPolicy',\n Properties: {\n ManagedPolicyName: policyName,\n Description: `${feature.description} - Module: ${feature.moduleName}`,\n PolicyDocument: policyDocument\n }\n }\n\n resources[resourceName] = policyResource\n\n // Calculate the policy ARN that CloudFormation will create\n const policyArn = {\n 'Fn::Sub': `arn:aws:iam::\\${AWS::AccountId}:policy/${policyName}`\n }\n\n // Store policy information in feature entry\n feature.policyArn = policyArn as any // CloudFormation reference\n feature.policyName = policyName\n\n generatedPolicies++\n this.logger.info(`Generated: ${resourceName}`)\n } catch (error) {\n this.logger.error(`✗ Error generating policy for ${feature.featureName}`, error)\n this.logger.error('🚨 Policy generation failure is critical - deployment cannot continue safely')\n throw new Error(`Policy generation failed for feature '${feature.featureName}': ${error instanceof Error ? error.message : String(error)}`)\n }\n }\n\n this.logger.info(`Generated ${generatedPolicies} CloudFormation policy resources`)\n return resources\n }\n\n /**\n * Generate policy name following AWS naming conventions\n */\n generatePolicyName(moduleName: string, featureName: string, stage: string): string {\n // AWS IAM policy names must contain only: alphanumeric chars and +=,.@_-\n // Max length is 128 characters\n const sanitizedPrefix = this.sanitizePolicyName(this.policyPrefix)\n const sanitizedStage = this.sanitizePolicyName(stage)\n const sanitizedModule = this.sanitizePolicyName(moduleName)\n const sanitizedFeature = this.sanitizePolicyName(featureName)\n\n const policyName = `${sanitizedPrefix}-${sanitizedFeature}`\n\n // Ensure we don't exceed 128 character limit\n if (policyName.length > 128) {\n // Truncate the feature name if needed, keeping at least 10 characters\n const availableLength = 128 - (sanitizedPrefix.length + sanitizedStage.length + sanitizedModule.length + 3) // 3 hyphens\n const truncatedFeature = sanitizedFeature.substring(0, Math.max(10, availableLength))\n return `${sanitizedPrefix}-${truncatedFeature}`\n }\n\n return policyName\n }\n\n /**\n * Sanitize string for AWS IAM policy name (only alphanumeric and +=,.@_-)\n */\n private sanitizePolicyName(str: string): string {\n return str\n .replace(/[^a-zA-Z0-9+=,.@_-]/g, '-') // Replace invalid chars with hyphens\n .replace(/-+/g, '-') // Replace multiple consecutive hyphens with single hyphen\n .replace(/^-|-$/g, '') // Remove leading/trailing hyphens\n }\n\n /**\n * Generate CloudFormation resource name (must be valid CloudFormation logical ID)\n */\n private generateCloudFormationResourceName(moduleName: string, featureName: string): string {\n // CloudFormation resource names must be alphanumeric and can contain hyphens\n // Convert to PascalCase and remove invalid characters\n const cleanFeatureName = this.toPascalCase(featureName)\n const cleanModuleName = this.toPascalCase(moduleName)\n return `${cleanModuleName}${cleanFeatureName}FeaturePolicy`\n }\n\n /**\n * Convert kebab-case or snake_case to PascalCase for CloudFormation resource names\n */\n private toPascalCase(str: string): string {\n return str\n .split(/[-_]/)\n .map(word => word.charAt(0).toUpperCase() + word.slice(1).toLowerCase())\n .join('')\n }\n\n /**\n * Generate IAM policy document from endpoints and custom policies\n */\n private generatePolicyDocument(endpoints: string[], customPolicies: any[] = []): PolicyDocument {\n const policyDocument: PolicyDocument = {\n Version: '2012-10-17',\n Statement: []\n }\n\n // Convert endpoints to API Gateway resource ARNs\n if (endpoints.length > 0) {\n const resources = endpoints\n .map(endpoint => {\n // Parse endpoint format: \"GET - /workforce/employees\" or \"GET - workforce/employees\"\n const parsed = parseEndpoint(endpoint)\n if (parsed) {\n const { method, path } = parsed\n // Replace path parameters with wildcards for ARN pattern\n const arnPath = replacePathParametersWithWildcards(path)\n // Generate API Gateway resource ARN pattern\n return `arn:aws:execute-api:*:*:*/*/${method}${arnPath}`\n }\n return null\n })\n .filter((resource): resource is string => resource !== null)\n\n if (resources.length > 0) {\n const apiGatewayStatement: PolicyStatement = {\n Effect: 'Allow',\n Action: ['execute-api:Invoke'],\n Resource: resources\n }\n policyDocument.Statement.push(apiGatewayStatement)\n }\n }\n\n // Add custom policies if provided\n if (customPolicies.length > 0) {\n // Validate custom policies\n const validPolicies = customPolicies.filter(this.validatePolicyStatement)\n policyDocument.Statement.push(...validPolicies)\n }\n\n return policyDocument\n }\n\n\n /**\n * Validate a policy statement structure\n */\n private validatePolicyStatement(statement: any): statement is PolicyStatement {\n if (!statement || typeof statement !== 'object') {\n this.logger.warning('Invalid policy statement: not an object')\n return false\n }\n\n if (!statement.Effect || !['Allow', 'Deny'].includes(statement.Effect)) {\n this.logger.warning('Invalid policy statement: missing or invalid Effect')\n return false\n }\n\n if (!statement.Action) {\n this.logger.warning('Invalid policy statement: missing Action')\n return false\n }\n\n if (!statement.Resource) {\n this.logger.warning('Invalid policy statement: missing Resource')\n return false\n }\n\n return true\n }\n\n /**\n * Resolve policy ARN from CloudFormation reference\n */\n resolvePolicyArn(feature: RegistryFeatureEntry, accountId: string): string {\n if (typeof feature.policyArn === 'object' && feature.policyArn['Fn::Sub']) {\n return `arn:aws:iam::${accountId}:policy/${feature.policyName}`\n }\n return feature.policyArn as string\n }\n}","/**\n * Registry Processing Core Logic\n */\n\nimport fs from 'fs'\nimport path from 'path'\nimport { glob } from 'glob'\nimport { parseCloudFormationYaml, resolveVarsSafely, dumpCloudFormationYaml } from '@hyperdrive.bot/serverless-utils'\nimport { generateFeatureId, validateFeatureId } from '../utils/feature-id.js'\nimport { AbacRoleScanner } from './abac-role-scanner.js'\nimport type {\n ModuleRegistryConfig,\n RegistryModuleEntry,\n RegistryFeatureEntry,\n ModuleMetadata,\n RegistryFeature,\n AbacRoleMetadata,\n Logger\n} from '../types.js'\nimport type { ConfigManager } from './config.js'\n\nexport interface ProcessingResult {\n moduleEntries: RegistryModuleEntry[]\n featureEntries: RegistryFeatureEntry[]\n abacRoleMetadata: AbacRoleMetadata[]\n}\n\nexport class RegistryProcessor {\n private servicePath: string\n private config: ConfigManager\n private logger: Logger\n private serverless: any\n\n constructor(servicePath: string, config: ConfigManager, logger: Logger, serverless?: any) {\n this.servicePath = servicePath\n this.config = config\n this.logger = logger\n this.serverless = serverless\n }\n\n /**\n * Main processing function - scans all modules for registry files\n */\n async processModuleRegistry(): Promise<ProcessingResult> {\n this.logger.info('🔍 Scanning modules for registry definitions...')\n this.logger.info(` Strict Mode: ${this.config.strict ? '🔒 ENABLED - Will FAIL deployment on missing registries' : '✅ DISABLED - Will gracefully skip missing registries'}`)\n\n const modulesDir = path.resolve(this.servicePath, 'serverless/modules')\n\n if (!fs.existsSync(modulesDir)) {\n this.logger.info('⚠️ No modules directory found, skipping registry processing')\n return { moduleEntries: [], featureEntries: [], abacRoleMetadata: [] }\n }\n\n // Determine which modules to process\n const moduleNames = this.getModulesToProcess(modulesDir)\n\n const moduleEntries: RegistryModuleEntry[] = []\n const featureEntries: RegistryFeatureEntry[] = []\n\n // Process each module\n for (const moduleName of moduleNames) {\n this.logger.info(`\\n🔍 Checking module: ${moduleName}`)\n const result = await this.processModule(moduleName, modulesDir)\n\n if (result.moduleEntry) {\n moduleEntries.push(result.moduleEntry)\n }\n featureEntries.push(...result.featureEntries)\n }\n\n // Resolve variables in collected entries\n await this.resolveVariablesInEntries(moduleEntries, featureEntries)\n\n this.logger.info(`✅ Registry processing complete. Found ${moduleEntries.length} modules, ${featureEntries.length} features`)\n\n // Process ABAC roles after feature registry is complete\n const abacRoleMetadata = await this.processAbacRoles(moduleNames, modulesDir, featureEntries)\n\n return { moduleEntries, featureEntries, abacRoleMetadata }\n }\n\n /**\n * Process ABAC role definitions for all modules\n */\n private async processAbacRoles(\n moduleNames: string[],\n modulesDir: string,\n featureEntries: RegistryFeatureEntry[]\n ): Promise<AbacRoleMetadata[]> {\n this.logger.info('\\n🔐 Processing ABAC role definitions...')\n\n // Build feature registry map for validation\n const featureRegistry = AbacRoleScanner.buildFeatureRegistryMap(featureEntries)\n\n // Initialize scanner\n const scanner = new AbacRoleScanner(this.servicePath, this.logger, featureRegistry)\n\n const allAbacMetadata: AbacRoleMetadata[] = []\n\n // Scan each module for ABAC roles\n for (const moduleName of moduleNames) {\n const moduleDir = path.join(modulesDir, moduleName)\n const abacMetadata = await scanner.scanModuleAbacRoles(moduleDir)\n\n if (abacMetadata.length > 0) {\n allAbacMetadata.push(...abacMetadata)\n }\n }\n\n if (allAbacMetadata.length === 0) {\n this.logger.info(' No ABAC role definitions found in any module')\n } else {\n this.logger.info(`✅ ABAC role processing complete. Found ${allAbacMetadata.length} role-module mappings`)\n }\n\n return allAbacMetadata\n }\n\n /**\n * Determine which modules to process based on deployment mode\n */\n private getModulesToProcess(modulesDir: string): string[] {\n // Check if we're in module-specific deployment mode\n const isModuleDeployment = this.isModuleSpecificDeployment()\n const targetModuleName = this.extractModuleName()\n\n if (isModuleDeployment && targetModuleName) {\n // Only process the target module\n this.logger.info(`🎯 Module-specific deployment detected - targeting module: ${targetModuleName}`)\n\n // Verify the target module exists\n const targetModuleDir = path.join(modulesDir, targetModuleName)\n if (!fs.existsSync(targetModuleDir)) {\n this.logger.warning(`⚠️ Target module '${targetModuleName}' not found at ${targetModuleDir}`)\n if (this.config.strict) {\n throw new Error(`Target module '${targetModuleName}' not found`)\n }\n return []\n }\n\n return [targetModuleName]\n } else {\n // Process all modules\n this.logger.info('🌐 Full deployment - scanning all modules')\n const allModules = fs.readdirSync(modulesDir)\n .filter(name => fs.statSync(path.join(modulesDir, name)).isDirectory())\n\n this.logger.info(`Found ${allModules.length} modules: ${allModules.join(', ')}`)\n return allModules\n }\n }\n\n /**\n * Check if we're in a module-specific deployment\n */\n private isModuleSpecificDeployment(): boolean {\n return this.config.isModuleSpecificDeployment()\n }\n\n /**\n * Extract module name from deployment context\n */\n private extractModuleName(): string | null {\n return this.config.extractModuleName()\n }\n\n /**\n * Process a single module's registry files\n */\n private async processModule(\n moduleName: string,\n modulesDir: string\n ): Promise<{ moduleEntry?: RegistryModuleEntry; featureEntries: RegistryFeatureEntry[] }> {\n const moduleDir = path.join(modulesDir, moduleName)\n const registryDir = path.join(moduleDir, 'registry')\n\n if (!fs.existsSync(registryDir)) {\n if (this.config.strict) {\n const errorMessage = `❌ STRICT MODE VIOLATION: Module '${moduleName}' is missing required registry folder!\\n Expected: ${registryDir}\\n Fix: Create the registry folder or set strict: false in configuration`\n this.logger.error('🚨 DEPLOYMENT FAILED - STRICT MODE VIOLATION 🚨')\n this.logger.error(errorMessage)\n throw new Error(`Module Registry Strict Mode Violation: Module '${moduleName}' missing registry folder`)\n }\n\n this.logger.info(`📁 Module '${moduleName}' has no registry folder, skipping (strict mode disabled)`)\n return { featureEntries: [] }\n }\n\n this.logger.info(`📋 Processing registry for module: ${moduleName}`)\n\n // Read module metadata\n const moduleMetadata = await this.readModuleMetadata(registryDir)\n\n // Create module entry\n const moduleEntry: RegistryModuleEntry = {\n moduleName,\n description: moduleMetadata.description || moduleMetadata.name,\n version: moduleMetadata.version,\n maintainer: (moduleMetadata as any).maintainer,\n tags: (moduleMetadata as any).tags,\n lastUpdated: new Date().toISOString()\n }\n this.logger.info(` ✓ Module metadata: ${moduleEntry.version}`)\n\n // Read all feature files\n const featureEntries = await this.processFeatureFiles(registryDir, moduleName, moduleMetadata)\n\n // Validate feature count limits\n await this.validateFeatureCount(moduleName, featureEntries.length)\n\n return { moduleEntry, featureEntries }\n }\n\n /**\n * Read module metadata from module.yml\n */\n private async readModuleMetadata(registryDir: string): Promise<ModuleMetadata> {\n const moduleFile = path.join(registryDir, 'module.yml')\n\n if (!fs.existsSync(moduleFile)) {\n return {\n name: 'unknown',\n version: '1.0.0',\n description: 'No module metadata available'\n }\n }\n\n try {\n const moduleData = parseCloudFormationYaml(fs.readFileSync(moduleFile, 'utf8')) as ModuleMetadata\n return moduleData\n } catch (error) {\n this.logger.warning(`⚠️ Error reading module metadata: ${(error as Error).message}`)\n return {\n name: 'unknown',\n version: '1.0.0',\n description: 'Error reading module metadata'\n }\n }\n }\n\n /**\n * Process all feature files in a module\n */\n private async processFeatureFiles(\n registryDir: string,\n moduleName: string,\n moduleMetadata: ModuleMetadata\n ): Promise<RegistryFeatureEntry[]> {\n const featuresDir = path.join(registryDir, 'features')\n if (!fs.existsSync(featuresDir)) {\n if (this.config.strict) {\n const errorMessage = `❌ STRICT MODE VIOLATION: Module '${moduleName}' is missing required features directory!\\n Expected: ${featuresDir}\\n Fix: Create the features directory with at least one .yml file`\n this.logger.error('🚨 DEPLOYMENT FAILED - STRICT MODE VIOLATION 🚨')\n this.logger.error(errorMessage)\n throw new Error(`Module Registry Strict Mode Violation: Module '${moduleName}' missing features directory`)\n }\n\n this.logger.warning(`⚠️ No features directory found in ${moduleName}/registry, skipping (strict mode disabled)`)\n return []\n }\n\n // Find all feature YAML files\n const featureFiles = await glob(path.join(featuresDir, '*.{yml,yaml}').replace(/\\\\/g, '/'))\n\n this.logger.info(` Found ${featureFiles.length} feature definitions`)\n\n const featureEntries: RegistryFeatureEntry[] = []\n\n // Process each feature file\n for (const featureFile of featureFiles) {\n const featureEntry = await this.processFeatureFile(featureFile, moduleName, moduleMetadata)\n if (featureEntry) {\n featureEntries.push(featureEntry)\n }\n }\n\n return featureEntries\n }\n\n /**\n * Process a single feature file\n */\n private async processFeatureFile(\n featureFile: string,\n moduleName: string,\n moduleMetadata: ModuleMetadata\n ): Promise<RegistryFeatureEntry | null> {\n const featureName = path.basename(featureFile, path.extname(featureFile))\n\n try {\n const featureData = parseCloudFormationYaml(fs.readFileSync(featureFile, 'utf8')) as RegistryFeature\n\n if (!featureData.name || !featureData.endpoints) {\n this.logger.warning(`⚠️ Invalid feature definition in ${featureFile}`)\n return null\n }\n\n // Generate feature ID if not provided\n let featureId = featureData.featureId\n let needsYamlUpdate = false\n \n if (!featureId) {\n featureId = generateFeatureId()\n needsYamlUpdate = true\n this.logger.info(`Generated new feature ID for ${featureName}: ${featureId}`)\n } else {\n // Validate existing feature ID\n if (!validateFeatureId(featureId)) {\n this.logger.warning(`Invalid feature ID: ${featureId} for feature: ${featureName}. Regenerating...`)\n featureId = generateFeatureId()\n needsYamlUpdate = true\n this.logger.info(`Generated replacement feature ID: ${featureId}`)\n }\n }\n \n // Update YAML file if feature ID was generated or regenerated\n if (needsYamlUpdate) {\n await this.updateFeatureYamlWithId(featureFile, featureId)\n }\n\n // Create registry entry\n const registryEntry: RegistryFeatureEntry = {\n moduleName,\n featureName,\n featureId,\n description: featureData.description || featureData.name,\n version: featureData.version || moduleMetadata.version,\n endpoints: featureData.endpoints || [],\n customPolicies: featureData.customPolicies || [],\n lastUpdated: new Date().toISOString()\n }\n\n this.logger.info(` ✓ ${featureName}: ${registryEntry.endpoints.length} endpoints`)\n return registryEntry\n } catch (error) {\n this.logger.error(`❌ Error processing feature ${featureFile}: ${(error as Error).message}`)\n return null\n }\n }\n\n /**\n * Resolve variables in collected registry entries using robust resolution\n */\n private async resolveVariablesInEntries(\n moduleEntries: RegistryModuleEntry[],\n featureEntries: RegistryFeatureEntry[]\n ): Promise<void> {\n if (!this.serverless) {\n this.logger.info('⏭️ Skipping variable resolution (no serverless context)')\n return\n }\n\n this.logger.info('🔧 Resolving variables in registry entries...')\n\n try {\n // Resolve variables in module entries\n if (moduleEntries.length > 0) {\n this.logger.info(` Resolving ${moduleEntries.length} module entries...`)\n for (let i = 0; i < moduleEntries.length; i++) {\n const resolved = await resolveVarsSafely(this.serverless, moduleEntries[i] as any)\n moduleEntries[i] = resolved as RegistryModuleEntry\n }\n }\n\n // Resolve variables in feature entries\n if (featureEntries.length > 0) {\n this.logger.info(` Resolving ${featureEntries.length} feature entries...`)\n for (let i = 0; i < featureEntries.length; i++) {\n const resolved = await resolveVarsSafely(this.serverless, featureEntries[i] as any)\n featureEntries[i] = resolved as RegistryFeatureEntry\n }\n }\n\n this.logger.info('✅ Variable resolution complete for all registry entries')\n } catch (error) {\n this.logger.warning(`⚠️ Variable resolution error: ${(error as Error).message}`)\n // Don't fail deployment for variable resolution errors\n }\n }\n\n /**\n * Update a feature YAML file to include the generated feature ID\n */\n private async updateFeatureYamlWithId(featureFile: string, featureId: string): Promise<void> {\n try {\n // Read the existing YAML file\n const yamlContent = fs.readFileSync(featureFile, 'utf8')\n const featureData = parseCloudFormationYaml(yamlContent) as any\n\n // Add the feature ID\n featureData.featureId = featureId\n\n // Write back to file\n const updatedYaml = dumpCloudFormationYaml(featureData)\n fs.writeFileSync(featureFile, updatedYaml, 'utf8')\n\n this.logger.info(` ✅ Updated ${path.basename(featureFile)} with featureId: ${featureId}`)\n } catch (error) {\n this.logger.error(`❌ Failed to update YAML file ${featureFile} with feature ID: ${(error as Error).message}`)\n // Don't throw - continue processing other features\n }\n }\n\n /**\n * Validate that module doesn't exceed feature count limits\n */\n private async validateFeatureCount(moduleName: string, featureCount: number): Promise<void> {\n const maxFeatures = this.config.config.maxFeaturesPerModule || 20\n\n if (featureCount > maxFeatures) {\n const errorMessage = [\n `❌ DEPLOYMENT FAILED: Module '${moduleName}' exceeds feature limit!`,\n ``,\n `📊 Module Statistics:`,\n ` • Current features: ${featureCount}`,\n ` • Maximum allowed: ${maxFeatures}`,\n ` • Excess features: ${featureCount - maxFeatures}`,\n ``,\n `🚨 Why this limit exists:`,\n ` • AWS IAM managed policies have a 10,240 character limit`,\n ` • Each feature adds ~322 characters to the policy`,\n ` • Exceeding this limit causes IAM policy creation to fail`,\n ` • This safety limit prevents deployment failures`,\n ``,\n `🔧 How to fix:`,\n ` 1. Split module into smaller, focused modules:`,\n ` • Split by business domain (users, reports, admin)`,\n ` • Group related features together`,\n ` • Aim for 15-20 features per module`,\n ``,\n ` 2. Combine similar features with multiple endpoints:`,\n ` • Group related CRUD operations`,\n ` • Merge features with identical permissions`,\n ``,\n ` 3. Configure custom limit (not recommended):`,\n ` • Set maxFeaturesPerModule: ${featureCount} in plugin config`,\n ` • Risk: May cause IAM policy deployment failures`,\n ``,\n `💡 Recommended module split for '${moduleName}':`,\n ` Consider creating: ${moduleName}-core, ${moduleName}-admin, ${moduleName}-reports`\n ].join('\\n')\n\n this.logger.error(errorMessage)\n throw new Error(`Module '${moduleName}' has ${featureCount} features, exceeding the limit of ${maxFeatures}. Split the module or reduce features to deploy successfully.`)\n }\n\n // Log success for modules approaching the limit\n if (featureCount > maxFeatures * 0.8) {\n this.logger.warning(`⚠️ Module '${moduleName}' is approaching the feature limit (${featureCount}/${maxFeatures} features). Consider module refactoring.`)\n } else {\n this.logger.info(`✅ Module '${moduleName}' feature count: ${featureCount}/${maxFeatures} (within limits)`)\n }\n }\n}","import expand from 'brace-expansion'\nimport { assertValidPattern } from './assert-valid-pattern.js'\nimport { AST, ExtglobType } from './ast.js'\nimport { escape } from './escape.js'\nimport { unescape } from './unescape.js'\n\ntype Platform =\n | 'aix'\n | 'android'\n | 'darwin'\n | 'freebsd'\n | 'haiku'\n | 'linux'\n | 'openbsd'\n | 'sunos'\n | 'win32'\n | 'cygwin'\n | 'netbsd'\n\nexport interface MinimatchOptions {\n nobrace?: boolean\n nocomment?: boolean\n nonegate?: boolean\n debug?: boolean\n noglobstar?: boolean\n noext?: boolean\n nonull?: boolean\n windowsPathsNoEscape?: boolean\n allowWindowsEscape?: boolean\n partial?: boolean\n dot?: boolean\n nocase?: boolean\n nocaseMagicOnly?: boolean\n magicalBraces?: boolean\n matchBase?: boolean\n flipNegate?: boolean\n preserveMultipleSlashes?: boolean\n optimizationLevel?: number\n platform?: Platform\n windowsNoMagicRoot?: boolean\n}\n\nexport const minimatch = (\n p: string,\n pattern: string,\n options: MinimatchOptions = {}\n) => {\n assertValidPattern(pattern)\n\n // shortcut: comments match nothing.\n if (!options.nocomment && pattern.charAt(0) === '#') {\n return false\n }\n\n return new Minimatch(pattern, options).match(p)\n}\n\n// Optimized checking for the most common glob patterns.\nconst starDotExtRE = /^\\*+([^+@!?\\*\\[\\(]*)$/\nconst starDotExtTest = (ext: string) => (f: string) =>\n !f.startsWith('.') && f.endsWith(ext)\nconst starDotExtTestDot = (ext: string) => (f: string) => f.endsWith(ext)\nconst starDotExtTestNocase = (ext: string) => {\n ext = ext.toLowerCase()\n return (f: string) => !f.startsWith('.') && f.toLowerCase().endsWith(ext)\n}\nconst starDotExtTestNocaseDot = (ext: string) => {\n ext = ext.toLowerCase()\n return (f: string) => f.toLowerCase().endsWith(ext)\n}\nconst starDotStarRE = /^\\*+\\.\\*+$/\nconst starDotStarTest = (f: string) => !f.startsWith('.') && f.includes('.')\nconst starDotStarTestDot = (f: string) =>\n f !== '.' && f !== '..' && f.includes('.')\nconst dotStarRE = /^\\.\\*+$/\nconst dotStarTest = (f: string) => f !== '.' && f !== '..' && f.startsWith('.')\nconst starRE = /^\\*+$/\nconst starTest = (f: string) => f.length !== 0 && !f.startsWith('.')\nconst starTestDot = (f: string) => f.length !== 0 && f !== '.' && f !== '..'\nconst qmarksRE = /^\\?+([^+@!?\\*\\[\\(]*)?$/\nconst qmarksTestNocase = ([$0, ext = '']: RegExpMatchArray) => {\n const noext = qmarksTestNoExt([$0])\n if (!ext) return noext\n ext = ext.toLowerCase()\n return (f: string) => noext(f) && f.toLowerCase().endsWith(ext)\n}\nconst qmarksTestNocaseDot = ([$0, ext = '']: RegExpMatchArray) => {\n const noext = qmarksTestNoExtDot([$0])\n if (!ext) return noext\n ext = ext.toLowerCase()\n return (f: string) => noext(f) && f.toLowerCase().endsWith(ext)\n}\nconst qmarksTestDot = ([$0, ext = '']: RegExpMatchArray) => {\n const noext = qmarksTestNoExtDot([$0])\n return !ext ? noext : (f: string) => noext(f) && f.endsWith(ext)\n}\nconst qmarksTest = ([$0, ext = '']: RegExpMatchArray) => {\n const noext = qmarksTestNoExt([$0])\n return !ext ? noext : (f: string) => noext(f) && f.endsWith(ext)\n}\nconst qmarksTestNoExt = ([$0]: RegExpMatchArray) => {\n const len = $0.length\n return (f: string) => f.length === len && !f.startsWith('.')\n}\nconst qmarksTestNoExtDot = ([$0]: RegExpMatchArray) => {\n const len = $0.length\n return (f: string) => f.length === len && f !== '.' && f !== '..'\n}\n\n/* c8 ignore start */\nconst defaultPlatform: Platform = (\n typeof process === 'object' && process\n ? (typeof process.env === 'object' &&\n process.env &&\n process.env.__MINIMATCH_TESTING_PLATFORM__) ||\n process.platform\n : 'posix'\n) as Platform\ntype Sep = '\\\\' | '/'\nconst path: { [k: string]: { sep: Sep } } = {\n win32: { sep: '\\\\' },\n posix: { sep: '/' },\n}\n/* c8 ignore stop */\n\nexport const sep = defaultPlatform === 'win32' ? path.win32.sep : path.posix.sep\nminimatch.sep = sep\n\nexport const GLOBSTAR = Symbol('globstar **')\nminimatch.GLOBSTAR = GLOBSTAR\n\n// any single thing other than /\n// don't need to escape / when using new RegExp()\nconst qmark = '[^/]'\n\n// * => any number of characters\nconst star = qmark + '*?'\n\n// ** when dots are allowed. Anything goes, except .. and .\n// not (^ or / followed by one or two dots followed by $ or /),\n// followed by anything, any number of times.\nconst twoStarDot = '(?:(?!(?:\\\\/|^)(?:\\\\.{1,2})($|\\\\/)).)*?'\n\n// not a ^ or / followed by a dot,\n// followed by anything, any number of times.\nconst twoStarNoDot = '(?:(?!(?:\\\\/|^)\\\\.).)*?'\n\nexport const filter =\n (pattern: string, options: MinimatchOptions = {}) =>\n (p: string) =>\n minimatch(p, pattern, options)\nminimatch.filter = filter\n\nconst ext = (a: MinimatchOptions, b: MinimatchOptions = {}) =>\n Object.assign({}, a, b)\n\nexport const defaults = (def: MinimatchOptions): typeof minimatch => {\n if (!def || typeof def !== 'object' || !Object.keys(def).length) {\n return minimatch\n }\n\n const orig = minimatch\n\n const m = (p: string, pattern: string, options: MinimatchOptions = {}) =>\n orig(p, pattern, ext(def, options))\n\n return Object.assign(m, {\n Minimatch: class Minimatch extends orig.Minimatch {\n constructor(pattern: string, options: MinimatchOptions = {}) {\n super(pattern, ext(def, options))\n }\n static defaults(options: MinimatchOptions) {\n return orig.defaults(ext(def, options)).Minimatch\n }\n },\n\n AST: class AST extends orig.AST {\n /* c8 ignore start */\n constructor(\n type: ExtglobType | null,\n parent?: AST,\n options: MinimatchOptions = {}\n ) {\n super(type, parent, ext(def, options))\n }\n /* c8 ignore stop */\n\n static fromGlob(pattern: string, options: MinimatchOptions = {}) {\n return orig.AST.fromGlob(pattern, ext(def, options))\n }\n },\n\n unescape: (\n s: string,\n options: Pick<MinimatchOptions, 'windowsPathsNoEscape'> = {}\n ) => orig.unescape(s, ext(def, options)),\n\n escape: (\n s: string,\n options: Pick<MinimatchOptions, 'windowsPathsNoEscape'> = {}\n ) => orig.escape(s, ext(def, options)),\n\n filter: (pattern: string, options: MinimatchOptions = {}) =>\n orig.filter(pattern, ext(def, options)),\n\n defaults: (options: MinimatchOptions) => orig.defaults(ext(def, options)),\n\n makeRe: (pattern: string, options: MinimatchOptions = {}) =>\n orig.makeRe(pattern, ext(def, options)),\n\n braceExpand: (pattern: string, options: MinimatchOptions = {}) =>\n orig.braceExpand(pattern, ext(def, options)),\n\n match: (list: string[], pattern: string, options: MinimatchOptions = {}) =>\n orig.match(list, pattern, ext(def, options)),\n\n sep: orig.sep,\n GLOBSTAR: GLOBSTAR as typeof GLOBSTAR,\n })\n}\nminimatch.defaults = defaults\n\n// Brace expansion:\n// a{b,c}d -> abd acd\n// a{b,}c -> abc ac\n// a{0..3}d -> a0d a1d a2d a3d\n// a{b,c{d,e}f}g -> abg acdfg acefg\n// a{b,c}d{e,f}g -> abdeg acdeg abdeg abdfg\n//\n// Invalid sets are not expanded.\n// a{2..}b -> a{2..}b\n// a{b}c -> a{b}c\nexport const braceExpand = (\n pattern: string,\n options: MinimatchOptions = {}\n) => {\n assertValidPattern(pattern)\n\n // Thanks to Yeting Li <https://github.com/yetingli> for\n // improving this regexp to avoid a ReDOS vulnerability.\n if (options.nobrace || !/\\{(?:(?!\\{).)*\\}/.test(pattern)) {\n // shortcut. no need to expand.\n return [pattern]\n }\n\n return expand(pattern)\n}\nminimatch.braceExpand = braceExpand\n\n// parse a component of the expanded set.\n// At this point, no pattern may contain \"/\" in it\n// so we're going to return a 2d array, where each entry is the full\n// pattern, split on '/', and then turned into a regular expression.\n// A regexp is made at the end which joins each array with an\n// escaped /, and another full one which joins each regexp with |.\n//\n// Following the lead of Bash 4.1, note that \"**\" only has special meaning\n// when it is the *only* thing in a path portion. Otherwise, any series\n// of * is equivalent to a single *. Globstar behavior is enabled by\n// default, and can be disabled by setting options.noglobstar.\n\nexport const makeRe = (pattern: string, options: MinimatchOptions = {}) =>\n new Minimatch(pattern, options).makeRe()\nminimatch.makeRe = makeRe\n\nexport const match = (\n list: string[],\n pattern: string,\n options: MinimatchOptions = {}\n) => {\n const mm = new Minimatch(pattern, options)\n list = list.filter(f => mm.match(f))\n if (mm.options.nonull && !list.length) {\n list.push(pattern)\n }\n return list\n}\nminimatch.match = match\n\n// replace stuff like \\* with *\nconst globMagic = /[?*]|[+@!]\\(.*?\\)|\\[|\\]/\nconst regExpEscape = (s: string) =>\n s.replace(/[-[\\]{}()*+?.,\\\\^$|#\\s]/g, '\\\\$&')\n\nexport type MMRegExp = RegExp & {\n _src?: string\n _glob?: string\n}\n\nexport type ParseReturnFiltered = string | MMRegExp | typeof GLOBSTAR\nexport type ParseReturn = ParseReturnFiltered | false\n\nexport class Minimatch {\n options: MinimatchOptions\n set: ParseReturnFiltered[][]\n pattern: string\n\n windowsPathsNoEscape: boolean\n nonegate: boolean\n negate: boolean\n comment: boolean\n empty: boolean\n preserveMultipleSlashes: boolean\n partial: boolean\n globSet: string[]\n globParts: string[][]\n nocase: boolean\n\n isWindows: boolean\n platform: Platform\n windowsNoMagicRoot: boolean\n\n regexp: false | null | MMRegExp\n constructor(pattern: string, options: MinimatchOptions = {}) {\n assertValidPattern(pattern)\n\n options = options || {}\n this.options = options\n this.pattern = pattern\n this.platform = options.platform || defaultPlatform\n this.isWindows = this.platform === 'win32'\n this.windowsPathsNoEscape =\n !!options.windowsPathsNoEscape || options.allowWindowsEscape === false\n if (this.windowsPathsNoEscape) {\n this.pattern = this.pattern.replace(/\\\\/g, '/')\n }\n this.preserveMultipleSlashes = !!options.preserveMultipleSlashes\n this.regexp = null\n this.negate = false\n this.nonegate = !!options.nonegate\n this.comment = false\n this.empty = false\n this.partial = !!options.partial\n this.nocase = !!this.options.nocase\n this.windowsNoMagicRoot =\n options.windowsNoMagicRoot !== undefined\n ? options.windowsNoMagicRoot\n : !!(this.isWindows && this.nocase)\n\n this.globSet = []\n this.globParts = []\n this.set = []\n\n // make the set of regexps etc.\n this.make()\n }\n\n hasMagic(): boolean {\n if (this.options.magicalBraces && this.set.length > 1) {\n return true\n }\n for (const pattern of this.set) {\n for (const part of pattern) {\n if (typeof part !== 'string') return true\n }\n }\n return false\n }\n\n debug(..._: any[]) {}\n\n make() {\n const pattern = this.pattern\n const options = this.options\n\n // empty patterns and comments match nothing.\n if (!options.nocomment && pattern.charAt(0) === '#') {\n this.comment = true\n return\n }\n\n if (!pattern) {\n this.empty = true\n return\n }\n\n // step 1: figure out negation, etc.\n this.parseNegate()\n\n // step 2: expand braces\n this.globSet = [...new Set(this.braceExpand())]\n\n if (options.debug) {\n this.debug = (...args: any[]) => console.error(...args)\n }\n\n this.debug(this.pattern, this.globSet)\n\n // step 3: now we have a set, so turn each one into a series of\n // path-portion matching patterns.\n // These will be regexps, except in the case of \"**\", which is\n // set to the GLOBSTAR object for globstar behavior,\n // and will not contain any / characters\n //\n // First, we preprocess to make the glob pattern sets a bit simpler\n // and deduped. There are some perf-killing patterns that can cause\n // problems with a glob walk, but we can simplify them down a bit.\n const rawGlobParts = this.globSet.map(s => this.slashSplit(s))\n this.globParts = this.preprocess(rawGlobParts)\n this.debug(this.pattern, this.globParts)\n\n // glob --> regexps\n let set = this.globParts.map((s, _, __) => {\n if (this.isWindows && this.windowsNoMagicRoot) {\n // check if it's a drive or unc path.\n const isUNC =\n s[0] === '' &&\n s[1] === '' &&\n (s[2] === '?' || !globMagic.test(s[2])) &&\n !globMagic.test(s[3])\n const isDrive = /^[a-z]:/i.test(s[0])\n if (isUNC) {\n return [...s.slice(0, 4), ...s.slice(4).map(ss => this.parse(ss))]\n } else if (isDrive) {\n return [s[0], ...s.slice(1).map(ss => this.parse(ss))]\n }\n }\n return s.map(ss => this.parse(ss))\n })\n\n this.debug(this.pattern, set)\n\n // filter out everything that didn't compile properly.\n this.set = set.filter(\n s => s.indexOf(false) === -1\n ) as ParseReturnFiltered[][]\n\n // do not treat the ? in UNC paths as magic\n if (this.isWindows) {\n for (let i = 0; i < this.set.length; i++) {\n const p = this.set[i]\n if (\n p[0] === '' &&\n p[1] === '' &&\n this.globParts[i][2] === '?' &&\n typeof p[3] === 'string' &&\n /^[a-z]:$/i.test(p[3])\n ) {\n p[2] = '?'\n }\n }\n }\n\n this.debug(this.pattern, this.set)\n }\n\n // various transforms to equivalent pattern sets that are\n // faster to process in a filesystem walk. The goal is to\n // eliminate what we can, and push all ** patterns as far\n // to the right as possible, even if it increases the number\n // of patterns that we have to process.\n preprocess(globParts: string[][]) {\n // if we're not in globstar mode, then turn all ** into *\n if (this.options.noglobstar) {\n for (let i = 0; i < globParts.length; i++) {\n for (let j = 0; j < globParts[i].length; j++) {\n if (globParts[i][j] === '**') {\n globParts[i][j] = '*'\n }\n }\n }\n }\n\n const { optimizationLevel = 1 } = this.options\n\n if (optimizationLevel >= 2) {\n // aggressive optimization for the purpose of fs walking\n globParts = this.firstPhasePreProcess(globParts)\n globParts = this.secondPhasePreProcess(globParts)\n } else if (optimizationLevel >= 1) {\n // just basic optimizations to remove some .. parts\n globParts = this.levelOneOptimize(globParts)\n } else {\n // just collapse multiple ** portions into one\n globParts = this.adjascentGlobstarOptimize(globParts)\n }\n\n return globParts\n }\n\n // just get rid of adjascent ** portions\n adjascentGlobstarOptimize(globParts: string[][]) {\n return globParts.map(parts => {\n let gs: number = -1\n while (-1 !== (gs = parts.indexOf('**', gs + 1))) {\n let i = gs\n while (parts[i + 1] === '**') {\n i++\n }\n if (i !== gs) {\n parts.splice(gs, i - gs)\n }\n }\n return parts\n })\n }\n\n // get rid of adjascent ** and resolve .. portions\n levelOneOptimize(globParts: string[][]) {\n return globParts.map(parts => {\n parts = parts.reduce((set: string[], part) => {\n const prev = set[set.length - 1]\n if (part === '**' && prev === '**') {\n return set\n }\n if (part === '..') {\n if (prev && prev !== '..' && prev !== '.' && prev !== '**') {\n set.pop()\n return set\n }\n }\n set.push(part)\n return set\n }, [])\n return parts.length === 0 ? [''] : parts\n })\n }\n\n levelTwoFileOptimize(parts: string | string[]) {\n if (!Array.isArray(parts)) {\n parts = this.slashSplit(parts)\n }\n let didSomething: boolean = false\n do {\n didSomething = false\n // <pre>/<e>/<rest> -> <pre>/<rest>\n if (!this.preserveMultipleSlashes) {\n for (let i = 1; i < parts.length - 1; i++) {\n const p = parts[i]\n // don't squeeze out UNC patterns\n if (i === 1 && p === '' && parts[0] === '') continue\n if (p === '.' || p === '') {\n didSomething = true\n parts.splice(i, 1)\n i--\n }\n }\n if (\n parts[0] === '.' &&\n parts.length === 2 &&\n (parts[1] === '.' || parts[1] === '')\n ) {\n didSomething = true\n parts.pop()\n }\n }\n\n // <pre>/<p>/../<rest> -> <pre>/<rest>\n let dd: number = 0\n while (-1 !== (dd = parts.indexOf('..', dd + 1))) {\n const p = parts[dd - 1]\n if (p && p !== '.' && p !== '..' && p !== '**') {\n didSomething = true\n parts.splice(dd - 1, 2)\n dd -= 2\n }\n }\n } while (didSomething)\n return parts.length === 0 ? [''] : parts\n }\n\n // First phase: single-pattern processing\n // <pre> is 1 or more portions\n // <rest> is 1 or more portions\n // <p> is any portion other than ., .., '', or **\n // <e> is . or ''\n //\n // **/.. is *brutal* for filesystem walking performance, because\n // it effectively resets the recursive walk each time it occurs,\n // and ** cannot be reduced out by a .. pattern part like a regexp\n // or most strings (other than .., ., and '') can be.\n //\n // <pre>/**/../<p>/<p>/<rest> -> {<pre>/../<p>/<p>/<rest>,<pre>/**/<p>/<p>/<rest>}\n // <pre>/<e>/<rest> -> <pre>/<rest>\n // <pre>/<p>/../<rest> -> <pre>/<rest>\n // **/**/<rest> -> **/<rest>\n //\n // **/*/<rest> -> */**/<rest> <== not valid because ** doesn't follow\n // this WOULD be allowed if ** did follow symlinks, or * didn't\n firstPhasePreProcess(globParts: string[][]) {\n let didSomething = false\n do {\n didSomething = false\n // <pre>/**/../<p>/<p>/<rest> -> {<pre>/../<p>/<p>/<rest>,<pre>/**/<p>/<p>/<rest>}\n for (let parts of globParts) {\n let gs: number = -1\n while (-1 !== (gs = parts.indexOf('**', gs + 1))) {\n let gss: number = gs\n while (parts[gss + 1] === '**') {\n // <pre>/**/**/<rest> -> <pre>/**/<rest>\n gss++\n }\n // eg, if gs is 2 and gss is 4, that means we have 3 **\n // parts, and can remove 2 of them.\n if (gss > gs) {\n parts.splice(gs + 1, gss - gs)\n }\n\n let next = parts[gs + 1]\n const p = parts[gs + 2]\n const p2 = parts[gs + 3]\n if (next !== '..') continue\n if (\n !p ||\n p === '.' ||\n p === '..' ||\n !p2 ||\n p2 === '.' ||\n p2 === '..'\n ) {\n continue\n }\n didSomething = true\n // edit parts in place, and push the new one\n parts.splice(gs, 1)\n const other = parts.slice(0)\n other[gs] = '**'\n globParts.push(other)\n gs--\n }\n\n // <pre>/<e>/<rest> -> <pre>/<rest>\n if (!this.preserveMultipleSlashes) {\n for (let i = 1; i < parts.length - 1; i++) {\n const p = parts[i]\n // don't squeeze out UNC patterns\n if (i === 1 && p === '' && parts[0] === '') continue\n if (p === '.' || p === '') {\n didSomething = true\n parts.splice(i, 1)\n i--\n }\n }\n if (\n parts[0] === '.' &&\n parts.length === 2 &&\n (parts[1] === '.' || parts[1] === '')\n ) {\n didSomething = true\n parts.pop()\n }\n }\n\n // <pre>/<p>/../<rest> -> <pre>/<rest>\n let dd: number = 0\n while (-1 !== (dd = parts.indexOf('..', dd + 1))) {\n const p = parts[dd - 1]\n if (p && p !== '.' && p !== '..' && p !== '**') {\n didSomething = true\n const needDot = dd === 1 && parts[dd + 1] === '**'\n const splin = needDot ? ['.'] : []\n parts.splice(dd - 1, 2, ...splin)\n if (parts.length === 0) parts.push('')\n dd -= 2\n }\n }\n }\n } while (didSomething)\n\n return globParts\n }\n\n // second phase: multi-pattern dedupes\n // {<pre>/*/<rest>,<pre>/<p>/<rest>} -> <pre>/*/<rest>\n // {<pre>/<rest>,<pre>/<rest>} -> <pre>/<rest>\n // {<pre>/**/<rest>,<pre>/<rest>} -> <pre>/**/<rest>\n //\n // {<pre>/**/<rest>,<pre>/**/<p>/<rest>} -> <pre>/**/<rest>\n // ^-- not valid because ** doens't follow symlinks\n secondPhasePreProcess(globParts: string[][]): string[][] {\n for (let i = 0; i < globParts.length - 1; i++) {\n for (let j = i + 1; j < globParts.length; j++) {\n const matched = this.partsMatch(\n globParts[i],\n globParts[j],\n !this.preserveMultipleSlashes\n )\n if (matched) {\n globParts[i] = []\n globParts[j] = matched\n break\n }\n }\n }\n return globParts.filter(gs => gs.length)\n }\n\n partsMatch(\n a: string[],\n b: string[],\n emptyGSMatch: boolean = false\n ): false | string[] {\n let ai = 0\n let bi = 0\n let result: string[] = []\n let which: string = ''\n while (ai < a.length && bi < b.length) {\n if (a[ai] === b[bi]) {\n result.push(which === 'b' ? b[bi] : a[ai])\n ai++\n bi++\n } else if (emptyGSMatch && a[ai] === '**' && b[bi] === a[ai + 1]) {\n result.push(a[ai])\n ai++\n } else if (emptyGSMatch && b[bi] === '**' && a[ai] === b[bi + 1]) {\n result.push(b[bi])\n bi++\n } else if (\n a[ai] === '*' &&\n b[bi] &&\n (this.options.dot || !b[bi].startsWith('.')) &&\n b[bi] !== '**'\n ) {\n if (which === 'b') return false\n which = 'a'\n result.push(a[ai])\n ai++\n bi++\n } else if (\n b[bi] === '*' &&\n a[ai] &&\n (this.options.dot || !a[ai].startsWith('.')) &&\n a[ai] !== '**'\n ) {\n if (which === 'a') return false\n which = 'b'\n result.push(b[bi])\n ai++\n bi++\n } else {\n return false\n }\n }\n // if we fall out of the loop, it means they two are identical\n // as long as their lengths match\n return a.length === b.length && result\n }\n\n parseNegate() {\n if (this.nonegate) return\n\n const pattern = this.pattern\n let negate = false\n let negateOffset = 0\n\n for (let i = 0; i < pattern.length && pattern.charAt(i) === '!'; i++) {\n negate = !negate\n negateOffset++\n }\n\n if (negateOffset) this.pattern = pattern.slice(negateOffset)\n this.negate = negate\n }\n\n // set partial to true to test if, for example,\n // \"/a/b\" matches the start of \"/*/b/*/d\"\n // Partial means, if you run out of file before you run\n // out of pattern, then that's fine, as long as all\n // the parts match.\n matchOne(file: string[], pattern: ParseReturn[], partial: boolean = false) {\n const options = this.options\n\n // UNC paths like //?/X:/... can match X:/... and vice versa\n // Drive letters in absolute drive or unc paths are always compared\n // case-insensitively.\n if (this.isWindows) {\n const fileDrive = typeof file[0] === 'string' && /^[a-z]:$/i.test(file[0])\n const fileUNC =\n !fileDrive &&\n file[0] === '' &&\n file[1] === '' &&\n file[2] === '?' &&\n /^[a-z]:$/i.test(file[3])\n\n const patternDrive =\n typeof pattern[0] === 'string' && /^[a-z]:$/i.test(pattern[0])\n const patternUNC =\n !patternDrive &&\n pattern[0] === '' &&\n pattern[1] === '' &&\n pattern[2] === '?' &&\n typeof pattern[3] === 'string' &&\n /^[a-z]:$/i.test(pattern[3])\n\n const fdi = fileUNC ? 3 : fileDrive ? 0 : undefined\n const pdi = patternUNC ? 3 : patternDrive ? 0 : undefined\n if (typeof fdi === 'number' && typeof pdi === 'number') {\n const [fd, pd]: [string, string] = [file[fdi], pattern[pdi] as string]\n if (fd.toLowerCase() === pd.toLowerCase()) {\n pattern[pdi] = fd\n if (pdi > fdi) {\n pattern = pattern.slice(pdi)\n } else if (fdi > pdi) {\n file = file.slice(fdi)\n }\n }\n }\n }\n\n // resolve and reduce . and .. portions in the file as well.\n // dont' need to do the second phase, because it's only one string[]\n const { optimizationLevel = 1 } = this.options\n if (optimizationLevel >= 2) {\n file = this.levelTwoFileOptimize(file)\n }\n\n this.debug('matchOne', this, { file, pattern })\n this.debug('matchOne', file.length, pattern.length)\n\n for (\n var fi = 0, pi = 0, fl = file.length, pl = pattern.length;\n fi < fl && pi < pl;\n fi++, pi++\n ) {\n this.debug('matchOne loop')\n var p = pattern[pi]\n var f = file[fi]\n\n this.debug(pattern, p, f)\n\n // should be impossible.\n // some invalid regexp stuff in the set.\n /* c8 ignore start */\n if (p === false) {\n return false\n }\n /* c8 ignore stop */\n\n if (p === GLOBSTAR) {\n this.debug('GLOBSTAR', [pattern, p, f])\n\n // \"**\"\n // a/**/b/**/c would match the following:\n // a/b/x/y/z/c\n // a/x/y/z/b/c\n // a/b/x/b/x/c\n // a/b/c\n // To do this, take the rest of the pattern after\n // the **, and see if it would match the file remainder.\n // If so, return success.\n // If not, the ** \"swallows\" a segment, and try again.\n // This is recursively awful.\n //\n // a/**/b/**/c matching a/b/x/y/z/c\n // - a matches a\n // - doublestar\n // - matchOne(b/x/y/z/c, b/**/c)\n // - b matches b\n // - doublestar\n // - matchOne(x/y/z/c, c) -> no\n // - matchOne(y/z/c, c) -> no\n // - matchOne(z/c, c) -> no\n // - matchOne(c, c) yes, hit\n var fr = fi\n var pr = pi + 1\n if (pr === pl) {\n this.debug('** at the end')\n // a ** at the end will just swallow the rest.\n // We have found a match.\n // however, it will not swallow /.x, unless\n // options.dot is set.\n // . and .. are *never* matched by **, for explosively\n // exponential reasons.\n for (; fi < fl; fi++) {\n if (\n file[fi] === '.' ||\n file[fi] === '..' ||\n (!options.dot && file[fi].charAt(0) === '.')\n )\n return false\n }\n return true\n }\n\n // ok, let's see if we can swallow whatever we can.\n while (fr < fl) {\n var swallowee = file[fr]\n\n this.debug('\\nglobstar while', file, fr, pattern, pr, swallowee)\n\n // XXX remove this slice. Just pass the start index.\n if (this.matchOne(file.slice(fr), pattern.slice(pr), partial)) {\n this.debug('globstar found match!', fr, fl, swallowee)\n // found a match.\n return true\n } else {\n // can't swallow \".\" or \"..\" ever.\n // can only swallow \".foo\" when explicitly asked.\n if (\n swallowee === '.' ||\n swallowee === '..' ||\n (!options.dot && swallowee.charAt(0) === '.')\n ) {\n this.debug('dot detected!', file, fr, pattern, pr)\n break\n }\n\n // ** swallows a segment, and continue.\n this.debug('globstar swallow a segment, and continue')\n fr++\n }\n }\n\n // no match was found.\n // However, in partial mode, we can't say this is necessarily over.\n /* c8 ignore start */\n if (partial) {\n // ran out of file\n this.debug('\\n>>> no match, partial?', file, fr, pattern, pr)\n if (fr === fl) {\n return true\n }\n }\n /* c8 ignore stop */\n return false\n }\n\n // something other than **\n // non-magic patterns just have to match exactly\n // patterns with magic have been turned into regexps.\n let hit: boolean\n if (typeof p === 'string') {\n hit = f === p\n this.debug('string match', p, f, hit)\n } else {\n hit = p.test(f)\n this.debug('pattern match', p, f, hit)\n }\n\n if (!hit) return false\n }\n\n // Note: ending in / means that we'll get a final \"\"\n // at the end of the pattern. This can only match a\n // corresponding \"\" at the end of the file.\n // If the file ends in /, then it can only match a\n // a pattern that ends in /, unless the pattern just\n // doesn't have any more for it. But, a/b/ should *not*\n // match \"a/b/*\", even though \"\" matches against the\n // [^/]*? pattern, except in partial mode, where it might\n // simply not be reached yet.\n // However, a/b/ should still satisfy a/*\n\n // now either we fell off the end of the pattern, or we're done.\n if (fi === fl && pi === pl) {\n // ran out of pattern and filename at the same time.\n // an exact hit!\n return true\n } else if (fi === fl) {\n // ran out of file, but still had pattern left.\n // this is ok if we're doing the match as part of\n // a glob fs traversal.\n return partial\n } else if (pi === pl) {\n // ran out of pattern, still have file left.\n // this is only acceptable if we're on the very last\n // empty segment of a file with a trailing slash.\n // a/* should match a/b/\n return fi === fl - 1 && file[fi] === ''\n\n /* c8 ignore start */\n } else {\n // should be unreachable.\n throw new Error('wtf?')\n }\n /* c8 ignore stop */\n }\n\n braceExpand() {\n return braceExpand(this.pattern, this.options)\n }\n\n parse(pattern: string): ParseReturn {\n assertValidPattern(pattern)\n\n const options = this.options\n\n // shortcuts\n if (pattern === '**') return GLOBSTAR\n if (pattern === '') return ''\n\n // far and away, the most common glob pattern parts are\n // *, *.*, and *.<ext> Add a fast check method for those.\n let m: RegExpMatchArray | null\n let fastTest: null | ((f: string) => boolean) = null\n if ((m = pattern.match(starRE))) {\n fastTest = options.dot ? starTestDot : starTest\n } else if ((m = pattern.match(starDotExtRE))) {\n fastTest = (\n options.nocase\n ? options.dot\n ? starDotExtTestNocaseDot\n : starDotExtTestNocase\n : options.dot\n ? starDotExtTestDot\n : starDotExtTest\n )(m[1])\n } else if ((m = pattern.match(qmarksRE))) {\n fastTest = (\n options.nocase\n ? options.dot\n ? qmarksTestNocaseDot\n : qmarksTestNocase\n : options.dot\n ? qmarksTestDot\n : qmarksTest\n )(m)\n } else if ((m = pattern.match(starDotStarRE))) {\n fastTest = options.dot ? starDotStarTestDot : starDotStarTest\n } else if ((m = pattern.match(dotStarRE))) {\n fastTest = dotStarTest\n }\n\n const re = AST.fromGlob(pattern, this.options).toMMPattern()\n if (fastTest && typeof re === 'object') {\n // Avoids overriding in frozen environments\n Reflect.defineProperty(re, 'test', { value: fastTest })\n }\n return re\n }\n\n makeRe() {\n if (this.regexp || this.regexp === false) return this.regexp\n\n // at this point, this.set is a 2d array of partial\n // pattern strings, or \"**\".\n //\n // It's better to use .match(). This function shouldn't\n // be used, really, but it's pretty convenient sometimes,\n // when you just want to work with a regex.\n const set = this.set\n\n if (!set.length) {\n this.regexp = false\n return this.regexp\n }\n const options = this.options\n\n const twoStar = options.noglobstar\n ? star\n : options.dot\n ? twoStarDot\n : twoStarNoDot\n const flags = new Set(options.nocase ? ['i'] : [])\n\n // regexpify non-globstar patterns\n // if ** is only item, then we just do one twoStar\n // if ** is first, and there are more, prepend (\\/|twoStar\\/)? to next\n // if ** is last, append (\\/twoStar|) to previous\n // if ** is in the middle, append (\\/|\\/twoStar\\/) to previous\n // then filter out GLOBSTAR symbols\n let re = set\n .map(pattern => {\n const pp: (string | typeof GLOBSTAR)[] = pattern.map(p => {\n if (p instanceof RegExp) {\n for (const f of p.flags.split('')) flags.add(f)\n }\n return typeof p === 'string'\n ? regExpEscape(p)\n : p === GLOBSTAR\n ? GLOBSTAR\n : p._src\n }) as (string | typeof GLOBSTAR)[]\n pp.forEach((p, i) => {\n const next = pp[i + 1]\n const prev = pp[i - 1]\n if (p !== GLOBSTAR || prev === GLOBSTAR) {\n return\n }\n if (prev === undefined) {\n if (next !== undefined && next !== GLOBSTAR) {\n pp[i + 1] = '(?:\\\\/|' + twoStar + '\\\\/)?' + next\n } else {\n pp[i] = twoStar\n }\n } else if (next === undefined) {\n pp[i - 1] = prev + '(?:\\\\/|' + twoStar + ')?'\n } else if (next !== GLOBSTAR) {\n pp[i - 1] = prev + '(?:\\\\/|\\\\/' + twoStar + '\\\\/)' + next\n pp[i + 1] = GLOBSTAR\n }\n })\n return pp.filter(p => p !== GLOBSTAR).join('/')\n })\n .join('|')\n\n // need to wrap in parens if we had more than one thing with |,\n // otherwise only the first will be anchored to ^ and the last to $\n const [open, close] = set.length > 1 ? ['(?:', ')'] : ['', '']\n // must match entire pattern\n // ending in a * or ** will make it less strict.\n re = '^' + open + re + close + '$'\n\n // can match anything, as long as it's not this.\n if (this.negate) re = '^(?!' + re + ').+$'\n\n try {\n this.regexp = new RegExp(re, [...flags].join(''))\n /* c8 ignore start */\n } catch (ex) {\n // should be impossible\n this.regexp = false\n }\n /* c8 ignore stop */\n return this.regexp\n }\n\n slashSplit(p: string) {\n // if p starts with // on windows, we preserve that\n // so that UNC paths aren't broken. Otherwise, any number of\n // / characters are coalesced into one, unless\n // preserveMultipleSlashes is set to true.\n if (this.preserveMultipleSlashes) {\n return p.split('/')\n } else if (this.isWindows && /^\\/\\/[^\\/]+/.test(p)) {\n // add an extra '' for the one we lose\n return ['', ...p.split(/\\/+/)]\n } else {\n return p.split(/\\/+/)\n }\n }\n\n match(f: string, partial = this.partial) {\n this.debug('match', f, this.pattern)\n // short-circuit in the case of busted things.\n // comments, etc.\n if (this.comment) {\n return false\n }\n if (this.empty) {\n return f === ''\n }\n\n if (f === '/' && partial) {\n return true\n }\n\n const options = this.options\n\n // windows: need to use /, not \\\n if (this.isWindows) {\n f = f.split('\\\\').join('/')\n }\n\n // treat the test path as a set of pathparts.\n const ff = this.slashSplit(f)\n this.debug(this.pattern, 'split', ff)\n\n // just ONE of the pattern sets in this.set needs to match\n // in order for it to be valid. If negating, then just one\n // match means that we have failed.\n // Either way, return on the first hit.\n\n const set = this.set\n this.debug(this.pattern, 'set', set)\n\n // Find the basename of the path by looking for the last non-empty segment\n let filename: string = ff[ff.length - 1]\n if (!filename) {\n for (let i = ff.length - 2; !filename && i >= 0; i--) {\n filename = ff[i]\n }\n }\n\n for (let i = 0; i < set.length; i++) {\n const pattern = set[i]\n let file = ff\n if (options.matchBase && pattern.length === 1) {\n file = [filename]\n }\n const hit = this.matchOne(file, pattern, partial)\n if (hit) {\n if (options.flipNegate) {\n return true\n }\n return !this.negate\n }\n }\n\n // didn't get any hits. this is success if it's a negative\n // pattern, failure otherwise.\n if (options.flipNegate) {\n return false\n }\n return this.negate\n }\n\n static defaults(def: MinimatchOptions) {\n return minimatch.defaults(def).Minimatch\n }\n}\n/* c8 ignore start */\nexport { AST } from './ast.js'\nexport { escape } from './escape.js'\nexport { unescape } from './unescape.js'\n/* c8 ignore stop */\nminimatch.AST = AST\nminimatch.Minimatch = Minimatch\nminimatch.escape = escape\nminimatch.unescape = unescape\n","const MAX_PATTERN_LENGTH = 1024 * 64\nexport const assertValidPattern: (pattern: any) => void = (\n pattern: any\n): asserts pattern is string => {\n if (typeof pattern !== 'string') {\n throw new TypeError('invalid pattern')\n }\n\n if (pattern.length > MAX_PATTERN_LENGTH) {\n throw new TypeError('pattern is too long')\n }\n}\n","// translate the various posix character classes into unicode properties\n// this works across all unicode locales\n\n// { <posix class>: [<translation>, /u flag required, negated]\nconst posixClasses: { [k: string]: [e: string, u: boolean, n?: boolean] } = {\n '[:alnum:]': ['\\\\p{L}\\\\p{Nl}\\\\p{Nd}', true],\n '[:alpha:]': ['\\\\p{L}\\\\p{Nl}', true],\n '[:ascii:]': ['\\\\x' + '00-\\\\x' + '7f', false],\n '[:blank:]': ['\\\\p{Zs}\\\\t', true],\n '[:cntrl:]': ['\\\\p{Cc}', true],\n '[:digit:]': ['\\\\p{Nd}', true],\n '[:graph:]': ['\\\\p{Z}\\\\p{C}', true, true],\n '[:lower:]': ['\\\\p{Ll}', true],\n '[:print:]': ['\\\\p{C}', true],\n '[:punct:]': ['\\\\p{P}', true],\n '[:space:]': ['\\\\p{Z}\\\\t\\\\r\\\\n\\\\v\\\\f', true],\n '[:upper:]': ['\\\\p{Lu}', true],\n '[:word:]': ['\\\\p{L}\\\\p{Nl}\\\\p{Nd}\\\\p{Pc}', true],\n '[:xdigit:]': ['A-Fa-f0-9', false],\n}\n\n// only need to escape a few things inside of brace expressions\n// escapes: [ \\ ] -\nconst braceEscape = (s: string) => s.replace(/[[\\]\\\\-]/g, '\\\\$&')\n// escape all regexp magic characters\nconst regexpEscape = (s: string) =>\n s.replace(/[-[\\]{}()*+?.,\\\\^$|#\\s]/g, '\\\\$&')\n\n// everything has already been escaped, we just have to join\nconst rangesToString = (ranges: string[]): string => ranges.join('')\n\nexport type ParseClassResult = [\n src: string,\n uFlag: boolean,\n consumed: number,\n hasMagic: boolean\n]\n\n// takes a glob string at a posix brace expression, and returns\n// an equivalent regular expression source, and boolean indicating\n// whether the /u flag needs to be applied, and the number of chars\n// consumed to parse the character class.\n// This also removes out of order ranges, and returns ($.) if the\n// entire class just no good.\nexport const parseClass = (\n glob: string,\n position: number\n): ParseClassResult => {\n const pos = position\n /* c8 ignore start */\n if (glob.charAt(pos) !== '[') {\n throw new Error('not in a brace expression')\n }\n /* c8 ignore stop */\n const ranges: string[] = []\n const negs: string[] = []\n\n let i = pos + 1\n let sawStart = false\n let uflag = false\n let escaping = false\n let negate = false\n let endPos = pos\n let rangeStart = ''\n WHILE: while (i < glob.length) {\n const c = glob.charAt(i)\n if ((c === '!' || c === '^') && i === pos + 1) {\n negate = true\n i++\n continue\n }\n\n if (c === ']' && sawStart && !escaping) {\n endPos = i + 1\n break\n }\n\n sawStart = true\n if (c === '\\\\') {\n if (!escaping) {\n escaping = true\n i++\n continue\n }\n // escaped \\ char, fall through and treat like normal char\n }\n if (c === '[' && !escaping) {\n // either a posix class, a collation equivalent, or just a [\n for (const [cls, [unip, u, neg]] of Object.entries(posixClasses)) {\n if (glob.startsWith(cls, i)) {\n // invalid, [a-[] is fine, but not [a-[:alpha]]\n if (rangeStart) {\n return ['$.', false, glob.length - pos, true]\n }\n i += cls.length\n if (neg) negs.push(unip)\n else ranges.push(unip)\n uflag = uflag || u\n continue WHILE\n }\n }\n }\n\n // now it's just a normal character, effectively\n escaping = false\n if (rangeStart) {\n // throw this range away if it's not valid, but others\n // can still match.\n if (c > rangeStart) {\n ranges.push(braceEscape(rangeStart) + '-' + braceEscape(c))\n } else if (c === rangeStart) {\n ranges.push(braceEscape(c))\n }\n rangeStart = ''\n i++\n continue\n }\n\n // now might be the start of a range.\n // can be either c-d or c-] or c<more...>] or c] at this point\n if (glob.startsWith('-]', i + 1)) {\n ranges.push(braceEscape(c + '-'))\n i += 2\n continue\n }\n if (glob.startsWith('-', i + 1)) {\n rangeStart = c\n i += 2\n continue\n }\n\n // not the start of a range, just a single character\n ranges.push(braceEscape(c))\n i++\n }\n\n if (endPos < i) {\n // didn't see the end of the class, not a valid class,\n // but might still be valid as a literal match.\n return ['', false, 0, false]\n }\n\n // if we got no ranges and no negates, then we have a range that\n // cannot possibly match anything, and that poisons the whole glob\n if (!ranges.length && !negs.length) {\n return ['$.', false, glob.length - pos, true]\n }\n\n // if we got one positive range, and it's a single character, then that's\n // not actually a magic pattern, it's just that one literal character.\n // we should not treat that as \"magic\", we should just return the literal\n // character. [_] is a perfectly valid way to escape glob magic chars.\n if (\n negs.length === 0 &&\n ranges.length === 1 &&\n /^\\\\?.$/.test(ranges[0]) &&\n !negate\n ) {\n const r = ranges[0].length === 2 ? ranges[0].slice(-1) : ranges[0]\n return [regexpEscape(r), false, endPos - pos, false]\n }\n\n const sranges = '[' + (negate ? '^' : '') + rangesToString(ranges) + ']'\n const snegs = '[' + (negate ? '' : '^') + rangesToString(negs) + ']'\n const comb =\n ranges.length && negs.length\n ? '(' + sranges + '|' + snegs + ')'\n : ranges.length\n ? sranges\n : snegs\n\n return [comb, uflag, endPos - pos, true]\n}\n","import { MinimatchOptions } from './index.js'\n/**\n * Un-escape a string that has been escaped with {@link escape}.\n *\n * If the {@link windowsPathsNoEscape} option is used, then square-brace\n * escapes are removed, but not backslash escapes. For example, it will turn\n * the string `'[*]'` into `*`, but it will not turn `'\\\\*'` into `'*'`,\n * becuase `\\` is a path separator in `windowsPathsNoEscape` mode.\n *\n * When `windowsPathsNoEscape` is not set, then both brace escapes and\n * backslash escapes are removed.\n *\n * Slashes (and backslashes in `windowsPathsNoEscape` mode) cannot be escaped\n * or unescaped.\n */\nexport const unescape = (\n s: string,\n {\n windowsPathsNoEscape = false,\n }: Pick<MinimatchOptions, 'windowsPathsNoEscape'> = {}\n) => {\n return windowsPathsNoEscape\n ? s.replace(/\\[([^\\/\\\\])\\]/g, '$1')\n : s.replace(/((?!\\\\).|^)\\[([^\\/\\\\])\\]/g, '$1$2').replace(/\\\\([^\\/])/g, '$1')\n}\n","// parse a single path portion\n\nimport { parseClass } from './brace-expressions.js'\nimport { MinimatchOptions, MMRegExp } from './index.js'\nimport { unescape } from './unescape.js'\n\n// classes [] are handled by the parseClass method\n// for positive extglobs, we sub-parse the contents, and combine,\n// with the appropriate regexp close.\n// for negative extglobs, we sub-parse the contents, but then\n// have to include the rest of the pattern, then the parent, etc.,\n// as the thing that cannot be because RegExp negative lookaheads\n// are different from globs.\n//\n// So for example:\n// a@(i|w!(x|y)z|j)b => ^a(i|w((!?(x|y)zb).*)z|j)b$\n// 1 2 3 4 5 6 1 2 3 46 5 6\n//\n// Assembling the extglob requires not just the negated patterns themselves,\n// but also anything following the negative patterns up to the boundary\n// of the current pattern, plus anything following in the parent pattern.\n//\n//\n// So, first, we parse the string into an AST of extglobs, without turning\n// anything into regexps yet.\n//\n// ['a', {@ [['i'], ['w', {!['x', 'y']}, 'z'], ['j']]}, 'b']\n//\n// Then, for all the negative extglobs, we append whatever comes after in\n// each parent as their tail\n//\n// ['a', {@ [['i'], ['w', {!['x', 'y'], 'z', 'b'}, 'z'], ['j']]}, 'b']\n//\n// Lastly, we turn each of these pieces into a regexp, and join\n//\n// v----- .* because there's more following,\n// v v otherwise, .+ because it must be\n// v v *something* there.\n// ['^a', {@ ['i', 'w(?:(!?(?:x|y).*zb$).*)z', 'j' ]}, 'b$']\n// copy what follows into here--^^^^^\n// ['^a', '(?:i|w(?:(?!(?:x|y).*zb$).*)z|j)', 'b$']\n// ['^a(?:i|w(?:(?!(?:x|y).*zb$).*)z|j)b$']\n\nexport type ExtglobType = '!' | '?' | '+' | '*' | '@'\nconst types = new Set<ExtglobType>(['!', '?', '+', '*', '@'])\nconst isExtglobType = (c: string): c is ExtglobType =>\n types.has(c as ExtglobType)\n\n// Patterns that get prepended to bind to the start of either the\n// entire string, or just a single path portion, to prevent dots\n// and/or traversal patterns, when needed.\n// Exts don't need the ^ or / bit, because the root binds that already.\nconst startNoTraversal = '(?!(?:^|/)\\\\.\\\\.?(?:$|/))'\nconst startNoDot = '(?!\\\\.)'\n\n// characters that indicate a start of pattern needs the \"no dots\" bit,\n// because a dot *might* be matched. ( is not in the list, because in\n// the case of a child extglob, it will handle the prevention itself.\nconst addPatternStart = new Set(['[', '.'])\n// cases where traversal is A-OK, no dot prevention needed\nconst justDots = new Set(['..', '.'])\nconst reSpecials = new Set('().*{}+?[]^$\\\\!')\nconst regExpEscape = (s: string) =>\n s.replace(/[-[\\]{}()*+?.,\\\\^$|#\\s]/g, '\\\\$&')\n\n// any single thing other than /\nconst qmark = '[^/]'\n\n// * => any number of characters\nconst star = qmark + '*?'\n// use + when we need to ensure that *something* matches, because the * is\n// the only thing in the path portion.\nconst starNoEmpty = qmark + '+?'\n\n// remove the \\ chars that we added if we end up doing a nonmagic compare\n// const deslash = (s: string) => s.replace(/\\\\(.)/g, '$1')\n\nexport class AST {\n type: ExtglobType | null\n readonly #root: AST\n\n #hasMagic?: boolean\n #uflag: boolean = false\n #parts: (string | AST)[] = []\n readonly #parent?: AST\n readonly #parentIndex: number\n #negs: AST[]\n #filledNegs: boolean = false\n #options: MinimatchOptions\n #toString?: string\n // set to true if it's an extglob with no children\n // (which really means one child of '')\n #emptyExt: boolean = false\n\n constructor(\n type: ExtglobType | null,\n parent?: AST,\n options: MinimatchOptions = {}\n ) {\n this.type = type\n // extglobs are inherently magical\n if (type) this.#hasMagic = true\n this.#parent = parent\n this.#root = this.#parent ? this.#parent.#root : this\n this.#options = this.#root === this ? options : this.#root.#options\n this.#negs = this.#root === this ? [] : this.#root.#negs\n if (type === '!' && !this.#root.#filledNegs) this.#negs.push(this)\n this.#parentIndex = this.#parent ? this.#parent.#parts.length : 0\n }\n\n get hasMagic(): boolean | undefined {\n /* c8 ignore start */\n if (this.#hasMagic !== undefined) return this.#hasMagic\n /* c8 ignore stop */\n for (const p of this.#parts) {\n if (typeof p === 'string') continue\n if (p.type || p.hasMagic) return (this.#hasMagic = true)\n }\n // note: will be undefined until we generate the regexp src and find out\n return this.#hasMagic\n }\n\n // reconstructs the pattern\n toString(): string {\n if (this.#toString !== undefined) return this.#toString\n if (!this.type) {\n return (this.#toString = this.#parts.map(p => String(p)).join(''))\n } else {\n return (this.#toString =\n this.type + '(' + this.#parts.map(p => String(p)).join('|') + ')')\n }\n }\n\n #fillNegs() {\n /* c8 ignore start */\n if (this !== this.#root) throw new Error('should only call on root')\n if (this.#filledNegs) return this\n /* c8 ignore stop */\n\n // call toString() once to fill this out\n this.toString()\n this.#filledNegs = true\n let n: AST | undefined\n while ((n = this.#negs.pop())) {\n if (n.type !== '!') continue\n // walk up the tree, appending everthing that comes AFTER parentIndex\n let p: AST | undefined = n\n let pp = p.#parent\n while (pp) {\n for (\n let i = p.#parentIndex + 1;\n !pp.type && i < pp.#parts.length;\n i++\n ) {\n for (const part of n.#parts) {\n /* c8 ignore start */\n if (typeof part === 'string') {\n throw new Error('string part in extglob AST??')\n }\n /* c8 ignore stop */\n part.copyIn(pp.#parts[i])\n }\n }\n p = pp\n pp = p.#parent\n }\n }\n return this\n }\n\n push(...parts: (string | AST)[]) {\n for (const p of parts) {\n if (p === '') continue\n /* c8 ignore start */\n if (typeof p !== 'string' && !(p instanceof AST && p.#parent === this)) {\n throw new Error('invalid part: ' + p)\n }\n /* c8 ignore stop */\n this.#parts.push(p)\n }\n }\n\n toJSON() {\n const ret: any[] =\n this.type === null\n ? this.#parts.slice().map(p => (typeof p === 'string' ? p : p.toJSON()))\n : [this.type, ...this.#parts.map(p => (p as AST).toJSON())]\n if (this.isStart() && !this.type) ret.unshift([])\n if (\n this.isEnd() &&\n (this === this.#root ||\n (this.#root.#filledNegs && this.#parent?.type === '!'))\n ) {\n ret.push({})\n }\n return ret\n }\n\n isStart(): boolean {\n if (this.#root === this) return true\n // if (this.type) return !!this.#parent?.isStart()\n if (!this.#parent?.isStart()) return false\n if (this.#parentIndex === 0) return true\n // if everything AHEAD of this is a negation, then it's still the \"start\"\n const p = this.#parent\n for (let i = 0; i < this.#parentIndex; i++) {\n const pp = p.#parts[i]\n if (!(pp instanceof AST && pp.type === '!')) {\n return false\n }\n }\n return true\n }\n\n isEnd(): boolean {\n if (this.#root === this) return true\n if (this.#parent?.type === '!') return true\n if (!this.#parent?.isEnd()) return false\n if (!this.type) return this.#parent?.isEnd()\n // if not root, it'll always have a parent\n /* c8 ignore start */\n const pl = this.#parent ? this.#parent.#parts.length : 0\n /* c8 ignore stop */\n return this.#parentIndex === pl - 1\n }\n\n copyIn(part: AST | string) {\n if (typeof part === 'string') this.push(part)\n else this.push(part.clone(this))\n }\n\n clone(parent: AST) {\n const c = new AST(this.type, parent)\n for (const p of this.#parts) {\n c.copyIn(p)\n }\n return c\n }\n\n static #parseAST(\n str: string,\n ast: AST,\n pos: number,\n opt: MinimatchOptions\n ): number {\n let escaping = false\n let inBrace = false\n let braceStart = -1\n let braceNeg = false\n if (ast.type === null) {\n // outside of a extglob, append until we find a start\n let i = pos\n let acc = ''\n while (i < str.length) {\n const c = str.charAt(i++)\n // still accumulate escapes at this point, but we do ignore\n // starts that are escaped\n if (escaping || c === '\\\\') {\n escaping = !escaping\n acc += c\n continue\n }\n\n if (inBrace) {\n if (i === braceStart + 1) {\n if (c === '^' || c === '!') {\n braceNeg = true\n }\n } else if (c === ']' && !(i === braceStart + 2 && braceNeg)) {\n inBrace = false\n }\n acc += c\n continue\n } else if (c === '[') {\n inBrace = true\n braceStart = i\n braceNeg = false\n acc += c\n continue\n }\n\n if (!opt.noext && isExtglobType(c) && str.charAt(i) === '(') {\n ast.push(acc)\n acc = ''\n const ext = new AST(c, ast)\n i = AST.#parseAST(str, ext, i, opt)\n ast.push(ext)\n continue\n }\n acc += c\n }\n ast.push(acc)\n return i\n }\n\n // some kind of extglob, pos is at the (\n // find the next | or )\n let i = pos + 1\n let part = new AST(null, ast)\n const parts: AST[] = []\n let acc = ''\n while (i < str.length) {\n const c = str.charAt(i++)\n // still accumulate escapes at this point, but we do ignore\n // starts that are escaped\n if (escaping || c === '\\\\') {\n escaping = !escaping\n acc += c\n continue\n }\n\n if (inBrace) {\n if (i === braceStart + 1) {\n if (c === '^' || c === '!') {\n braceNeg = true\n }\n } else if (c === ']' && !(i === braceStart + 2 && braceNeg)) {\n inBrace = false\n }\n acc += c\n continue\n } else if (c === '[') {\n inBrace = true\n braceStart = i\n braceNeg = false\n acc += c\n continue\n }\n\n if (isExtglobType(c) && str.charAt(i) === '(') {\n part.push(acc)\n acc = ''\n const ext = new AST(c, part)\n part.push(ext)\n i = AST.#parseAST(str, ext, i, opt)\n continue\n }\n if (c === '|') {\n part.push(acc)\n acc = ''\n parts.push(part)\n part = new AST(null, ast)\n continue\n }\n if (c === ')') {\n if (acc === '' && ast.#parts.length === 0) {\n ast.#emptyExt = true\n }\n part.push(acc)\n acc = ''\n ast.push(...parts, part)\n return i\n }\n acc += c\n }\n\n // unfinished extglob\n // if we got here, it was a malformed extglob! not an extglob, but\n // maybe something else in there.\n ast.type = null\n ast.#hasMagic = undefined\n ast.#parts = [str.substring(pos - 1)]\n return i\n }\n\n static fromGlob(pattern: string, options: MinimatchOptions = {}) {\n const ast = new AST(null, undefined, options)\n AST.#parseAST(pattern, ast, 0, options)\n return ast\n }\n\n // returns the regular expression if there's magic, or the unescaped\n // string if not.\n toMMPattern(): MMRegExp | string {\n // should only be called on root\n /* c8 ignore start */\n if (this !== this.#root) return this.#root.toMMPattern()\n /* c8 ignore stop */\n const glob = this.toString()\n const [re, body, hasMagic, uflag] = this.toRegExpSource()\n // if we're in nocase mode, and not nocaseMagicOnly, then we do\n // still need a regular expression if we have to case-insensitively\n // match capital/lowercase characters.\n const anyMagic =\n hasMagic ||\n this.#hasMagic ||\n (this.#options.nocase &&\n !this.#options.nocaseMagicOnly &&\n glob.toUpperCase() !== glob.toLowerCase())\n if (!anyMagic) {\n return body\n }\n\n const flags = (this.#options.nocase ? 'i' : '') + (uflag ? 'u' : '')\n return Object.assign(new RegExp(`^${re}$`, flags), {\n _src: re,\n _glob: glob,\n })\n }\n\n get options() {\n return this.#options\n }\n\n // returns the string match, the regexp source, whether there's magic\n // in the regexp (so a regular expression is required) and whether or\n // not the uflag is needed for the regular expression (for posix classes)\n // TODO: instead of injecting the start/end at this point, just return\n // the BODY of the regexp, along with the start/end portions suitable\n // for binding the start/end in either a joined full-path makeRe context\n // (where we bind to (^|/), or a standalone matchPart context (where\n // we bind to ^, and not /). Otherwise slashes get duped!\n //\n // In part-matching mode, the start is:\n // - if not isStart: nothing\n // - if traversal possible, but not allowed: ^(?!\\.\\.?$)\n // - if dots allowed or not possible: ^\n // - if dots possible and not allowed: ^(?!\\.)\n // end is:\n // - if not isEnd(): nothing\n // - else: $\n //\n // In full-path matching mode, we put the slash at the START of the\n // pattern, so start is:\n // - if first pattern: same as part-matching mode\n // - if not isStart(): nothing\n // - if traversal possible, but not allowed: /(?!\\.\\.?(?:$|/))\n // - if dots allowed or not possible: /\n // - if dots possible and not allowed: /(?!\\.)\n // end is:\n // - if last pattern, same as part-matching mode\n // - else nothing\n //\n // Always put the (?:$|/) on negated tails, though, because that has to be\n // there to bind the end of the negated pattern portion, and it's easier to\n // just stick it in now rather than try to inject it later in the middle of\n // the pattern.\n //\n // We can just always return the same end, and leave it up to the caller\n // to know whether it's going to be used joined or in parts.\n // And, if the start is adjusted slightly, can do the same there:\n // - if not isStart: nothing\n // - if traversal possible, but not allowed: (?:/|^)(?!\\.\\.?$)\n // - if dots allowed or not possible: (?:/|^)\n // - if dots possible and not allowed: (?:/|^)(?!\\.)\n //\n // But it's better to have a simpler binding without a conditional, for\n // performance, so probably better to return both start options.\n //\n // Then the caller just ignores the end if it's not the first pattern,\n // and the start always gets applied.\n //\n // But that's always going to be $ if it's the ending pattern, or nothing,\n // so the caller can just attach $ at the end of the pattern when building.\n //\n // So the todo is:\n // - better detect what kind of start is needed\n // - return both flavors of starting pattern\n // - attach $ at the end of the pattern when creating the actual RegExp\n //\n // Ah, but wait, no, that all only applies to the root when the first pattern\n // is not an extglob. If the first pattern IS an extglob, then we need all\n // that dot prevention biz to live in the extglob portions, because eg\n // +(*|.x*) can match .xy but not .yx.\n //\n // So, return the two flavors if it's #root and the first child is not an\n // AST, otherwise leave it to the child AST to handle it, and there,\n // use the (?:^|/) style of start binding.\n //\n // Even simplified further:\n // - Since the start for a join is eg /(?!\\.) and the start for a part\n // is ^(?!\\.), we can just prepend (?!\\.) to the pattern (either root\n // or start or whatever) and prepend ^ or / at the Regexp construction.\n toRegExpSource(\n allowDot?: boolean\n ): [re: string, body: string, hasMagic: boolean, uflag: boolean] {\n const dot = allowDot ?? !!this.#options.dot\n if (this.#root === this) this.#fillNegs()\n if (!this.type) {\n const noEmpty = this.isStart() && this.isEnd()\n const src = this.#parts\n .map(p => {\n const [re, _, hasMagic, uflag] =\n typeof p === 'string'\n ? AST.#parseGlob(p, this.#hasMagic, noEmpty)\n : p.toRegExpSource(allowDot)\n this.#hasMagic = this.#hasMagic || hasMagic\n this.#uflag = this.#uflag || uflag\n return re\n })\n .join('')\n\n let start = ''\n if (this.isStart()) {\n if (typeof this.#parts[0] === 'string') {\n // this is the string that will match the start of the pattern,\n // so we need to protect against dots and such.\n\n // '.' and '..' cannot match unless the pattern is that exactly,\n // even if it starts with . or dot:true is set.\n const dotTravAllowed =\n this.#parts.length === 1 && justDots.has(this.#parts[0])\n if (!dotTravAllowed) {\n const aps = addPatternStart\n // check if we have a possibility of matching . or ..,\n // and prevent that.\n const needNoTrav =\n // dots are allowed, and the pattern starts with [ or .\n (dot && aps.has(src.charAt(0))) ||\n // the pattern starts with \\., and then [ or .\n (src.startsWith('\\\\.') && aps.has(src.charAt(2))) ||\n // the pattern starts with \\.\\., and then [ or .\n (src.startsWith('\\\\.\\\\.') && aps.has(src.charAt(4)))\n // no need to prevent dots if it can't match a dot, or if a\n // sub-pattern will be preventing it anyway.\n const needNoDot = !dot && !allowDot && aps.has(src.charAt(0))\n\n start = needNoTrav ? startNoTraversal : needNoDot ? startNoDot : ''\n }\n }\n }\n\n // append the \"end of path portion\" pattern to negation tails\n let end = ''\n if (\n this.isEnd() &&\n this.#root.#filledNegs &&\n this.#parent?.type === '!'\n ) {\n end = '(?:$|\\\\/)'\n }\n const final = start + src + end\n return [\n final,\n unescape(src),\n (this.#hasMagic = !!this.#hasMagic),\n this.#uflag,\n ]\n }\n\n // We need to calculate the body *twice* if it's a repeat pattern\n // at the start, once in nodot mode, then again in dot mode, so a\n // pattern like *(?) can match 'x.y'\n\n const repeated = this.type === '*' || this.type === '+'\n // some kind of extglob\n const start = this.type === '!' ? '(?:(?!(?:' : '(?:'\n let body = this.#partsToRegExp(dot)\n\n if (this.isStart() && this.isEnd() && !body && this.type !== '!') {\n // invalid extglob, has to at least be *something* present, if it's\n // the entire path portion.\n const s = this.toString()\n this.#parts = [s]\n this.type = null\n this.#hasMagic = undefined\n return [s, unescape(this.toString()), false, false]\n }\n\n // XXX abstract out this map method\n let bodyDotAllowed =\n !repeated || allowDot || dot || !startNoDot\n ? ''\n : this.#partsToRegExp(true)\n if (bodyDotAllowed === body) {\n bodyDotAllowed = ''\n }\n if (bodyDotAllowed) {\n body = `(?:${body})(?:${bodyDotAllowed})*?`\n }\n\n // an empty !() is exactly equivalent to a starNoEmpty\n let final = ''\n if (this.type === '!' && this.#emptyExt) {\n final = (this.isStart() && !dot ? startNoDot : '') + starNoEmpty\n } else {\n const close =\n this.type === '!'\n ? // !() must match something,but !(x) can match ''\n '))' +\n (this.isStart() && !dot && !allowDot ? startNoDot : '') +\n star +\n ')'\n : this.type === '@'\n ? ')'\n : this.type === '?'\n ? ')?'\n : this.type === '+' && bodyDotAllowed\n ? ')'\n : this.type === '*' && bodyDotAllowed\n ? `)?`\n : `)${this.type}`\n final = start + body + close\n }\n return [\n final,\n unescape(body),\n (this.#hasMagic = !!this.#hasMagic),\n this.#uflag,\n ]\n }\n\n #partsToRegExp(dot: boolean) {\n return this.#parts\n .map(p => {\n // extglob ASTs should only contain parent ASTs\n /* c8 ignore start */\n if (typeof p === 'string') {\n throw new Error('string type in extglob ast??')\n }\n /* c8 ignore stop */\n // can ignore hasMagic, because extglobs are already always magic\n const [re, _, _hasMagic, uflag] = p.toRegExpSource(dot)\n this.#uflag = this.#uflag || uflag\n return re\n })\n .filter(p => !(this.isStart() && this.isEnd()) || !!p)\n .join('|')\n }\n\n static #parseGlob(\n glob: string,\n hasMagic: boolean | undefined,\n noEmpty: boolean = false\n ): [re: string, body: string, hasMagic: boolean, uflag: boolean] {\n let escaping = false\n let re = ''\n let uflag = false\n for (let i = 0; i < glob.length; i++) {\n const c = glob.charAt(i)\n if (escaping) {\n escaping = false\n re += (reSpecials.has(c) ? '\\\\' : '') + c\n continue\n }\n if (c === '\\\\') {\n if (i === glob.length - 1) {\n re += '\\\\\\\\'\n } else {\n escaping = true\n }\n continue\n }\n if (c === '[') {\n const [src, needUflag, consumed, magic] = parseClass(glob, i)\n if (consumed) {\n re += src\n uflag = uflag || needUflag\n i += consumed - 1\n hasMagic = hasMagic || magic\n continue\n }\n }\n if (c === '*') {\n if (noEmpty && glob === '*') re += starNoEmpty\n else re += star\n hasMagic = true\n continue\n }\n if (c === '?') {\n re += qmark\n hasMagic = true\n continue\n }\n re += regExpEscape(c)\n }\n return [re, unescape(glob), !!hasMagic, uflag]\n }\n}\n","import { MinimatchOptions } from './index.js'\n/**\n * Escape all magic characters in a glob pattern.\n *\n * If the {@link windowsPathsNoEscape | GlobOptions.windowsPathsNoEscape}\n * option is used, then characters are escaped by wrapping in `[]`, because\n * a magic character wrapped in a character class can only be satisfied by\n * that exact character. In this mode, `\\` is _not_ escaped, because it is\n * not interpreted as a magic character, but instead as a path separator.\n */\nexport const escape = (\n s: string,\n {\n windowsPathsNoEscape = false,\n }: Pick<MinimatchOptions, 'windowsPathsNoEscape'> = {}\n) => {\n // don't need to escape +@! because we escape the parens\n // that make those magic, and escaping ! as [!] isn't valid,\n // because [!]] is a valid glob class meaning not ']'.\n return windowsPathsNoEscape\n ? s.replace(/[?*()[\\]]/g, '[$&]')\n : s.replace(/[?*()[\\]\\\\]/g, '\\\\$&')\n}\n","import { Minimatch, MinimatchOptions } from 'minimatch'\nimport { Minipass } from 'minipass'\nimport { fileURLToPath } from 'node:url'\nimport {\n FSOption,\n Path,\n PathScurry,\n PathScurryDarwin,\n PathScurryPosix,\n PathScurryWin32,\n} from 'path-scurry'\nimport { IgnoreLike } from './ignore.js'\nimport { Pattern } from './pattern.js'\nimport { GlobStream, GlobWalker } from './walker.js'\n\nexport type MatchSet = Minimatch['set']\nexport type GlobParts = Exclude<Minimatch['globParts'], undefined>\n\n// if no process global, just call it linux.\n// so we default to case-sensitive, / separators\nconst defaultPlatform: NodeJS.Platform =\n (\n typeof process === 'object' &&\n process &&\n typeof process.platform === 'string'\n ) ?\n process.platform\n : 'linux'\n\n/**\n * A `GlobOptions` object may be provided to any of the exported methods, and\n * must be provided to the `Glob` constructor.\n *\n * All options are optional, boolean, and false by default, unless otherwise\n * noted.\n *\n * All resolved options are added to the Glob object as properties.\n *\n * If you are running many `glob` operations, you can pass a Glob object as the\n * `options` argument to a subsequent operation to share the previously loaded\n * cache.\n */\nexport interface GlobOptions {\n /**\n * Set to `true` to always receive absolute paths for\n * matched files. Set to `false` to always return relative paths.\n *\n * When this option is not set, absolute paths are returned for patterns\n * that are absolute, and otherwise paths are returned that are relative\n * to the `cwd` setting.\n *\n * This does _not_ make an extra system call to get\n * the realpath, it only does string path resolution.\n *\n * Conflicts with {@link withFileTypes}\n */\n absolute?: boolean\n\n /**\n * Set to false to enable {@link windowsPathsNoEscape}\n *\n * @deprecated\n */\n allowWindowsEscape?: boolean\n\n /**\n * The current working directory in which to search. Defaults to\n * `process.cwd()`.\n *\n * May be eiher a string path or a `file://` URL object or string.\n */\n cwd?: string | URL\n\n /**\n * Include `.dot` files in normal matches and `globstar`\n * matches. Note that an explicit dot in a portion of the pattern\n * will always match dot files.\n */\n dot?: boolean\n\n /**\n * Prepend all relative path strings with `./` (or `.\\` on Windows).\n *\n * Without this option, returned relative paths are \"bare\", so instead of\n * returning `'./foo/bar'`, they are returned as `'foo/bar'`.\n *\n * Relative patterns starting with `'../'` are not prepended with `./`, even\n * if this option is set.\n */\n dotRelative?: boolean\n\n /**\n * Follow symlinked directories when expanding `**`\n * patterns. This can result in a lot of duplicate references in\n * the presence of cyclic links, and make performance quite bad.\n *\n * By default, a `**` in a pattern will follow 1 symbolic link if\n * it is not the first item in the pattern, or none if it is the\n * first item in the pattern, following the same behavior as Bash.\n */\n follow?: boolean\n\n /**\n * string or string[], or an object with `ignore` and `ignoreChildren`\n * methods.\n *\n * If a string or string[] is provided, then this is treated as a glob\n * pattern or array of glob patterns to exclude from matches. To ignore all\n * children within a directory, as well as the entry itself, append `'/**'`\n * to the ignore pattern.\n *\n * **Note** `ignore` patterns are _always_ in `dot:true` mode, regardless of\n * any other settings.\n *\n * If an object is provided that has `ignored(path)` and/or\n * `childrenIgnored(path)` methods, then these methods will be called to\n * determine whether any Path is a match or if its children should be\n * traversed, respectively.\n */\n ignore?: string | string[] | IgnoreLike\n\n /**\n * Treat brace expansion like `{a,b}` as a \"magic\" pattern. Has no\n * effect if {@link nobrace} is set.\n *\n * Only has effect on the {@link hasMagic} function.\n */\n magicalBraces?: boolean\n\n /**\n * Add a `/` character to directory matches. Note that this requires\n * additional stat calls in some cases.\n */\n mark?: boolean\n\n /**\n * Perform a basename-only match if the pattern does not contain any slash\n * characters. That is, `*.js` would be treated as equivalent to\n * `**\\/*.js`, matching all js files in all directories.\n */\n matchBase?: boolean\n\n /**\n * Limit the directory traversal to a given depth below the cwd.\n * Note that this does NOT prevent traversal to sibling folders,\n * root patterns, and so on. It only limits the maximum folder depth\n * that the walk will descend, relative to the cwd.\n */\n maxDepth?: number\n\n /**\n * Do not expand `{a,b}` and `{1..3}` brace sets.\n */\n nobrace?: boolean\n\n /**\n * Perform a case-insensitive match. This defaults to `true` on macOS and\n * Windows systems, and `false` on all others.\n *\n * **Note** `nocase` should only be explicitly set when it is\n * known that the filesystem's case sensitivity differs from the\n * platform default. If set `true` on case-sensitive file\n * systems, or `false` on case-insensitive file systems, then the\n * walk may return more or less results than expected.\n */\n nocase?: boolean\n\n /**\n * Do not match directories, only files. (Note: to match\n * _only_ directories, put a `/` at the end of the pattern.)\n */\n nodir?: boolean\n\n /**\n * Do not match \"extglob\" patterns such as `+(a|b)`.\n */\n noext?: boolean\n\n /**\n * Do not match `**` against multiple filenames. (Ie, treat it as a normal\n * `*` instead.)\n *\n * Conflicts with {@link matchBase}\n */\n noglobstar?: boolean\n\n /**\n * Defaults to value of `process.platform` if available, or `'linux'` if\n * not. Setting `platform:'win32'` on non-Windows systems may cause strange\n * behavior.\n */\n platform?: NodeJS.Platform\n\n /**\n * Set to true to call `fs.realpath` on all of the\n * results. In the case of an entry that cannot be resolved, the\n * entry is omitted. This incurs a slight performance penalty, of\n * course, because of the added system calls.\n */\n realpath?: boolean\n\n /**\n *\n * A string path resolved against the `cwd` option, which\n * is used as the starting point for absolute patterns that start\n * with `/`, (but not drive letters or UNC paths on Windows).\n *\n * Note that this _doesn't_ necessarily limit the walk to the\n * `root` directory, and doesn't affect the cwd starting point for\n * non-absolute patterns. A pattern containing `..` will still be\n * able to traverse out of the root directory, if it is not an\n * actual root directory on the filesystem, and any non-absolute\n * patterns will be matched in the `cwd`. For example, the\n * pattern `/../*` with `{root:'/some/path'}` will return all\n * files in `/some`, not all files in `/some/path`. The pattern\n * `*` with `{root:'/some/path'}` will return all the entries in\n * the cwd, not the entries in `/some/path`.\n *\n * To start absolute and non-absolute patterns in the same\n * path, you can use `{root:''}`. However, be aware that on\n * Windows systems, a pattern like `x:/*` or `//host/share/*` will\n * _always_ start in the `x:/` or `//host/share` directory,\n * regardless of the `root` setting.\n */\n root?: string\n\n /**\n * A [PathScurry](http://npm.im/path-scurry) object used\n * to traverse the file system. If the `nocase` option is set\n * explicitly, then any provided `scurry` object must match this\n * setting.\n */\n scurry?: PathScurry\n\n /**\n * Call `lstat()` on all entries, whether required or not to determine\n * if it's a valid match. When used with {@link withFileTypes}, this means\n * that matches will include data such as modified time, permissions, and\n * so on. Note that this will incur a performance cost due to the added\n * system calls.\n */\n stat?: boolean\n\n /**\n * An AbortSignal which will cancel the Glob walk when\n * triggered.\n */\n signal?: AbortSignal\n\n /**\n * Use `\\\\` as a path separator _only_, and\n * _never_ as an escape character. If set, all `\\\\` characters are\n * replaced with `/` in the pattern.\n *\n * Note that this makes it **impossible** to match against paths\n * containing literal glob pattern characters, but allows matching\n * with patterns constructed using `path.join()` and\n * `path.resolve()` on Windows platforms, mimicking the (buggy!)\n * behavior of Glob v7 and before on Windows. Please use with\n * caution, and be mindful of [the caveat below about Windows\n * paths](#windows). (For legacy reasons, this is also set if\n * `allowWindowsEscape` is set to the exact value `false`.)\n */\n windowsPathsNoEscape?: boolean\n\n /**\n * Return [PathScurry](http://npm.im/path-scurry)\n * `Path` objects instead of strings. These are similar to a\n * NodeJS `Dirent` object, but with additional methods and\n * properties.\n *\n * Conflicts with {@link absolute}\n */\n withFileTypes?: boolean\n\n /**\n * An fs implementation to override some or all of the defaults. See\n * http://npm.im/path-scurry for details about what can be overridden.\n */\n fs?: FSOption\n\n /**\n * Just passed along to Minimatch. Note that this makes all pattern\n * matching operations slower and *extremely* noisy.\n */\n debug?: boolean\n\n /**\n * Return `/` delimited paths, even on Windows.\n *\n * On posix systems, this has no effect. But, on Windows, it means that\n * paths will be `/` delimited, and absolute paths will be their full\n * resolved UNC forms, eg instead of `'C:\\\\foo\\\\bar'`, it would return\n * `'//?/C:/foo/bar'`\n */\n posix?: boolean\n\n /**\n * Do not match any children of any matches. For example, the pattern\n * `**\\/foo` would match `a/foo`, but not `a/foo/b/foo` in this mode.\n *\n * This is especially useful for cases like \"find all `node_modules`\n * folders, but not the ones in `node_modules`\".\n *\n * In order to support this, the `Ignore` implementation must support an\n * `add(pattern: string)` method. If using the default `Ignore` class, then\n * this is fine, but if this is set to `false`, and a custom `Ignore` is\n * provided that does not have an `add()` method, then it will throw an\n * error.\n *\n * **Caveat** It *only* ignores matches that would be a descendant of a\n * previous match, and only if that descendant is matched *after* the\n * ancestor is encountered. Since the file system walk happens in\n * indeterminate order, it's possible that a match will already be added\n * before its ancestor, if multiple or braced patterns are used.\n *\n * For example:\n *\n * ```ts\n * const results = await glob([\n * // likely to match first, since it's just a stat\n * 'a/b/c/d/e/f',\n *\n * // this pattern is more complicated! It must to various readdir()\n * // calls and test the results against a regular expression, and that\n * // is certainly going to take a little bit longer.\n * //\n * // So, later on, it encounters a match at 'a/b/c/d/e', but it's too\n * // late to ignore a/b/c/d/e/f, because it's already been emitted.\n * 'a/[bdf]/?/[a-z]/*',\n * ], { includeChildMatches: false })\n * ```\n *\n * It's best to only set this to `false` if you can be reasonably sure that\n * no components of the pattern will potentially match one another's file\n * system descendants, or if the occasional included child entry will not\n * cause problems.\n *\n * @default true\n */\n includeChildMatches?: boolean\n}\n\nexport type GlobOptionsWithFileTypesTrue = GlobOptions & {\n withFileTypes: true\n // string options not relevant if returning Path objects.\n absolute?: undefined\n mark?: undefined\n posix?: undefined\n}\n\nexport type GlobOptionsWithFileTypesFalse = GlobOptions & {\n withFileTypes?: false\n}\n\nexport type GlobOptionsWithFileTypesUnset = GlobOptions & {\n withFileTypes?: undefined\n}\n\nexport type Result<Opts> =\n Opts extends GlobOptionsWithFileTypesTrue ? Path\n : Opts extends GlobOptionsWithFileTypesFalse ? string\n : Opts extends GlobOptionsWithFileTypesUnset ? string\n : string | Path\nexport type Results<Opts> = Result<Opts>[]\n\nexport type FileTypes<Opts> =\n Opts extends GlobOptionsWithFileTypesTrue ? true\n : Opts extends GlobOptionsWithFileTypesFalse ? false\n : Opts extends GlobOptionsWithFileTypesUnset ? false\n : boolean\n\n/**\n * An object that can perform glob pattern traversals.\n */\nexport class Glob<Opts extends GlobOptions> implements GlobOptions {\n absolute?: boolean\n cwd: string\n root?: string\n dot: boolean\n dotRelative: boolean\n follow: boolean\n ignore?: string | string[] | IgnoreLike\n magicalBraces: boolean\n mark?: boolean\n matchBase: boolean\n maxDepth: number\n nobrace: boolean\n nocase: boolean\n nodir: boolean\n noext: boolean\n noglobstar: boolean\n pattern: string[]\n platform: NodeJS.Platform\n realpath: boolean\n scurry: PathScurry\n stat: boolean\n signal?: AbortSignal\n windowsPathsNoEscape: boolean\n withFileTypes: FileTypes<Opts>\n includeChildMatches: boolean\n\n /**\n * The options provided to the constructor.\n */\n opts: Opts\n\n /**\n * An array of parsed immutable {@link Pattern} objects.\n */\n patterns: Pattern[]\n\n /**\n * All options are stored as properties on the `Glob` object.\n *\n * See {@link GlobOptions} for full options descriptions.\n *\n * Note that a previous `Glob` object can be passed as the\n * `GlobOptions` to another `Glob` instantiation to re-use settings\n * and caches with a new pattern.\n *\n * Traversal functions can be called multiple times to run the walk\n * again.\n */\n constructor(pattern: string | string[], opts: Opts) {\n /* c8 ignore start */\n if (!opts) throw new TypeError('glob options required')\n /* c8 ignore stop */\n this.withFileTypes = !!opts.withFileTypes as FileTypes<Opts>\n this.signal = opts.signal\n this.follow = !!opts.follow\n this.dot = !!opts.dot\n this.dotRelative = !!opts.dotRelative\n this.nodir = !!opts.nodir\n this.mark = !!opts.mark\n if (!opts.cwd) {\n this.cwd = ''\n } else if (opts.cwd instanceof URL || opts.cwd.startsWith('file://')) {\n opts.cwd = fileURLToPath(opts.cwd)\n }\n this.cwd = opts.cwd || ''\n this.root = opts.root\n this.magicalBraces = !!opts.magicalBraces\n this.nobrace = !!opts.nobrace\n this.noext = !!opts.noext\n this.realpath = !!opts.realpath\n this.absolute = opts.absolute\n this.includeChildMatches = opts.includeChildMatches !== false\n\n this.noglobstar = !!opts.noglobstar\n this.matchBase = !!opts.matchBase\n this.maxDepth =\n typeof opts.maxDepth === 'number' ? opts.maxDepth : Infinity\n this.stat = !!opts.stat\n this.ignore = opts.ignore\n\n if (this.withFileTypes && this.absolute !== undefined) {\n throw new Error('cannot set absolute and withFileTypes:true')\n }\n\n if (typeof pattern === 'string') {\n pattern = [pattern]\n }\n\n this.windowsPathsNoEscape =\n !!opts.windowsPathsNoEscape ||\n (opts as { allowWindowsEscape?: boolean }).allowWindowsEscape ===\n false\n\n if (this.windowsPathsNoEscape) {\n pattern = pattern.map(p => p.replace(/\\\\/g, '/'))\n }\n\n if (this.matchBase) {\n if (opts.noglobstar) {\n throw new TypeError('base matching requires globstar')\n }\n pattern = pattern.map(p => (p.includes('/') ? p : `./**/${p}`))\n }\n\n this.pattern = pattern\n\n this.platform = opts.platform || defaultPlatform\n this.opts = { ...opts, platform: this.platform }\n if (opts.scurry) {\n this.scurry = opts.scurry\n if (\n opts.nocase !== undefined &&\n opts.nocase !== opts.scurry.nocase\n ) {\n throw new Error('nocase option contradicts provided scurry option')\n }\n } else {\n const Scurry =\n opts.platform === 'win32' ? PathScurryWin32\n : opts.platform === 'darwin' ? PathScurryDarwin\n : opts.platform ? PathScurryPosix\n : PathScurry\n this.scurry = new Scurry(this.cwd, {\n nocase: opts.nocase,\n fs: opts.fs,\n })\n }\n this.nocase = this.scurry.nocase\n\n // If you do nocase:true on a case-sensitive file system, then\n // we need to use regexps instead of strings for non-magic\n // path portions, because statting `aBc` won't return results\n // for the file `AbC` for example.\n const nocaseMagicOnly =\n this.platform === 'darwin' || this.platform === 'win32'\n\n const mmo: MinimatchOptions = {\n // default nocase based on platform\n ...opts,\n dot: this.dot,\n matchBase: this.matchBase,\n nobrace: this.nobrace,\n nocase: this.nocase,\n nocaseMagicOnly,\n nocomment: true,\n noext: this.noext,\n nonegate: true,\n optimizationLevel: 2,\n platform: this.platform,\n windowsPathsNoEscape: this.windowsPathsNoEscape,\n debug: !!this.opts.debug,\n }\n\n const mms = this.pattern.map(p => new Minimatch(p, mmo))\n const [matchSet, globParts] = mms.reduce(\n (set: [MatchSet, GlobParts], m) => {\n set[0].push(...m.set)\n set[1].push(...m.globParts)\n return set\n },\n [[], []],\n )\n this.patterns = matchSet.map((set, i) => {\n const g = globParts[i]\n /* c8 ignore start */\n if (!g) throw new Error('invalid pattern object')\n /* c8 ignore stop */\n return new Pattern(set, g, 0, this.platform)\n })\n }\n\n /**\n * Returns a Promise that resolves to the results array.\n */\n async walk(): Promise<Results<Opts>>\n async walk(): Promise<(string | Path)[]> {\n // Walkers always return array of Path objects, so we just have to\n // coerce them into the right shape. It will have already called\n // realpath() if the option was set to do so, so we know that's cached.\n // start out knowing the cwd, at least\n return [\n ...(await new GlobWalker(this.patterns, this.scurry.cwd, {\n ...this.opts,\n maxDepth:\n this.maxDepth !== Infinity ?\n this.maxDepth + this.scurry.cwd.depth()\n : Infinity,\n platform: this.platform,\n nocase: this.nocase,\n includeChildMatches: this.includeChildMatches,\n }).walk()),\n ]\n }\n\n /**\n * synchronous {@link Glob.walk}\n */\n walkSync(): Results<Opts>\n walkSync(): (string | Path)[] {\n return [\n ...new GlobWalker(this.patterns, this.scurry.cwd, {\n ...this.opts,\n maxDepth:\n this.maxDepth !== Infinity ?\n this.maxDepth + this.scurry.cwd.depth()\n : Infinity,\n platform: this.platform,\n nocase: this.nocase,\n includeChildMatches: this.includeChildMatches,\n }).walkSync(),\n ]\n }\n\n /**\n * Stream results asynchronously.\n */\n stream(): Minipass<Result<Opts>, Result<Opts>>\n stream(): Minipass<string | Path, string | Path> {\n return new GlobStream(this.patterns, this.scurry.cwd, {\n ...this.opts,\n maxDepth:\n this.maxDepth !== Infinity ?\n this.maxDepth + this.scurry.cwd.depth()\n : Infinity,\n platform: this.platform,\n nocase: this.nocase,\n includeChildMatches: this.includeChildMatches,\n }).stream()\n }\n\n /**\n * Stream results synchronously.\n */\n streamSync(): Minipass<Result<Opts>, Result<Opts>>\n streamSync(): Minipass<string | Path, string | Path> {\n return new GlobStream(this.patterns, this.scurry.cwd, {\n ...this.opts,\n maxDepth:\n this.maxDepth !== Infinity ?\n this.maxDepth + this.scurry.cwd.depth()\n : Infinity,\n platform: this.platform,\n nocase: this.nocase,\n includeChildMatches: this.includeChildMatches,\n }).streamSync()\n }\n\n /**\n * Default sync iteration function. Returns a Generator that\n * iterates over the results.\n */\n iterateSync(): Generator<Result<Opts>, void, void> {\n return this.streamSync()[Symbol.iterator]()\n }\n [Symbol.iterator]() {\n return this.iterateSync()\n }\n\n /**\n * Default async iteration function. Returns an AsyncGenerator that\n * iterates over the results.\n */\n iterate(): AsyncGenerator<Result<Opts>, void, void> {\n return this.stream()[Symbol.asyncIterator]()\n }\n [Symbol.asyncIterator]() {\n return this.iterate()\n }\n}\n","/**\n * @module LRUCache\n */\n\n// module-private names and types\ntype Perf = { now: () => number }\nconst perf: Perf =\n typeof performance === 'object' &&\n performance &&\n typeof performance.now === 'function'\n ? performance\n : Date\n\nconst warned = new Set<string>()\n\n// either a function or a class\ntype ForC = ((...a: any[]) => any) | { new (...a: any[]): any }\n\n/* c8 ignore start */\nconst PROCESS = (\n typeof process === 'object' && !!process ? process : {}\n) as { [k: string]: any }\n/* c8 ignore start */\n\nconst emitWarning = (\n msg: string,\n type: string,\n code: string,\n fn: ForC\n) => {\n typeof PROCESS.emitWarning === 'function'\n ? PROCESS.emitWarning(msg, type, code, fn)\n : console.error(`[${code}] ${type}: ${msg}`)\n}\n\nlet AC = globalThis.AbortController\nlet AS = globalThis.AbortSignal\n\n/* c8 ignore start */\nif (typeof AC === 'undefined') {\n //@ts-ignore\n AS = class AbortSignal {\n onabort?: (...a: any[]) => any\n _onabort: ((...a: any[]) => any)[] = []\n reason?: any\n aborted: boolean = false\n addEventListener(_: string, fn: (...a: any[]) => any) {\n this._onabort.push(fn)\n }\n }\n //@ts-ignore\n AC = class AbortController {\n constructor() {\n warnACPolyfill()\n }\n signal = new AS()\n abort(reason: any) {\n if (this.signal.aborted) return\n //@ts-ignore\n this.signal.reason = reason\n //@ts-ignore\n this.signal.aborted = true\n //@ts-ignore\n for (const fn of this.signal._onabort) {\n fn(reason)\n }\n this.signal.onabort?.(reason)\n }\n }\n let printACPolyfillWarning =\n PROCESS.env?.LRU_CACHE_IGNORE_AC_WARNING !== '1'\n const warnACPolyfill = () => {\n if (!printACPolyfillWarning) return\n printACPolyfillWarning = false\n emitWarning(\n 'AbortController is not defined. If using lru-cache in ' +\n 'node 14, load an AbortController polyfill from the ' +\n '`node-abort-controller` package. A minimal polyfill is ' +\n 'provided for use by LRUCache.fetch(), but it should not be ' +\n 'relied upon in other contexts (eg, passing it to other APIs that ' +\n 'use AbortController/AbortSignal might have undesirable effects). ' +\n 'You may disable this with LRU_CACHE_IGNORE_AC_WARNING=1 in the env.',\n 'NO_ABORT_CONTROLLER',\n 'ENOTSUP',\n warnACPolyfill\n )\n }\n}\n/* c8 ignore stop */\n\nconst shouldWarn = (code: string) => !warned.has(code)\n\nconst TYPE = Symbol('type')\nexport type PosInt = number & { [TYPE]: 'Positive Integer' }\nexport type Index = number & { [TYPE]: 'LRUCache Index' }\n\nconst isPosInt = (n: any): n is PosInt =>\n n && n === Math.floor(n) && n > 0 && isFinite(n)\n\nexport type UintArray = Uint8Array | Uint16Array | Uint32Array\nexport type NumberArray = UintArray | number[]\n\n/* c8 ignore start */\n// This is a little bit ridiculous, tbh.\n// The maximum array length is 2^32-1 or thereabouts on most JS impls.\n// And well before that point, you're caching the entire world, I mean,\n// that's ~32GB of just integers for the next/prev links, plus whatever\n// else to hold that many keys and values. Just filling the memory with\n// zeroes at init time is brutal when you get that big.\n// But why not be complete?\n// Maybe in the future, these limits will have expanded.\nconst getUintArray = (max: number) =>\n !isPosInt(max)\n ? null\n : max <= Math.pow(2, 8)\n ? Uint8Array\n : max <= Math.pow(2, 16)\n ? Uint16Array\n : max <= Math.pow(2, 32)\n ? Uint32Array\n : max <= Number.MAX_SAFE_INTEGER\n ? ZeroArray\n : null\n/* c8 ignore stop */\n\nclass ZeroArray extends Array<number> {\n constructor(size: number) {\n super(size)\n this.fill(0)\n }\n}\nexport type { ZeroArray }\nexport type { Stack }\n\nexport type StackLike = Stack | Index[]\nclass Stack {\n heap: NumberArray\n length: number\n // private constructor\n static #constructing: boolean = false\n static create(max: number): StackLike {\n const HeapCls = getUintArray(max)\n if (!HeapCls) return []\n Stack.#constructing = true\n const s = new Stack(max, HeapCls)\n Stack.#constructing = false\n return s\n }\n constructor(\n max: number,\n HeapCls: { new (n: number): NumberArray }\n ) {\n /* c8 ignore start */\n if (!Stack.#constructing) {\n throw new TypeError('instantiate Stack using Stack.create(n)')\n }\n /* c8 ignore stop */\n this.heap = new HeapCls(max)\n this.length = 0\n }\n push(n: Index) {\n this.heap[this.length++] = n\n }\n pop(): Index {\n return this.heap[--this.length] as Index\n }\n}\n\n/**\n * Promise representing an in-progress {@link LRUCache#fetch} call\n */\nexport type BackgroundFetch<V> = Promise<V | undefined> & {\n __returned: BackgroundFetch<V> | undefined\n __abortController: AbortController\n __staleWhileFetching: V | undefined\n}\n\nexport type DisposeTask<K, V> = [\n value: V,\n key: K,\n reason: LRUCache.DisposeReason\n]\n\nexport namespace LRUCache {\n /**\n * An integer greater than 0, reflecting the calculated size of items\n */\n export type Size = number\n\n /**\n * Integer greater than 0, representing some number of milliseconds, or the\n * time at which a TTL started counting from.\n */\n export type Milliseconds = number\n\n /**\n * An integer greater than 0, reflecting a number of items\n */\n export type Count = number\n\n /**\n * The reason why an item was removed from the cache, passed\n * to the {@link Disposer} methods.\n *\n * - `evict`: The item was evicted because it is the least recently used,\n * and the cache is full.\n * - `set`: A new value was set, overwriting the old value being disposed.\n * - `delete`: The item was explicitly deleted, either by calling\n * {@link LRUCache#delete}, {@link LRUCache#clear}, or\n * {@link LRUCache#set} with an undefined value.\n * - `expire`: The item was removed due to exceeding its TTL.\n * - `fetch`: A {@link OptionsBase#fetchMethod} operation returned\n * `undefined` or was aborted, causing the item to be deleted.\n */\n export type DisposeReason =\n | 'evict'\n | 'set'\n | 'delete'\n | 'expire'\n | 'fetch'\n /**\n * A method called upon item removal, passed as the\n * {@link OptionsBase.dispose} and/or\n * {@link OptionsBase.disposeAfter} options.\n */\n export type Disposer<K, V> = (\n value: V,\n key: K,\n reason: DisposeReason\n ) => void\n\n /**\n * A function that returns the effective calculated size\n * of an entry in the cache.\n */\n export type SizeCalculator<K, V> = (value: V, key: K) => Size\n\n /**\n * Options provided to the\n * {@link OptionsBase.fetchMethod} function.\n */\n export interface FetcherOptions<K, V, FC = unknown> {\n signal: AbortSignal\n options: FetcherFetchOptions<K, V, FC>\n /**\n * Object provided in the {@link FetchOptions.context} option to\n * {@link LRUCache#fetch}\n */\n context: FC\n }\n\n /**\n * Occasionally, it may be useful to track the internal behavior of the\n * cache, particularly for logging, debugging, or for behavior within the\n * `fetchMethod`. To do this, you can pass a `status` object to the\n * {@link LRUCache#fetch}, {@link LRUCache#get}, {@link LRUCache#set},\n * {@link LRUCache#memo}, and {@link LRUCache#has} methods.\n *\n * The `status` option should be a plain JavaScript object. The following\n * fields will be set on it appropriately, depending on the situation.\n */\n export interface Status<V> {\n /**\n * The status of a set() operation.\n *\n * - add: the item was not found in the cache, and was added\n * - update: the item was in the cache, with the same value provided\n * - replace: the item was in the cache, and replaced\n * - miss: the item was not added to the cache for some reason\n */\n set?: 'add' | 'update' | 'replace' | 'miss'\n\n /**\n * the ttl stored for the item, or undefined if ttls are not used.\n */\n ttl?: Milliseconds\n\n /**\n * the start time for the item, or undefined if ttls are not used.\n */\n start?: Milliseconds\n\n /**\n * The timestamp used for TTL calculation\n */\n now?: Milliseconds\n\n /**\n * the remaining ttl for the item, or undefined if ttls are not used.\n */\n remainingTTL?: Milliseconds\n\n /**\n * The calculated size for the item, if sizes are used.\n */\n entrySize?: Size\n\n /**\n * The total calculated size of the cache, if sizes are used.\n */\n totalCalculatedSize?: Size\n\n /**\n * A flag indicating that the item was not stored, due to exceeding the\n * {@link OptionsBase.maxEntrySize}\n */\n maxEntrySizeExceeded?: true\n\n /**\n * The old value, specified in the case of `set:'update'` or\n * `set:'replace'`\n */\n oldValue?: V\n\n /**\n * The results of a {@link LRUCache#has} operation\n *\n * - hit: the item was found in the cache\n * - stale: the item was found in the cache, but is stale\n * - miss: the item was not found in the cache\n */\n has?: 'hit' | 'stale' | 'miss'\n\n /**\n * The status of a {@link LRUCache#fetch} operation.\n * Note that this can change as the underlying fetch() moves through\n * various states.\n *\n * - inflight: there is another fetch() for this key which is in process\n * - get: there is no {@link OptionsBase.fetchMethod}, so\n * {@link LRUCache#get} was called.\n * - miss: the item is not in cache, and will be fetched.\n * - hit: the item is in the cache, and was resolved immediately.\n * - stale: the item is in the cache, but stale.\n * - refresh: the item is in the cache, and not stale, but\n * {@link FetchOptions.forceRefresh} was specified.\n */\n fetch?: 'get' | 'inflight' | 'miss' | 'hit' | 'stale' | 'refresh'\n\n /**\n * The {@link OptionsBase.fetchMethod} was called\n */\n fetchDispatched?: true\n\n /**\n * The cached value was updated after a successful call to\n * {@link OptionsBase.fetchMethod}\n */\n fetchUpdated?: true\n\n /**\n * The reason for a fetch() rejection. Either the error raised by the\n * {@link OptionsBase.fetchMethod}, or the reason for an\n * AbortSignal.\n */\n fetchError?: Error\n\n /**\n * The fetch received an abort signal\n */\n fetchAborted?: true\n\n /**\n * The abort signal received was ignored, and the fetch was allowed to\n * continue.\n */\n fetchAbortIgnored?: true\n\n /**\n * The fetchMethod promise resolved successfully\n */\n fetchResolved?: true\n\n /**\n * The fetchMethod promise was rejected\n */\n fetchRejected?: true\n\n /**\n * The status of a {@link LRUCache#get} operation.\n *\n * - fetching: The item is currently being fetched. If a previous value\n * is present and allowed, that will be returned.\n * - stale: The item is in the cache, and is stale.\n * - hit: the item is in the cache\n * - miss: the item is not in the cache\n */\n get?: 'stale' | 'hit' | 'miss'\n\n /**\n * A fetch or get operation returned a stale value.\n */\n returnedStale?: true\n }\n\n /**\n * options which override the options set in the LRUCache constructor\n * when calling {@link LRUCache#fetch}.\n *\n * This is the union of {@link GetOptions} and {@link SetOptions}, plus\n * {@link OptionsBase.noDeleteOnFetchRejection},\n * {@link OptionsBase.allowStaleOnFetchRejection},\n * {@link FetchOptions.forceRefresh}, and\n * {@link FetcherOptions.context}\n *\n * Any of these may be modified in the {@link OptionsBase.fetchMethod}\n * function, but the {@link GetOptions} fields will of course have no\n * effect, as the {@link LRUCache#get} call already happened by the time\n * the fetchMethod is called.\n */\n export interface FetcherFetchOptions<K, V, FC = unknown>\n extends Pick<\n OptionsBase<K, V, FC>,\n | 'allowStale'\n | 'updateAgeOnGet'\n | 'noDeleteOnStaleGet'\n | 'sizeCalculation'\n | 'ttl'\n | 'noDisposeOnSet'\n | 'noUpdateTTL'\n | 'noDeleteOnFetchRejection'\n | 'allowStaleOnFetchRejection'\n | 'ignoreFetchAbort'\n | 'allowStaleOnFetchAbort'\n > {\n status?: Status<V>\n size?: Size\n }\n\n /**\n * Options that may be passed to the {@link LRUCache#fetch} method.\n */\n export interface FetchOptions<K, V, FC>\n extends FetcherFetchOptions<K, V, FC> {\n /**\n * Set to true to force a re-load of the existing data, even if it\n * is not yet stale.\n */\n forceRefresh?: boolean\n /**\n * Context provided to the {@link OptionsBase.fetchMethod} as\n * the {@link FetcherOptions.context} param.\n *\n * If the FC type is specified as unknown (the default),\n * undefined or void, then this is optional. Otherwise, it will\n * be required.\n */\n context?: FC\n signal?: AbortSignal\n status?: Status<V>\n }\n /**\n * Options provided to {@link LRUCache#fetch} when the FC type is something\n * other than `unknown`, `undefined`, or `void`\n */\n export interface FetchOptionsWithContext<K, V, FC>\n extends FetchOptions<K, V, FC> {\n context: FC\n }\n /**\n * Options provided to {@link LRUCache#fetch} when the FC type is\n * `undefined` or `void`\n */\n export interface FetchOptionsNoContext<K, V>\n extends FetchOptions<K, V, undefined> {\n context?: undefined\n }\n\n export interface MemoOptions<K, V, FC = unknown>\n extends Pick<\n OptionsBase<K, V, FC>,\n | 'allowStale'\n | 'updateAgeOnGet'\n | 'noDeleteOnStaleGet'\n | 'sizeCalculation'\n | 'ttl'\n | 'noDisposeOnSet'\n | 'noUpdateTTL'\n | 'noDeleteOnFetchRejection'\n | 'allowStaleOnFetchRejection'\n | 'ignoreFetchAbort'\n | 'allowStaleOnFetchAbort'\n > {\n /**\n * Set to true to force a re-load of the existing data, even if it\n * is not yet stale.\n */\n forceRefresh?: boolean\n /**\n * Context provided to the {@link OptionsBase.memoMethod} as\n * the {@link MemoizerOptions.context} param.\n *\n * If the FC type is specified as unknown (the default),\n * undefined or void, then this is optional. Otherwise, it will\n * be required.\n */\n context?: FC\n status?: Status<V>\n }\n /**\n * Options provided to {@link LRUCache#memo} when the FC type is something\n * other than `unknown`, `undefined`, or `void`\n */\n export interface MemoOptionsWithContext<K, V, FC>\n extends MemoOptions<K, V, FC> {\n context: FC\n }\n /**\n * Options provided to {@link LRUCache#memo} when the FC type is\n * `undefined` or `void`\n */\n export interface MemoOptionsNoContext<K, V>\n extends MemoOptions<K, V, undefined> {\n context?: undefined\n }\n\n /**\n * Options provided to the\n * {@link OptionsBase.memoMethod} function.\n */\n export interface MemoizerOptions<K, V, FC = unknown> {\n options: MemoizerMemoOptions<K, V, FC>\n /**\n * Object provided in the {@link MemoOptions.context} option to\n * {@link LRUCache#memo}\n */\n context: FC\n }\n\n /**\n * options which override the options set in the LRUCache constructor\n * when calling {@link LRUCache#memo}.\n *\n * This is the union of {@link GetOptions} and {@link SetOptions}, plus\n * {@link MemoOptions.forceRefresh}, and\n * {@link MemoerOptions.context}\n *\n * Any of these may be modified in the {@link OptionsBase.memoMethod}\n * function, but the {@link GetOptions} fields will of course have no\n * effect, as the {@link LRUCache#get} call already happened by the time\n * the memoMethod is called.\n */\n export interface MemoizerMemoOptions<K, V, FC = unknown>\n extends Pick<\n OptionsBase<K, V, FC>,\n | 'allowStale'\n | 'updateAgeOnGet'\n | 'noDeleteOnStaleGet'\n | 'sizeCalculation'\n | 'ttl'\n | 'noDisposeOnSet'\n | 'noUpdateTTL'\n > {\n status?: Status<V>\n size?: Size\n start?: Milliseconds\n }\n\n /**\n * Options that may be passed to the {@link LRUCache#has} method.\n */\n export interface HasOptions<K, V, FC>\n extends Pick<OptionsBase<K, V, FC>, 'updateAgeOnHas'> {\n status?: Status<V>\n }\n\n /**\n * Options that may be passed to the {@link LRUCache#get} method.\n */\n export interface GetOptions<K, V, FC>\n extends Pick<\n OptionsBase<K, V, FC>,\n 'allowStale' | 'updateAgeOnGet' | 'noDeleteOnStaleGet'\n > {\n status?: Status<V>\n }\n\n /**\n * Options that may be passed to the {@link LRUCache#peek} method.\n */\n export interface PeekOptions<K, V, FC>\n extends Pick<OptionsBase<K, V, FC>, 'allowStale'> {}\n\n /**\n * Options that may be passed to the {@link LRUCache#set} method.\n */\n export interface SetOptions<K, V, FC>\n extends Pick<\n OptionsBase<K, V, FC>,\n 'sizeCalculation' | 'ttl' | 'noDisposeOnSet' | 'noUpdateTTL'\n > {\n /**\n * If size tracking is enabled, then setting an explicit size\n * in the {@link LRUCache#set} call will prevent calling the\n * {@link OptionsBase.sizeCalculation} function.\n */\n size?: Size\n /**\n * If TTL tracking is enabled, then setting an explicit start\n * time in the {@link LRUCache#set} call will override the\n * default time from `performance.now()` or `Date.now()`.\n *\n * Note that it must be a valid value for whichever time-tracking\n * method is in use.\n */\n start?: Milliseconds\n status?: Status<V>\n }\n\n /**\n * The type signature for the {@link OptionsBase.fetchMethod} option.\n */\n export type Fetcher<K, V, FC = unknown> = (\n key: K,\n staleValue: V | undefined,\n options: FetcherOptions<K, V, FC>\n ) => Promise<V | undefined | void> | V | undefined | void\n\n /**\n * the type signature for the {@link OptionsBase.memoMethod} option.\n */\n export type Memoizer<K, V, FC = unknown> = (\n key: K,\n staleValue: V | undefined,\n options: MemoizerOptions<K, V, FC>\n ) => V\n\n /**\n * Options which may be passed to the {@link LRUCache} constructor.\n *\n * Most of these may be overridden in the various options that use\n * them.\n *\n * Despite all being technically optional, the constructor requires that\n * a cache is at minimum limited by one or more of {@link OptionsBase.max},\n * {@link OptionsBase.ttl}, or {@link OptionsBase.maxSize}.\n *\n * If {@link OptionsBase.ttl} is used alone, then it is strongly advised\n * (and in fact required by the type definitions here) that the cache\n * also set {@link OptionsBase.ttlAutopurge}, to prevent potentially\n * unbounded storage.\n *\n * All options are also available on the {@link LRUCache} instance, making\n * it safe to pass an LRUCache instance as the options argumemnt to\n * make another empty cache of the same type.\n *\n * Some options are marked as read-only, because changing them after\n * instantiation is not safe. Changing any of the other options will of\n * course only have an effect on subsequent method calls.\n */\n export interface OptionsBase<K, V, FC> {\n /**\n * The maximum number of items to store in the cache before evicting\n * old entries. This is read-only on the {@link LRUCache} instance,\n * and may not be overridden.\n *\n * If set, then storage space will be pre-allocated at construction\n * time, and the cache will perform significantly faster.\n *\n * Note that significantly fewer items may be stored, if\n * {@link OptionsBase.maxSize} and/or {@link OptionsBase.ttl} are also\n * set.\n *\n * **It is strongly recommended to set a `max` to prevent unbounded growth\n * of the cache.**\n */\n max?: Count\n\n /**\n * Max time in milliseconds for items to live in cache before they are\n * considered stale. Note that stale items are NOT preemptively removed by\n * default, and MAY live in the cache, contributing to its LRU max, long\n * after they have expired, unless {@link OptionsBase.ttlAutopurge} is\n * set.\n *\n * If set to `0` (the default value), then that means \"do not track\n * TTL\", not \"expire immediately\".\n *\n * Also, as this cache is optimized for LRU/MRU operations, some of\n * the staleness/TTL checks will reduce performance, as they will incur\n * overhead by deleting items.\n *\n * This is not primarily a TTL cache, and does not make strong TTL\n * guarantees. There is no pre-emptive pruning of expired items, but you\n * _may_ set a TTL on the cache, and it will treat expired items as missing\n * when they are fetched, and delete them.\n *\n * Optional, but must be a non-negative integer in ms if specified.\n *\n * This may be overridden by passing an options object to `cache.set()`.\n *\n * At least one of `max`, `maxSize`, or `TTL` is required. This must be a\n * positive integer if set.\n *\n * Even if ttl tracking is enabled, **it is strongly recommended to set a\n * `max` to prevent unbounded growth of the cache.**\n *\n * If ttl tracking is enabled, and `max` and `maxSize` are not set,\n * and `ttlAutopurge` is not set, then a warning will be emitted\n * cautioning about the potential for unbounded memory consumption.\n * (The TypeScript definitions will also discourage this.)\n */\n ttl?: Milliseconds\n\n /**\n * Minimum amount of time in ms in which to check for staleness.\n * Defaults to 1, which means that the current time is checked\n * at most once per millisecond.\n *\n * Set to 0 to check the current time every time staleness is tested.\n * (This reduces performance, and is theoretically unnecessary.)\n *\n * Setting this to a higher value will improve performance somewhat\n * while using ttl tracking, albeit at the expense of keeping stale\n * items around a bit longer than their TTLs would indicate.\n *\n * @default 1\n */\n ttlResolution?: Milliseconds\n\n /**\n * Preemptively remove stale items from the cache.\n *\n * Note that this may *significantly* degrade performance, especially if\n * the cache is storing a large number of items. It is almost always best\n * to just leave the stale items in the cache, and let them fall out as new\n * items are added.\n *\n * Note that this means that {@link OptionsBase.allowStale} is a bit\n * pointless, as stale items will be deleted almost as soon as they\n * expire.\n *\n * Use with caution!\n */\n ttlAutopurge?: boolean\n\n /**\n * When using time-expiring entries with `ttl`, setting this to `true` will\n * make each item's age reset to 0 whenever it is retrieved from cache with\n * {@link LRUCache#get}, causing it to not expire. (It can still fall out\n * of cache based on recency of use, of course.)\n *\n * Has no effect if {@link OptionsBase.ttl} is not set.\n *\n * This may be overridden by passing an options object to `cache.get()`.\n */\n updateAgeOnGet?: boolean\n\n /**\n * When using time-expiring entries with `ttl`, setting this to `true` will\n * make each item's age reset to 0 whenever its presence in the cache is\n * checked with {@link LRUCache#has}, causing it to not expire. (It can\n * still fall out of cache based on recency of use, of course.)\n *\n * Has no effect if {@link OptionsBase.ttl} is not set.\n */\n updateAgeOnHas?: boolean\n\n /**\n * Allow {@link LRUCache#get} and {@link LRUCache#fetch} calls to return\n * stale data, if available.\n *\n * By default, if you set `ttl`, stale items will only be deleted from the\n * cache when you `get(key)`. That is, it's not preemptively pruning items,\n * unless {@link OptionsBase.ttlAutopurge} is set.\n *\n * If you set `allowStale:true`, it'll return the stale value *as well as*\n * deleting it. If you don't set this, then it'll return `undefined` when\n * you try to get a stale entry.\n *\n * Note that when a stale entry is fetched, _even if it is returned due to\n * `allowStale` being set_, it is removed from the cache immediately. You\n * can suppress this behavior by setting\n * {@link OptionsBase.noDeleteOnStaleGet}, either in the constructor, or in\n * the options provided to {@link LRUCache#get}.\n *\n * This may be overridden by passing an options object to `cache.get()`.\n * The `cache.has()` method will always return `false` for stale items.\n *\n * Only relevant if a ttl is set.\n */\n allowStale?: boolean\n\n /**\n * Function that is called on items when they are dropped from the\n * cache, as `dispose(value, key, reason)`.\n *\n * This can be handy if you want to close file descriptors or do\n * other cleanup tasks when items are no longer stored in the cache.\n *\n * **NOTE**: It is called _before_ the item has been fully removed\n * from the cache, so if you want to put it right back in, you need\n * to wait until the next tick. If you try to add it back in during\n * the `dispose()` function call, it will break things in subtle and\n * weird ways.\n *\n * Unlike several other options, this may _not_ be overridden by\n * passing an option to `set()`, for performance reasons.\n *\n * The `reason` will be one of the following strings, corresponding\n * to the reason for the item's deletion:\n *\n * - `evict` Item was evicted to make space for a new addition\n * - `set` Item was overwritten by a new value\n * - `expire` Item expired its TTL\n * - `fetch` Item was deleted due to a failed or aborted fetch, or a\n * fetchMethod returning `undefined.\n * - `delete` Item was removed by explicit `cache.delete(key)`,\n * `cache.clear()`, or `cache.set(key, undefined)`.\n */\n dispose?: Disposer<K, V>\n\n /**\n * The same as {@link OptionsBase.dispose}, but called *after* the entry\n * is completely removed and the cache is once again in a clean state.\n *\n * It is safe to add an item right back into the cache at this point.\n * However, note that it is *very* easy to inadvertently create infinite\n * recursion this way.\n */\n disposeAfter?: Disposer<K, V>\n\n /**\n * Set to true to suppress calling the\n * {@link OptionsBase.dispose} function if the entry key is\n * still accessible within the cache.\n *\n * This may be overridden by passing an options object to\n * {@link LRUCache#set}.\n *\n * Only relevant if `dispose` or `disposeAfter` are set.\n */\n noDisposeOnSet?: boolean\n\n /**\n * Boolean flag to tell the cache to not update the TTL when setting a new\n * value for an existing key (ie, when updating a value rather than\n * inserting a new value). Note that the TTL value is _always_ set (if\n * provided) when adding a new entry into the cache.\n *\n * Has no effect if a {@link OptionsBase.ttl} is not set.\n *\n * May be passed as an option to {@link LRUCache#set}.\n */\n noUpdateTTL?: boolean\n\n /**\n * Set to a positive integer to track the sizes of items added to the\n * cache, and automatically evict items in order to stay below this size.\n * Note that this may result in fewer than `max` items being stored.\n *\n * Attempting to add an item to the cache whose calculated size is greater\n * that this amount will be a no-op. The item will not be cached, and no\n * other items will be evicted.\n *\n * Optional, must be a positive integer if provided.\n *\n * Sets `maxEntrySize` to the same value, unless a different value is\n * provided for `maxEntrySize`.\n *\n * At least one of `max`, `maxSize`, or `TTL` is required. This must be a\n * positive integer if set.\n *\n * Even if size tracking is enabled, **it is strongly recommended to set a\n * `max` to prevent unbounded growth of the cache.**\n *\n * Note also that size tracking can negatively impact performance,\n * though for most cases, only minimally.\n */\n maxSize?: Size\n\n /**\n * The maximum allowed size for any single item in the cache.\n *\n * If a larger item is passed to {@link LRUCache#set} or returned by a\n * {@link OptionsBase.fetchMethod} or {@link OptionsBase.memoMethod}, then\n * it will not be stored in the cache.\n *\n * Attempting to add an item whose calculated size is greater than\n * this amount will not cache the item or evict any old items, but\n * WILL delete an existing value if one is already present.\n *\n * Optional, must be a positive integer if provided. Defaults to\n * the value of `maxSize` if provided.\n */\n maxEntrySize?: Size\n\n /**\n * A function that returns a number indicating the item's size.\n *\n * Requires {@link OptionsBase.maxSize} to be set.\n *\n * If not provided, and {@link OptionsBase.maxSize} or\n * {@link OptionsBase.maxEntrySize} are set, then all\n * {@link LRUCache#set} calls **must** provide an explicit\n * {@link SetOptions.size} or sizeCalculation param.\n */\n sizeCalculation?: SizeCalculator<K, V>\n\n /**\n * Method that provides the implementation for {@link LRUCache#fetch}\n *\n * ```ts\n * fetchMethod(key, staleValue, { signal, options, context })\n * ```\n *\n * If `fetchMethod` is not provided, then `cache.fetch(key)` is equivalent\n * to `Promise.resolve(cache.get(key))`.\n *\n * If at any time, `signal.aborted` is set to `true`, or if the\n * `signal.onabort` method is called, or if it emits an `'abort'` event\n * which you can listen to with `addEventListener`, then that means that\n * the fetch should be abandoned. This may be passed along to async\n * functions aware of AbortController/AbortSignal behavior.\n *\n * The `fetchMethod` should **only** return `undefined` or a Promise\n * resolving to `undefined` if the AbortController signaled an `abort`\n * event. In all other cases, it should return or resolve to a value\n * suitable for adding to the cache.\n *\n * The `options` object is a union of the options that may be provided to\n * `set()` and `get()`. If they are modified, then that will result in\n * modifying the settings to `cache.set()` when the value is resolved, and\n * in the case of\n * {@link OptionsBase.noDeleteOnFetchRejection} and\n * {@link OptionsBase.allowStaleOnFetchRejection}, the handling of\n * `fetchMethod` failures.\n *\n * For example, a DNS cache may update the TTL based on the value returned\n * from a remote DNS server by changing `options.ttl` in the `fetchMethod`.\n */\n fetchMethod?: Fetcher<K, V, FC>\n\n /**\n * Method that provides the implementation for {@link LRUCache#memo}\n */\n memoMethod?: Memoizer<K, V, FC>\n\n /**\n * Set to true to suppress the deletion of stale data when a\n * {@link OptionsBase.fetchMethod} returns a rejected promise.\n */\n noDeleteOnFetchRejection?: boolean\n\n /**\n * Do not delete stale items when they are retrieved with\n * {@link LRUCache#get}.\n *\n * Note that the `get` return value will still be `undefined`\n * unless {@link OptionsBase.allowStale} is true.\n *\n * When using time-expiring entries with `ttl`, by default stale\n * items will be removed from the cache when the key is accessed\n * with `cache.get()`.\n *\n * Setting this option will cause stale items to remain in the cache, until\n * they are explicitly deleted with `cache.delete(key)`, or retrieved with\n * `noDeleteOnStaleGet` set to `false`.\n *\n * This may be overridden by passing an options object to `cache.get()`.\n *\n * Only relevant if a ttl is used.\n */\n noDeleteOnStaleGet?: boolean\n\n /**\n * Set to true to allow returning stale data when a\n * {@link OptionsBase.fetchMethod} throws an error or returns a rejected\n * promise.\n *\n * This differs from using {@link OptionsBase.allowStale} in that stale\n * data will ONLY be returned in the case that the {@link LRUCache#fetch}\n * fails, not any other times.\n *\n * If a `fetchMethod` fails, and there is no stale value available, the\n * `fetch()` will resolve to `undefined`. Ie, all `fetchMethod` errors are\n * suppressed.\n *\n * Implies `noDeleteOnFetchRejection`.\n *\n * This may be set in calls to `fetch()`, or defaulted on the constructor,\n * or overridden by modifying the options object in the `fetchMethod`.\n */\n allowStaleOnFetchRejection?: boolean\n\n /**\n * Set to true to return a stale value from the cache when the\n * `AbortSignal` passed to the {@link OptionsBase.fetchMethod} dispatches\n * an `'abort'` event, whether user-triggered, or due to internal cache\n * behavior.\n *\n * Unless {@link OptionsBase.ignoreFetchAbort} is also set, the underlying\n * {@link OptionsBase.fetchMethod} will still be considered canceled, and\n * any value it returns will be ignored and not cached.\n *\n * Caveat: since fetches are aborted when a new value is explicitly\n * set in the cache, this can lead to fetch returning a stale value,\n * since that was the fallback value _at the moment the `fetch()` was\n * initiated_, even though the new updated value is now present in\n * the cache.\n *\n * For example:\n *\n * ```ts\n * const cache = new LRUCache<string, any>({\n * ttl: 100,\n * fetchMethod: async (url, oldValue, { signal }) => {\n * const res = await fetch(url, { signal })\n * return await res.json()\n * }\n * })\n * cache.set('https://example.com/', { some: 'data' })\n * // 100ms go by...\n * const result = cache.fetch('https://example.com/')\n * cache.set('https://example.com/', { other: 'thing' })\n * console.log(await result) // { some: 'data' }\n * console.log(cache.get('https://example.com/')) // { other: 'thing' }\n * ```\n */\n allowStaleOnFetchAbort?: boolean\n\n /**\n * Set to true to ignore the `abort` event emitted by the `AbortSignal`\n * object passed to {@link OptionsBase.fetchMethod}, and still cache the\n * resulting resolution value, as long as it is not `undefined`.\n *\n * When used on its own, this means aborted {@link LRUCache#fetch} calls\n * are not immediately resolved or rejected when they are aborted, and\n * instead take the full time to await.\n *\n * When used with {@link OptionsBase.allowStaleOnFetchAbort}, aborted\n * {@link LRUCache#fetch} calls will resolve immediately to their stale\n * cached value or `undefined`, and will continue to process and eventually\n * update the cache when they resolve, as long as the resulting value is\n * not `undefined`, thus supporting a \"return stale on timeout while\n * refreshing\" mechanism by passing `AbortSignal.timeout(n)` as the signal.\n *\n * For example:\n *\n * ```ts\n * const c = new LRUCache({\n * ttl: 100,\n * ignoreFetchAbort: true,\n * allowStaleOnFetchAbort: true,\n * fetchMethod: async (key, oldValue, { signal }) => {\n * // note: do NOT pass the signal to fetch()!\n * // let's say this fetch can take a long time.\n * const res = await fetch(`https://slow-backend-server/${key}`)\n * return await res.json()\n * },\n * })\n *\n * // this will return the stale value after 100ms, while still\n * // updating in the background for next time.\n * const val = await c.fetch('key', { signal: AbortSignal.timeout(100) })\n * ```\n *\n * **Note**: regardless of this setting, an `abort` event _is still\n * emitted on the `AbortSignal` object_, so may result in invalid results\n * when passed to other underlying APIs that use AbortSignals.\n *\n * This may be overridden in the {@link OptionsBase.fetchMethod} or the\n * call to {@link LRUCache#fetch}.\n */\n ignoreFetchAbort?: boolean\n }\n\n export interface OptionsMaxLimit<K, V, FC>\n extends OptionsBase<K, V, FC> {\n max: Count\n }\n export interface OptionsTTLLimit<K, V, FC>\n extends OptionsBase<K, V, FC> {\n ttl: Milliseconds\n ttlAutopurge: boolean\n }\n export interface OptionsSizeLimit<K, V, FC>\n extends OptionsBase<K, V, FC> {\n maxSize: Size\n }\n\n /**\n * The valid safe options for the {@link LRUCache} constructor\n */\n export type Options<K, V, FC> =\n | OptionsMaxLimit<K, V, FC>\n | OptionsSizeLimit<K, V, FC>\n | OptionsTTLLimit<K, V, FC>\n\n /**\n * Entry objects used by {@link LRUCache#load} and {@link LRUCache#dump},\n * and returned by {@link LRUCache#info}.\n */\n export interface Entry<V> {\n value: V\n ttl?: Milliseconds\n size?: Size\n start?: Milliseconds\n }\n}\n\n/**\n * Default export, the thing you're using this module to get.\n *\n * The `K` and `V` types define the key and value types, respectively. The\n * optional `FC` type defines the type of the `context` object passed to\n * `cache.fetch()` and `cache.memo()`.\n *\n * Keys and values **must not** be `null` or `undefined`.\n *\n * All properties from the options object (with the exception of `max`,\n * `maxSize`, `fetchMethod`, `memoMethod`, `dispose` and `disposeAfter`) are\n * added as normal public members. (The listed options are read-only getters.)\n *\n * Changing any of these will alter the defaults for subsequent method calls.\n */\nexport class LRUCache<K extends {}, V extends {}, FC = unknown>\n implements Map<K, V>\n{\n // options that cannot be changed without disaster\n readonly #max: LRUCache.Count\n readonly #maxSize: LRUCache.Size\n readonly #dispose?: LRUCache.Disposer<K, V>\n readonly #disposeAfter?: LRUCache.Disposer<K, V>\n readonly #fetchMethod?: LRUCache.Fetcher<K, V, FC>\n readonly #memoMethod?: LRUCache.Memoizer<K, V, FC>\n\n /**\n * {@link LRUCache.OptionsBase.ttl}\n */\n ttl: LRUCache.Milliseconds\n\n /**\n * {@link LRUCache.OptionsBase.ttlResolution}\n */\n ttlResolution: LRUCache.Milliseconds\n /**\n * {@link LRUCache.OptionsBase.ttlAutopurge}\n */\n ttlAutopurge: boolean\n /**\n * {@link LRUCache.OptionsBase.updateAgeOnGet}\n */\n updateAgeOnGet: boolean\n /**\n * {@link LRUCache.OptionsBase.updateAgeOnHas}\n */\n updateAgeOnHas: boolean\n /**\n * {@link LRUCache.OptionsBase.allowStale}\n */\n allowStale: boolean\n\n /**\n * {@link LRUCache.OptionsBase.noDisposeOnSet}\n */\n noDisposeOnSet: boolean\n /**\n * {@link LRUCache.OptionsBase.noUpdateTTL}\n */\n noUpdateTTL: boolean\n /**\n * {@link LRUCache.OptionsBase.maxEntrySize}\n */\n maxEntrySize: LRUCache.Size\n /**\n * {@link LRUCache.OptionsBase.sizeCalculation}\n */\n sizeCalculation?: LRUCache.SizeCalculator<K, V>\n /**\n * {@link LRUCache.OptionsBase.noDeleteOnFetchRejection}\n */\n noDeleteOnFetchRejection: boolean\n /**\n * {@link LRUCache.OptionsBase.noDeleteOnStaleGet}\n */\n noDeleteOnStaleGet: boolean\n /**\n * {@link LRUCache.OptionsBase.allowStaleOnFetchAbort}\n */\n allowStaleOnFetchAbort: boolean\n /**\n * {@link LRUCache.OptionsBase.allowStaleOnFetchRejection}\n */\n allowStaleOnFetchRejection: boolean\n /**\n * {@link LRUCache.OptionsBase.ignoreFetchAbort}\n */\n ignoreFetchAbort: boolean\n\n // computed properties\n #size: LRUCache.Count\n #calculatedSize: LRUCache.Size\n #keyMap: Map<K, Index>\n #keyList: (K | undefined)[]\n #valList: (V | BackgroundFetch<V> | undefined)[]\n #next: NumberArray\n #prev: NumberArray\n #head: Index\n #tail: Index\n #free: StackLike\n #disposed?: DisposeTask<K, V>[]\n #sizes?: ZeroArray\n #starts?: ZeroArray\n #ttls?: ZeroArray\n\n #hasDispose: boolean\n #hasFetchMethod: boolean\n #hasDisposeAfter: boolean\n\n /**\n * Do not call this method unless you need to inspect the\n * inner workings of the cache. If anything returned by this\n * object is modified in any way, strange breakage may occur.\n *\n * These fields are private for a reason!\n *\n * @internal\n */\n static unsafeExposeInternals<\n K extends {},\n V extends {},\n FC extends unknown = unknown\n >(c: LRUCache<K, V, FC>) {\n return {\n // properties\n starts: c.#starts,\n ttls: c.#ttls,\n sizes: c.#sizes,\n keyMap: c.#keyMap as Map<K, number>,\n keyList: c.#keyList,\n valList: c.#valList,\n next: c.#next,\n prev: c.#prev,\n get head() {\n return c.#head\n },\n get tail() {\n return c.#tail\n },\n free: c.#free,\n // methods\n isBackgroundFetch: (p: any) => c.#isBackgroundFetch(p),\n backgroundFetch: (\n k: K,\n index: number | undefined,\n options: LRUCache.FetchOptions<K, V, FC>,\n context: any\n ): BackgroundFetch<V> =>\n c.#backgroundFetch(\n k,\n index as Index | undefined,\n options,\n context\n ),\n moveToTail: (index: number): void =>\n c.#moveToTail(index as Index),\n indexes: (options?: { allowStale: boolean }) =>\n c.#indexes(options),\n rindexes: (options?: { allowStale: boolean }) =>\n c.#rindexes(options),\n isStale: (index: number | undefined) =>\n c.#isStale(index as Index),\n }\n }\n\n // Protected read-only members\n\n /**\n * {@link LRUCache.OptionsBase.max} (read-only)\n */\n get max(): LRUCache.Count {\n return this.#max\n }\n /**\n * {@link LRUCache.OptionsBase.maxSize} (read-only)\n */\n get maxSize(): LRUCache.Count {\n return this.#maxSize\n }\n /**\n * The total computed size of items in the cache (read-only)\n */\n get calculatedSize(): LRUCache.Size {\n return this.#calculatedSize\n }\n /**\n * The number of items stored in the cache (read-only)\n */\n get size(): LRUCache.Count {\n return this.#size\n }\n /**\n * {@link LRUCache.OptionsBase.fetchMethod} (read-only)\n */\n get fetchMethod(): LRUCache.Fetcher<K, V, FC> | undefined {\n return this.#fetchMethod\n }\n get memoMethod(): LRUCache.Memoizer<K, V, FC> | undefined {\n return this.#memoMethod\n }\n /**\n * {@link LRUCache.OptionsBase.dispose} (read-only)\n */\n get dispose() {\n return this.#dispose\n }\n /**\n * {@link LRUCache.OptionsBase.disposeAfter} (read-only)\n */\n get disposeAfter() {\n return this.#disposeAfter\n }\n\n constructor(\n options: LRUCache.Options<K, V, FC> | LRUCache<K, V, FC>\n ) {\n const {\n max = 0,\n ttl,\n ttlResolution = 1,\n ttlAutopurge,\n updateAgeOnGet,\n updateAgeOnHas,\n allowStale,\n dispose,\n disposeAfter,\n noDisposeOnSet,\n noUpdateTTL,\n maxSize = 0,\n maxEntrySize = 0,\n sizeCalculation,\n fetchMethod,\n memoMethod,\n noDeleteOnFetchRejection,\n noDeleteOnStaleGet,\n allowStaleOnFetchRejection,\n allowStaleOnFetchAbort,\n ignoreFetchAbort,\n } = options\n\n if (max !== 0 && !isPosInt(max)) {\n throw new TypeError('max option must be a nonnegative integer')\n }\n\n const UintArray = max ? getUintArray(max) : Array\n if (!UintArray) {\n throw new Error('invalid max value: ' + max)\n }\n\n this.#max = max\n this.#maxSize = maxSize\n this.maxEntrySize = maxEntrySize || this.#maxSize\n this.sizeCalculation = sizeCalculation\n if (this.sizeCalculation) {\n if (!this.#maxSize && !this.maxEntrySize) {\n throw new TypeError(\n 'cannot set sizeCalculation without setting maxSize or maxEntrySize'\n )\n }\n if (typeof this.sizeCalculation !== 'function') {\n throw new TypeError('sizeCalculation set to non-function')\n }\n }\n\n if (\n memoMethod !== undefined &&\n typeof memoMethod !== 'function'\n ) {\n throw new TypeError('memoMethod must be a function if defined')\n }\n this.#memoMethod = memoMethod\n\n if (\n fetchMethod !== undefined &&\n typeof fetchMethod !== 'function'\n ) {\n throw new TypeError(\n 'fetchMethod must be a function if specified'\n )\n }\n this.#fetchMethod = fetchMethod\n this.#hasFetchMethod = !!fetchMethod\n\n this.#keyMap = new Map()\n this.#keyList = new Array(max).fill(undefined)\n this.#valList = new Array(max).fill(undefined)\n this.#next = new UintArray(max)\n this.#prev = new UintArray(max)\n this.#head = 0 as Index\n this.#tail = 0 as Index\n this.#free = Stack.create(max)\n this.#size = 0\n this.#calculatedSize = 0\n\n if (typeof dispose === 'function') {\n this.#dispose = dispose\n }\n if (typeof disposeAfter === 'function') {\n this.#disposeAfter = disposeAfter\n this.#disposed = []\n } else {\n this.#disposeAfter = undefined\n this.#disposed = undefined\n }\n this.#hasDispose = !!this.#dispose\n this.#hasDisposeAfter = !!this.#disposeAfter\n\n this.noDisposeOnSet = !!noDisposeOnSet\n this.noUpdateTTL = !!noUpdateTTL\n this.noDeleteOnFetchRejection = !!noDeleteOnFetchRejection\n this.allowStaleOnFetchRejection = !!allowStaleOnFetchRejection\n this.allowStaleOnFetchAbort = !!allowStaleOnFetchAbort\n this.ignoreFetchAbort = !!ignoreFetchAbort\n\n // NB: maxEntrySize is set to maxSize if it's set\n if (this.maxEntrySize !== 0) {\n if (this.#maxSize !== 0) {\n if (!isPosInt(this.#maxSize)) {\n throw new TypeError(\n 'maxSize must be a positive integer if specified'\n )\n }\n }\n if (!isPosInt(this.maxEntrySize)) {\n throw new TypeError(\n 'maxEntrySize must be a positive integer if specified'\n )\n }\n this.#initializeSizeTracking()\n }\n\n this.allowStale = !!allowStale\n this.noDeleteOnStaleGet = !!noDeleteOnStaleGet\n this.updateAgeOnGet = !!updateAgeOnGet\n this.updateAgeOnHas = !!updateAgeOnHas\n this.ttlResolution =\n isPosInt(ttlResolution) || ttlResolution === 0\n ? ttlResolution\n : 1\n this.ttlAutopurge = !!ttlAutopurge\n this.ttl = ttl || 0\n if (this.ttl) {\n if (!isPosInt(this.ttl)) {\n throw new TypeError(\n 'ttl must be a positive integer if specified'\n )\n }\n this.#initializeTTLTracking()\n }\n\n // do not allow completely unbounded caches\n if (this.#max === 0 && this.ttl === 0 && this.#maxSize === 0) {\n throw new TypeError(\n 'At least one of max, maxSize, or ttl is required'\n )\n }\n if (!this.ttlAutopurge && !this.#max && !this.#maxSize) {\n const code = 'LRU_CACHE_UNBOUNDED'\n if (shouldWarn(code)) {\n warned.add(code)\n const msg =\n 'TTL caching without ttlAutopurge, max, or maxSize can ' +\n 'result in unbounded memory consumption.'\n emitWarning(msg, 'UnboundedCacheWarning', code, LRUCache)\n }\n }\n }\n\n /**\n * Return the number of ms left in the item's TTL. If item is not in cache,\n * returns `0`. Returns `Infinity` if item is in cache without a defined TTL.\n */\n getRemainingTTL(key: K) {\n return this.#keyMap.has(key) ? Infinity : 0\n }\n\n #initializeTTLTracking() {\n const ttls = new ZeroArray(this.#max)\n const starts = new ZeroArray(this.#max)\n this.#ttls = ttls\n this.#starts = starts\n\n this.#setItemTTL = (index, ttl, start = perf.now()) => {\n starts[index] = ttl !== 0 ? start : 0\n ttls[index] = ttl\n if (ttl !== 0 && this.ttlAutopurge) {\n const t = setTimeout(() => {\n if (this.#isStale(index)) {\n this.#delete(this.#keyList[index] as K, 'expire')\n }\n }, ttl + 1)\n // unref() not supported on all platforms\n /* c8 ignore start */\n if (t.unref) {\n t.unref()\n }\n /* c8 ignore stop */\n }\n }\n\n this.#updateItemAge = index => {\n starts[index] = ttls[index] !== 0 ? perf.now() : 0\n }\n\n this.#statusTTL = (status, index) => {\n if (ttls[index]) {\n const ttl = ttls[index]\n const start = starts[index]\n /* c8 ignore next */\n if (!ttl || !start) return\n status.ttl = ttl\n status.start = start\n status.now = cachedNow || getNow()\n const age = status.now - start\n status.remainingTTL = ttl - age\n }\n }\n\n // debounce calls to perf.now() to 1s so we're not hitting\n // that costly call repeatedly.\n let cachedNow = 0\n const getNow = () => {\n const n = perf.now()\n if (this.ttlResolution > 0) {\n cachedNow = n\n const t = setTimeout(\n () => (cachedNow = 0),\n this.ttlResolution\n )\n // not available on all platforms\n /* c8 ignore start */\n if (t.unref) {\n t.unref()\n }\n /* c8 ignore stop */\n }\n return n\n }\n\n this.getRemainingTTL = key => {\n const index = this.#keyMap.get(key)\n if (index === undefined) {\n return 0\n }\n const ttl = ttls[index]\n const start = starts[index]\n if (!ttl || !start) {\n return Infinity\n }\n const age = (cachedNow || getNow()) - start\n return ttl - age\n }\n\n this.#isStale = index => {\n const s = starts[index]\n const t = ttls[index]\n return !!t && !!s && (cachedNow || getNow()) - s > t\n }\n }\n\n // conditionally set private methods related to TTL\n #updateItemAge: (index: Index) => void = () => {}\n #statusTTL: (status: LRUCache.Status<V>, index: Index) => void =\n () => {}\n #setItemTTL: (\n index: Index,\n ttl: LRUCache.Milliseconds,\n start?: LRUCache.Milliseconds\n // ignore because we never call this if we're not already in TTL mode\n /* c8 ignore start */\n ) => void = () => {}\n /* c8 ignore stop */\n\n #isStale: (index: Index) => boolean = () => false\n\n #initializeSizeTracking() {\n const sizes = new ZeroArray(this.#max)\n this.#calculatedSize = 0\n this.#sizes = sizes\n this.#removeItemSize = index => {\n this.#calculatedSize -= sizes[index] as number\n sizes[index] = 0\n }\n this.#requireSize = (k, v, size, sizeCalculation) => {\n // provisionally accept background fetches.\n // actual value size will be checked when they return.\n if (this.#isBackgroundFetch(v)) {\n return 0\n }\n if (!isPosInt(size)) {\n if (sizeCalculation) {\n if (typeof sizeCalculation !== 'function') {\n throw new TypeError('sizeCalculation must be a function')\n }\n size = sizeCalculation(v, k)\n if (!isPosInt(size)) {\n throw new TypeError(\n 'sizeCalculation return invalid (expect positive integer)'\n )\n }\n } else {\n throw new TypeError(\n 'invalid size value (must be positive integer). ' +\n 'When maxSize or maxEntrySize is used, sizeCalculation ' +\n 'or size must be set.'\n )\n }\n }\n return size\n }\n this.#addItemSize = (\n index: Index,\n size: LRUCache.Size,\n status?: LRUCache.Status<V>\n ) => {\n sizes[index] = size\n if (this.#maxSize) {\n const maxSize = this.#maxSize - (sizes[index] as number)\n while (this.#calculatedSize > maxSize) {\n this.#evict(true)\n }\n }\n this.#calculatedSize += sizes[index] as number\n if (status) {\n status.entrySize = size\n status.totalCalculatedSize = this.#calculatedSize\n }\n }\n }\n\n #removeItemSize: (index: Index) => void = _i => {}\n #addItemSize: (\n index: Index,\n size: LRUCache.Size,\n status?: LRUCache.Status<V>\n ) => void = (_i, _s, _st) => {}\n #requireSize: (\n k: K,\n v: V | BackgroundFetch<V>,\n size?: LRUCache.Size,\n sizeCalculation?: LRUCache.SizeCalculator<K, V>\n ) => LRUCache.Size = (\n _k: K,\n _v: V | BackgroundFetch<V>,\n size?: LRUCache.Size,\n sizeCalculation?: LRUCache.SizeCalculator<K, V>\n ) => {\n if (size || sizeCalculation) {\n throw new TypeError(\n 'cannot set size without setting maxSize or maxEntrySize on cache'\n )\n }\n return 0\n };\n\n *#indexes({ allowStale = this.allowStale } = {}) {\n if (this.#size) {\n for (let i = this.#tail; true; ) {\n if (!this.#isValidIndex(i)) {\n break\n }\n if (allowStale || !this.#isStale(i)) {\n yield i\n }\n if (i === this.#head) {\n break\n } else {\n i = this.#prev[i] as Index\n }\n }\n }\n }\n\n *#rindexes({ allowStale = this.allowStale } = {}) {\n if (this.#size) {\n for (let i = this.#head; true; ) {\n if (!this.#isValidIndex(i)) {\n break\n }\n if (allowStale || !this.#isStale(i)) {\n yield i\n }\n if (i === this.#tail) {\n break\n } else {\n i = this.#next[i] as Index\n }\n }\n }\n }\n\n #isValidIndex(index: Index) {\n return (\n index !== undefined &&\n this.#keyMap.get(this.#keyList[index] as K) === index\n )\n }\n\n /**\n * Return a generator yielding `[key, value]` pairs,\n * in order from most recently used to least recently used.\n */\n *entries() {\n for (const i of this.#indexes()) {\n if (\n this.#valList[i] !== undefined &&\n this.#keyList[i] !== undefined &&\n !this.#isBackgroundFetch(this.#valList[i])\n ) {\n yield [this.#keyList[i], this.#valList[i]] as [K, V]\n }\n }\n }\n\n /**\n * Inverse order version of {@link LRUCache.entries}\n *\n * Return a generator yielding `[key, value]` pairs,\n * in order from least recently used to most recently used.\n */\n *rentries() {\n for (const i of this.#rindexes()) {\n if (\n this.#valList[i] !== undefined &&\n this.#keyList[i] !== undefined &&\n !this.#isBackgroundFetch(this.#valList[i])\n ) {\n yield [this.#keyList[i], this.#valList[i]]\n }\n }\n }\n\n /**\n * Return a generator yielding the keys in the cache,\n * in order from most recently used to least recently used.\n */\n *keys() {\n for (const i of this.#indexes()) {\n const k = this.#keyList[i]\n if (\n k !== undefined &&\n !this.#isBackgroundFetch(this.#valList[i])\n ) {\n yield k\n }\n }\n }\n\n /**\n * Inverse order version of {@link LRUCache.keys}\n *\n * Return a generator yielding the keys in the cache,\n * in order from least recently used to most recently used.\n */\n *rkeys() {\n for (const i of this.#rindexes()) {\n const k = this.#keyList[i]\n if (\n k !== undefined &&\n !this.#isBackgroundFetch(this.#valList[i])\n ) {\n yield k\n }\n }\n }\n\n /**\n * Return a generator yielding the values in the cache,\n * in order from most recently used to least recently used.\n */\n *values() {\n for (const i of this.#indexes()) {\n const v = this.#valList[i]\n if (\n v !== undefined &&\n !this.#isBackgroundFetch(this.#valList[i])\n ) {\n yield this.#valList[i] as V\n }\n }\n }\n\n /**\n * Inverse order version of {@link LRUCache.values}\n *\n * Return a generator yielding the values in the cache,\n * in order from least recently used to most recently used.\n */\n *rvalues() {\n for (const i of this.#rindexes()) {\n const v = this.#valList[i]\n if (\n v !== undefined &&\n !this.#isBackgroundFetch(this.#valList[i])\n ) {\n yield this.#valList[i]\n }\n }\n }\n\n /**\n * Iterating over the cache itself yields the same results as\n * {@link LRUCache.entries}\n */\n [Symbol.iterator]() {\n return this.entries()\n }\n\n /**\n * A String value that is used in the creation of the default string\n * description of an object. Called by the built-in method\n * `Object.prototype.toString`.\n */\n [Symbol.toStringTag] = 'LRUCache'\n\n /**\n * Find a value for which the supplied fn method returns a truthy value,\n * similar to `Array.find()`. fn is called as `fn(value, key, cache)`.\n */\n find(\n fn: (v: V, k: K, self: LRUCache<K, V, FC>) => boolean,\n getOptions: LRUCache.GetOptions<K, V, FC> = {}\n ) {\n for (const i of this.#indexes()) {\n const v = this.#valList[i]\n const value = this.#isBackgroundFetch(v)\n ? v.__staleWhileFetching\n : v\n if (value === undefined) continue\n if (fn(value, this.#keyList[i] as K, this)) {\n return this.get(this.#keyList[i] as K, getOptions)\n }\n }\n }\n\n /**\n * Call the supplied function on each item in the cache, in order from most\n * recently used to least recently used.\n *\n * `fn` is called as `fn(value, key, cache)`.\n *\n * If `thisp` is provided, function will be called in the `this`-context of\n * the provided object, or the cache if no `thisp` object is provided.\n *\n * Does not update age or recenty of use, or iterate over stale values.\n */\n forEach(\n fn: (v: V, k: K, self: LRUCache<K, V, FC>) => any,\n thisp: any = this\n ) {\n for (const i of this.#indexes()) {\n const v = this.#valList[i]\n const value = this.#isBackgroundFetch(v)\n ? v.__staleWhileFetching\n : v\n if (value === undefined) continue\n fn.call(thisp, value, this.#keyList[i] as K, this)\n }\n }\n\n /**\n * The same as {@link LRUCache.forEach} but items are iterated over in\n * reverse order. (ie, less recently used items are iterated over first.)\n */\n rforEach(\n fn: (v: V, k: K, self: LRUCache<K, V, FC>) => any,\n thisp: any = this\n ) {\n for (const i of this.#rindexes()) {\n const v = this.#valList[i]\n const value = this.#isBackgroundFetch(v)\n ? v.__staleWhileFetching\n : v\n if (value === undefined) continue\n fn.call(thisp, value, this.#keyList[i] as K, this)\n }\n }\n\n /**\n * Delete any stale entries. Returns true if anything was removed,\n * false otherwise.\n */\n purgeStale() {\n let deleted = false\n for (const i of this.#rindexes({ allowStale: true })) {\n if (this.#isStale(i)) {\n this.#delete(this.#keyList[i] as K, 'expire')\n deleted = true\n }\n }\n return deleted\n }\n\n /**\n * Get the extended info about a given entry, to get its value, size, and\n * TTL info simultaneously. Returns `undefined` if the key is not present.\n *\n * Unlike {@link LRUCache#dump}, which is designed to be portable and survive\n * serialization, the `start` value is always the current timestamp, and the\n * `ttl` is a calculated remaining time to live (negative if expired).\n *\n * Always returns stale values, if their info is found in the cache, so be\n * sure to check for expirations (ie, a negative {@link LRUCache.Entry#ttl})\n * if relevant.\n */\n info(key: K): LRUCache.Entry<V> | undefined {\n const i = this.#keyMap.get(key)\n if (i === undefined) return undefined\n const v = this.#valList[i]\n const value: V | undefined = this.#isBackgroundFetch(v)\n ? v.__staleWhileFetching\n : v\n if (value === undefined) return undefined\n const entry: LRUCache.Entry<V> = { value }\n if (this.#ttls && this.#starts) {\n const ttl = this.#ttls[i]\n const start = this.#starts[i]\n if (ttl && start) {\n const remain = ttl - (perf.now() - start)\n entry.ttl = remain\n entry.start = Date.now()\n }\n }\n if (this.#sizes) {\n entry.size = this.#sizes[i]\n }\n return entry\n }\n\n /**\n * Return an array of [key, {@link LRUCache.Entry}] tuples which can be\n * passed to {@link LRLUCache#load}.\n *\n * The `start` fields are calculated relative to a portable `Date.now()`\n * timestamp, even if `performance.now()` is available.\n *\n * Stale entries are always included in the `dump`, even if\n * {@link LRUCache.OptionsBase.allowStale} is false.\n *\n * Note: this returns an actual array, not a generator, so it can be more\n * easily passed around.\n */\n dump() {\n const arr: [K, LRUCache.Entry<V>][] = []\n for (const i of this.#indexes({ allowStale: true })) {\n const key = this.#keyList[i]\n const v = this.#valList[i]\n const value: V | undefined = this.#isBackgroundFetch(v)\n ? v.__staleWhileFetching\n : v\n if (value === undefined || key === undefined) continue\n const entry: LRUCache.Entry<V> = { value }\n if (this.#ttls && this.#starts) {\n entry.ttl = this.#ttls[i]\n // always dump the start relative to a portable timestamp\n // it's ok for this to be a bit slow, it's a rare operation.\n const age = perf.now() - (this.#starts[i] as number)\n entry.start = Math.floor(Date.now() - age)\n }\n if (this.#sizes) {\n entry.size = this.#sizes[i]\n }\n arr.unshift([key, entry])\n }\n return arr\n }\n\n /**\n * Reset the cache and load in the items in entries in the order listed.\n *\n * The shape of the resulting cache may be different if the same options are\n * not used in both caches.\n *\n * The `start` fields are assumed to be calculated relative to a portable\n * `Date.now()` timestamp, even if `performance.now()` is available.\n */\n load(arr: [K, LRUCache.Entry<V>][]) {\n this.clear()\n for (const [key, entry] of arr) {\n if (entry.start) {\n // entry.start is a portable timestamp, but we may be using\n // node's performance.now(), so calculate the offset, so that\n // we get the intended remaining TTL, no matter how long it's\n // been on ice.\n //\n // it's ok for this to be a bit slow, it's a rare operation.\n const age = Date.now() - entry.start\n entry.start = perf.now() - age\n }\n this.set(key, entry.value, entry)\n }\n }\n\n /**\n * Add a value to the cache.\n *\n * Note: if `undefined` is specified as a value, this is an alias for\n * {@link LRUCache#delete}\n *\n * Fields on the {@link LRUCache.SetOptions} options param will override\n * their corresponding values in the constructor options for the scope\n * of this single `set()` operation.\n *\n * If `start` is provided, then that will set the effective start\n * time for the TTL calculation. Note that this must be a previous\n * value of `performance.now()` if supported, or a previous value of\n * `Date.now()` if not.\n *\n * Options object may also include `size`, which will prevent\n * calling the `sizeCalculation` function and just use the specified\n * number if it is a positive integer, and `noDisposeOnSet` which\n * will prevent calling a `dispose` function in the case of\n * overwrites.\n *\n * If the `size` (or return value of `sizeCalculation`) for a given\n * entry is greater than `maxEntrySize`, then the item will not be\n * added to the cache.\n *\n * Will update the recency of the entry.\n *\n * If the value is `undefined`, then this is an alias for\n * `cache.delete(key)`. `undefined` is never stored in the cache.\n */\n set(\n k: K,\n v: V | BackgroundFetch<V> | undefined,\n setOptions: LRUCache.SetOptions<K, V, FC> = {}\n ) {\n if (v === undefined) {\n this.delete(k)\n return this\n }\n const {\n ttl = this.ttl,\n start,\n noDisposeOnSet = this.noDisposeOnSet,\n sizeCalculation = this.sizeCalculation,\n status,\n } = setOptions\n let { noUpdateTTL = this.noUpdateTTL } = setOptions\n\n const size = this.#requireSize(\n k,\n v,\n setOptions.size || 0,\n sizeCalculation\n )\n // if the item doesn't fit, don't do anything\n // NB: maxEntrySize set to maxSize by default\n if (this.maxEntrySize && size > this.maxEntrySize) {\n if (status) {\n status.set = 'miss'\n status.maxEntrySizeExceeded = true\n }\n // have to delete, in case something is there already.\n this.#delete(k, 'set')\n return this\n }\n let index = this.#size === 0 ? undefined : this.#keyMap.get(k)\n if (index === undefined) {\n // addition\n index = (\n this.#size === 0\n ? this.#tail\n : this.#free.length !== 0\n ? this.#free.pop()\n : this.#size === this.#max\n ? this.#evict(false)\n : this.#size\n ) as Index\n this.#keyList[index] = k\n this.#valList[index] = v\n this.#keyMap.set(k, index)\n this.#next[this.#tail] = index\n this.#prev[index] = this.#tail\n this.#tail = index\n this.#size++\n this.#addItemSize(index, size, status)\n if (status) status.set = 'add'\n noUpdateTTL = false\n } else {\n // update\n this.#moveToTail(index)\n const oldVal = this.#valList[index] as V | BackgroundFetch<V>\n if (v !== oldVal) {\n if (this.#hasFetchMethod && this.#isBackgroundFetch(oldVal)) {\n oldVal.__abortController.abort(new Error('replaced'))\n const { __staleWhileFetching: s } = oldVal\n if (s !== undefined && !noDisposeOnSet) {\n if (this.#hasDispose) {\n this.#dispose?.(s as V, k, 'set')\n }\n if (this.#hasDisposeAfter) {\n this.#disposed?.push([s as V, k, 'set'])\n }\n }\n } else if (!noDisposeOnSet) {\n if (this.#hasDispose) {\n this.#dispose?.(oldVal as V, k, 'set')\n }\n if (this.#hasDisposeAfter) {\n this.#disposed?.push([oldVal as V, k, 'set'])\n }\n }\n this.#removeItemSize(index)\n this.#addItemSize(index, size, status)\n this.#valList[index] = v\n if (status) {\n status.set = 'replace'\n const oldValue =\n oldVal && this.#isBackgroundFetch(oldVal)\n ? oldVal.__staleWhileFetching\n : oldVal\n if (oldValue !== undefined) status.oldValue = oldValue\n }\n } else if (status) {\n status.set = 'update'\n }\n }\n if (ttl !== 0 && !this.#ttls) {\n this.#initializeTTLTracking()\n }\n if (this.#ttls) {\n if (!noUpdateTTL) {\n this.#setItemTTL(index, ttl, start)\n }\n if (status) this.#statusTTL(status, index)\n }\n if (!noDisposeOnSet && this.#hasDisposeAfter && this.#disposed) {\n const dt = this.#disposed\n let task: DisposeTask<K, V> | undefined\n while ((task = dt?.shift())) {\n this.#disposeAfter?.(...task)\n }\n }\n return this\n }\n\n /**\n * Evict the least recently used item, returning its value or\n * `undefined` if cache is empty.\n */\n pop(): V | undefined {\n try {\n while (this.#size) {\n const val = this.#valList[this.#head]\n this.#evict(true)\n if (this.#isBackgroundFetch(val)) {\n if (val.__staleWhileFetching) {\n return val.__staleWhileFetching\n }\n } else if (val !== undefined) {\n return val\n }\n }\n } finally {\n if (this.#hasDisposeAfter && this.#disposed) {\n const dt = this.#disposed\n let task: DisposeTask<K, V> | undefined\n while ((task = dt?.shift())) {\n this.#disposeAfter?.(...task)\n }\n }\n }\n }\n\n #evict(free: boolean) {\n const head = this.#head\n const k = this.#keyList[head] as K\n const v = this.#valList[head] as V\n if (this.#hasFetchMethod && this.#isBackgroundFetch(v)) {\n v.__abortController.abort(new Error('evicted'))\n } else if (this.#hasDispose || this.#hasDisposeAfter) {\n if (this.#hasDispose) {\n this.#dispose?.(v, k, 'evict')\n }\n if (this.#hasDisposeAfter) {\n this.#disposed?.push([v, k, 'evict'])\n }\n }\n this.#removeItemSize(head)\n // if we aren't about to use the index, then null these out\n if (free) {\n this.#keyList[head] = undefined\n this.#valList[head] = undefined\n this.#free.push(head)\n }\n if (this.#size === 1) {\n this.#head = this.#tail = 0 as Index\n this.#free.length = 0\n } else {\n this.#head = this.#next[head] as Index\n }\n this.#keyMap.delete(k)\n this.#size--\n return head\n }\n\n /**\n * Check if a key is in the cache, without updating the recency of use.\n * Will return false if the item is stale, even though it is technically\n * in the cache.\n *\n * Check if a key is in the cache, without updating the recency of\n * use. Age is updated if {@link LRUCache.OptionsBase.updateAgeOnHas} is set\n * to `true` in either the options or the constructor.\n *\n * Will return `false` if the item is stale, even though it is technically in\n * the cache. The difference can be determined (if it matters) by using a\n * `status` argument, and inspecting the `has` field.\n *\n * Will not update item age unless\n * {@link LRUCache.OptionsBase.updateAgeOnHas} is set.\n */\n has(k: K, hasOptions: LRUCache.HasOptions<K, V, FC> = {}) {\n const { updateAgeOnHas = this.updateAgeOnHas, status } =\n hasOptions\n const index = this.#keyMap.get(k)\n if (index !== undefined) {\n const v = this.#valList[index]\n if (\n this.#isBackgroundFetch(v) &&\n v.__staleWhileFetching === undefined\n ) {\n return false\n }\n if (!this.#isStale(index)) {\n if (updateAgeOnHas) {\n this.#updateItemAge(index)\n }\n if (status) {\n status.has = 'hit'\n this.#statusTTL(status, index)\n }\n return true\n } else if (status) {\n status.has = 'stale'\n this.#statusTTL(status, index)\n }\n } else if (status) {\n status.has = 'miss'\n }\n return false\n }\n\n /**\n * Like {@link LRUCache#get} but doesn't update recency or delete stale\n * items.\n *\n * Returns `undefined` if the item is stale, unless\n * {@link LRUCache.OptionsBase.allowStale} is set.\n */\n peek(k: K, peekOptions: LRUCache.PeekOptions<K, V, FC> = {}) {\n const { allowStale = this.allowStale } = peekOptions\n const index = this.#keyMap.get(k)\n if (\n index === undefined ||\n (!allowStale && this.#isStale(index))\n ) {\n return\n }\n const v = this.#valList[index]\n // either stale and allowed, or forcing a refresh of non-stale value\n return this.#isBackgroundFetch(v) ? v.__staleWhileFetching : v\n }\n\n #backgroundFetch(\n k: K,\n index: Index | undefined,\n options: LRUCache.FetchOptions<K, V, FC>,\n context: any\n ): BackgroundFetch<V> {\n const v = index === undefined ? undefined : this.#valList[index]\n if (this.#isBackgroundFetch(v)) {\n return v\n }\n\n const ac = new AC()\n const { signal } = options\n // when/if our AC signals, then stop listening to theirs.\n signal?.addEventListener('abort', () => ac.abort(signal.reason), {\n signal: ac.signal,\n })\n\n const fetchOpts = {\n signal: ac.signal,\n options,\n context,\n }\n\n const cb = (\n v: V | undefined,\n updateCache = false\n ): V | undefined => {\n const { aborted } = ac.signal\n const ignoreAbort = options.ignoreFetchAbort && v !== undefined\n if (options.status) {\n if (aborted && !updateCache) {\n options.status.fetchAborted = true\n options.status.fetchError = ac.signal.reason\n if (ignoreAbort) options.status.fetchAbortIgnored = true\n } else {\n options.status.fetchResolved = true\n }\n }\n if (aborted && !ignoreAbort && !updateCache) {\n return fetchFail(ac.signal.reason)\n }\n // either we didn't abort, and are still here, or we did, and ignored\n const bf = p as BackgroundFetch<V>\n if (this.#valList[index as Index] === p) {\n if (v === undefined) {\n if (bf.__staleWhileFetching) {\n this.#valList[index as Index] = bf.__staleWhileFetching\n } else {\n this.#delete(k, 'fetch')\n }\n } else {\n if (options.status) options.status.fetchUpdated = true\n this.set(k, v, fetchOpts.options)\n }\n }\n return v\n }\n\n const eb = (er: any) => {\n if (options.status) {\n options.status.fetchRejected = true\n options.status.fetchError = er\n }\n return fetchFail(er)\n }\n\n const fetchFail = (er: any): V | undefined => {\n const { aborted } = ac.signal\n const allowStaleAborted =\n aborted && options.allowStaleOnFetchAbort\n const allowStale =\n allowStaleAborted || options.allowStaleOnFetchRejection\n const noDelete = allowStale || options.noDeleteOnFetchRejection\n const bf = p as BackgroundFetch<V>\n if (this.#valList[index as Index] === p) {\n // if we allow stale on fetch rejections, then we need to ensure that\n // the stale value is not removed from the cache when the fetch fails.\n const del = !noDelete || bf.__staleWhileFetching === undefined\n if (del) {\n this.#delete(k, 'fetch')\n } else if (!allowStaleAborted) {\n // still replace the *promise* with the stale value,\n // since we are done with the promise at this point.\n // leave it untouched if we're still waiting for an\n // aborted background fetch that hasn't yet returned.\n this.#valList[index as Index] = bf.__staleWhileFetching\n }\n }\n if (allowStale) {\n if (options.status && bf.__staleWhileFetching !== undefined) {\n options.status.returnedStale = true\n }\n return bf.__staleWhileFetching\n } else if (bf.__returned === bf) {\n throw er\n }\n }\n\n const pcall = (\n res: (v: V | undefined) => void,\n rej: (e: any) => void\n ) => {\n const fmp = this.#fetchMethod?.(k, v, fetchOpts)\n if (fmp && fmp instanceof Promise) {\n fmp.then(v => res(v === undefined ? undefined : v), rej)\n }\n // ignored, we go until we finish, regardless.\n // defer check until we are actually aborting,\n // so fetchMethod can override.\n ac.signal.addEventListener('abort', () => {\n if (\n !options.ignoreFetchAbort ||\n options.allowStaleOnFetchAbort\n ) {\n res(undefined)\n // when it eventually resolves, update the cache.\n if (options.allowStaleOnFetchAbort) {\n res = v => cb(v, true)\n }\n }\n })\n }\n\n if (options.status) options.status.fetchDispatched = true\n const p = new Promise(pcall).then(cb, eb)\n const bf: BackgroundFetch<V> = Object.assign(p, {\n __abortController: ac,\n __staleWhileFetching: v,\n __returned: undefined,\n })\n\n if (index === undefined) {\n // internal, don't expose status.\n this.set(k, bf, { ...fetchOpts.options, status: undefined })\n index = this.#keyMap.get(k)\n } else {\n this.#valList[index] = bf\n }\n return bf\n }\n\n #isBackgroundFetch(p: any): p is BackgroundFetch<V> {\n if (!this.#hasFetchMethod) return false\n const b = p as BackgroundFetch<V>\n return (\n !!b &&\n b instanceof Promise &&\n b.hasOwnProperty('__staleWhileFetching') &&\n b.__abortController instanceof AC\n )\n }\n\n /**\n * Make an asynchronous cached fetch using the\n * {@link LRUCache.OptionsBase.fetchMethod} function.\n *\n * If the value is in the cache and not stale, then the returned\n * Promise resolves to the value.\n *\n * If not in the cache, or beyond its TTL staleness, then\n * `fetchMethod(key, staleValue, { options, signal, context })` is\n * called, and the value returned will be added to the cache once\n * resolved.\n *\n * If called with `allowStale`, and an asynchronous fetch is\n * currently in progress to reload a stale value, then the former\n * stale value will be returned.\n *\n * If called with `forceRefresh`, then the cached item will be\n * re-fetched, even if it is not stale. However, if `allowStale` is also\n * set, then the old value will still be returned. This is useful\n * in cases where you want to force a reload of a cached value. If\n * a background fetch is already in progress, then `forceRefresh`\n * has no effect.\n *\n * If multiple fetches for the same key are issued, then they will all be\n * coalesced into a single call to fetchMethod.\n *\n * Note that this means that handling options such as\n * {@link LRUCache.OptionsBase.allowStaleOnFetchAbort},\n * {@link LRUCache.FetchOptions.signal},\n * and {@link LRUCache.OptionsBase.allowStaleOnFetchRejection} will be\n * determined by the FIRST fetch() call for a given key.\n *\n * This is a known (fixable) shortcoming which will be addresed on when\n * someone complains about it, as the fix would involve added complexity and\n * may not be worth the costs for this edge case.\n *\n * If {@link LRUCache.OptionsBase.fetchMethod} is not specified, then this is\n * effectively an alias for `Promise.resolve(cache.get(key))`.\n *\n * When the fetch method resolves to a value, if the fetch has not\n * been aborted due to deletion, eviction, or being overwritten,\n * then it is added to the cache using the options provided.\n *\n * If the key is evicted or deleted before the `fetchMethod`\n * resolves, then the AbortSignal passed to the `fetchMethod` will\n * receive an `abort` event, and the promise returned by `fetch()`\n * will reject with the reason for the abort.\n *\n * If a `signal` is passed to the `fetch()` call, then aborting the\n * signal will abort the fetch and cause the `fetch()` promise to\n * reject with the reason provided.\n *\n * **Setting `context`**\n *\n * If an `FC` type is set to a type other than `unknown`, `void`, or\n * `undefined` in the {@link LRUCache} constructor, then all\n * calls to `cache.fetch()` _must_ provide a `context` option. If\n * set to `undefined` or `void`, then calls to fetch _must not_\n * provide a `context` option.\n *\n * The `context` param allows you to provide arbitrary data that\n * might be relevant in the course of fetching the data. It is only\n * relevant for the course of a single `fetch()` operation, and\n * discarded afterwards.\n *\n * **Note: `fetch()` calls are inflight-unique**\n *\n * If you call `fetch()` multiple times with the same key value,\n * then every call after the first will resolve on the same\n * promise<sup>1</sup>,\n * _even if they have different settings that would otherwise change\n * the behavior of the fetch_, such as `noDeleteOnFetchRejection`\n * or `ignoreFetchAbort`.\n *\n * In most cases, this is not a problem (in fact, only fetching\n * something once is what you probably want, if you're caching in\n * the first place). If you are changing the fetch() options\n * dramatically between runs, there's a good chance that you might\n * be trying to fit divergent semantics into a single object, and\n * would be better off with multiple cache instances.\n *\n * **1**: Ie, they're not the \"same Promise\", but they resolve at\n * the same time, because they're both waiting on the same\n * underlying fetchMethod response.\n */\n\n fetch(\n k: K,\n fetchOptions: unknown extends FC\n ? LRUCache.FetchOptions<K, V, FC>\n : FC extends undefined | void\n ? LRUCache.FetchOptionsNoContext<K, V>\n : LRUCache.FetchOptionsWithContext<K, V, FC>\n ): Promise<undefined | V>\n\n // this overload not allowed if context is required\n fetch(\n k: unknown extends FC\n ? K\n : FC extends undefined | void\n ? K\n : never,\n fetchOptions?: unknown extends FC\n ? LRUCache.FetchOptions<K, V, FC>\n : FC extends undefined | void\n ? LRUCache.FetchOptionsNoContext<K, V>\n : never\n ): Promise<undefined | V>\n\n async fetch(\n k: K,\n fetchOptions: LRUCache.FetchOptions<K, V, FC> = {}\n ): Promise<undefined | V> {\n const {\n // get options\n allowStale = this.allowStale,\n updateAgeOnGet = this.updateAgeOnGet,\n noDeleteOnStaleGet = this.noDeleteOnStaleGet,\n // set options\n ttl = this.ttl,\n noDisposeOnSet = this.noDisposeOnSet,\n size = 0,\n sizeCalculation = this.sizeCalculation,\n noUpdateTTL = this.noUpdateTTL,\n // fetch exclusive options\n noDeleteOnFetchRejection = this.noDeleteOnFetchRejection,\n allowStaleOnFetchRejection = this.allowStaleOnFetchRejection,\n ignoreFetchAbort = this.ignoreFetchAbort,\n allowStaleOnFetchAbort = this.allowStaleOnFetchAbort,\n context,\n forceRefresh = false,\n status,\n signal,\n } = fetchOptions\n\n if (!this.#hasFetchMethod) {\n if (status) status.fetch = 'get'\n return this.get(k, {\n allowStale,\n updateAgeOnGet,\n noDeleteOnStaleGet,\n status,\n })\n }\n\n const options = {\n allowStale,\n updateAgeOnGet,\n noDeleteOnStaleGet,\n ttl,\n noDisposeOnSet,\n size,\n sizeCalculation,\n noUpdateTTL,\n noDeleteOnFetchRejection,\n allowStaleOnFetchRejection,\n allowStaleOnFetchAbort,\n ignoreFetchAbort,\n status,\n signal,\n }\n\n let index = this.#keyMap.get(k)\n if (index === undefined) {\n if (status) status.fetch = 'miss'\n const p = this.#backgroundFetch(k, index, options, context)\n return (p.__returned = p)\n } else {\n // in cache, maybe already fetching\n const v = this.#valList[index]\n if (this.#isBackgroundFetch(v)) {\n const stale =\n allowStale && v.__staleWhileFetching !== undefined\n if (status) {\n status.fetch = 'inflight'\n if (stale) status.returnedStale = true\n }\n return stale ? v.__staleWhileFetching : (v.__returned = v)\n }\n\n // if we force a refresh, that means do NOT serve the cached value,\n // unless we are already in the process of refreshing the cache.\n const isStale = this.#isStale(index)\n if (!forceRefresh && !isStale) {\n if (status) status.fetch = 'hit'\n this.#moveToTail(index)\n if (updateAgeOnGet) {\n this.#updateItemAge(index)\n }\n if (status) this.#statusTTL(status, index)\n return v\n }\n\n // ok, it is stale or a forced refresh, and not already fetching.\n // refresh the cache.\n const p = this.#backgroundFetch(k, index, options, context)\n const hasStale = p.__staleWhileFetching !== undefined\n const staleVal = hasStale && allowStale\n if (status) {\n status.fetch = isStale ? 'stale' : 'refresh'\n if (staleVal && isStale) status.returnedStale = true\n }\n return staleVal ? p.__staleWhileFetching : (p.__returned = p)\n }\n }\n\n /**\n * In some cases, `cache.fetch()` may resolve to `undefined`, either because\n * a {@link LRUCache.OptionsBase#fetchMethod} was not provided (turning\n * `cache.fetch(k)` into just an async wrapper around `cache.get(k)`) or\n * because `ignoreFetchAbort` was specified (either to the constructor or\n * in the {@link LRUCache.FetchOptions}). Also, the\n * {@link OptionsBase.fetchMethod} may return `undefined` or `void`, making\n * the test even more complicated.\n *\n * Because inferring the cases where `undefined` might be returned are so\n * cumbersome, but testing for `undefined` can also be annoying, this method\n * can be used, which will reject if `this.fetch()` resolves to undefined.\n */\n forceFetch(\n k: K,\n fetchOptions: unknown extends FC\n ? LRUCache.FetchOptions<K, V, FC>\n : FC extends undefined | void\n ? LRUCache.FetchOptionsNoContext<K, V>\n : LRUCache.FetchOptionsWithContext<K, V, FC>\n ): Promise<V>\n // this overload not allowed if context is required\n forceFetch(\n k: unknown extends FC\n ? K\n : FC extends undefined | void\n ? K\n : never,\n fetchOptions?: unknown extends FC\n ? LRUCache.FetchOptions<K, V, FC>\n : FC extends undefined | void\n ? LRUCache.FetchOptionsNoContext<K, V>\n : never\n ): Promise<V>\n async forceFetch(\n k: K,\n fetchOptions: LRUCache.FetchOptions<K, V, FC> = {}\n ): Promise<V> {\n const v = await this.fetch(\n k,\n fetchOptions as unknown extends FC\n ? LRUCache.FetchOptions<K, V, FC>\n : FC extends undefined | void\n ? LRUCache.FetchOptionsNoContext<K, V>\n : LRUCache.FetchOptionsWithContext<K, V, FC>\n )\n if (v === undefined) throw new Error('fetch() returned undefined')\n return v\n }\n\n /**\n * If the key is found in the cache, then this is equivalent to\n * {@link LRUCache#get}. If not, in the cache, then calculate the value using\n * the {@link LRUCache.OptionsBase.memoMethod}, and add it to the cache.\n *\n * If an `FC` type is set to a type other than `unknown`, `void`, or\n * `undefined` in the LRUCache constructor, then all calls to `cache.memo()`\n * _must_ provide a `context` option. If set to `undefined` or `void`, then\n * calls to memo _must not_ provide a `context` option.\n *\n * The `context` param allows you to provide arbitrary data that might be\n * relevant in the course of fetching the data. It is only relevant for the\n * course of a single `memo()` operation, and discarded afterwards.\n */\n memo(\n k: K,\n memoOptions: unknown extends FC\n ? LRUCache.MemoOptions<K, V, FC>\n : FC extends undefined | void\n ? LRUCache.MemoOptionsNoContext<K, V>\n : LRUCache.MemoOptionsWithContext<K, V, FC>\n ): V\n // this overload not allowed if context is required\n memo(\n k: unknown extends FC\n ? K\n : FC extends undefined | void\n ? K\n : never,\n memoOptions?: unknown extends FC\n ? LRUCache.MemoOptions<K, V, FC>\n : FC extends undefined | void\n ? LRUCache.MemoOptionsNoContext<K, V>\n : never\n ): V\n memo(k: K, memoOptions: LRUCache.MemoOptions<K, V, FC> = {}) {\n const memoMethod = this.#memoMethod\n if (!memoMethod) {\n throw new Error('no memoMethod provided to constructor')\n }\n const { context, forceRefresh, ...options } = memoOptions\n const v = this.get(k, options)\n if (!forceRefresh && v !== undefined) return v\n const vv = memoMethod(k, v, {\n options,\n context,\n } as LRUCache.MemoizerOptions<K, V, FC>)\n this.set(k, vv, options)\n return vv\n }\n\n /**\n * Return a value from the cache. Will update the recency of the cache\n * entry found.\n *\n * If the key is not found, get() will return `undefined`.\n */\n get(k: K, getOptions: LRUCache.GetOptions<K, V, FC> = {}) {\n const {\n allowStale = this.allowStale,\n updateAgeOnGet = this.updateAgeOnGet,\n noDeleteOnStaleGet = this.noDeleteOnStaleGet,\n status,\n } = getOptions\n const index = this.#keyMap.get(k)\n if (index !== undefined) {\n const value = this.#valList[index]\n const fetching = this.#isBackgroundFetch(value)\n if (status) this.#statusTTL(status, index)\n if (this.#isStale(index)) {\n if (status) status.get = 'stale'\n // delete only if not an in-flight background fetch\n if (!fetching) {\n if (!noDeleteOnStaleGet) {\n this.#delete(k, 'expire')\n }\n if (status && allowStale) status.returnedStale = true\n return allowStale ? value : undefined\n } else {\n if (\n status &&\n allowStale &&\n value.__staleWhileFetching !== undefined\n ) {\n status.returnedStale = true\n }\n return allowStale ? value.__staleWhileFetching : undefined\n }\n } else {\n if (status) status.get = 'hit'\n // if we're currently fetching it, we don't actually have it yet\n // it's not stale, which means this isn't a staleWhileRefetching.\n // If it's not stale, and fetching, AND has a __staleWhileFetching\n // value, then that means the user fetched with {forceRefresh:true},\n // so it's safe to return that value.\n if (fetching) {\n return value.__staleWhileFetching\n }\n this.#moveToTail(index)\n if (updateAgeOnGet) {\n this.#updateItemAge(index)\n }\n return value\n }\n } else if (status) {\n status.get = 'miss'\n }\n }\n\n #connect(p: Index, n: Index) {\n this.#prev[n] = p\n this.#next[p] = n\n }\n\n #moveToTail(index: Index): void {\n // if tail already, nothing to do\n // if head, move head to next[index]\n // else\n // move next[prev[index]] to next[index] (head has no prev)\n // move prev[next[index]] to prev[index]\n // prev[index] = tail\n // next[tail] = index\n // tail = index\n if (index !== this.#tail) {\n if (index === this.#head) {\n this.#head = this.#next[index] as Index\n } else {\n this.#connect(\n this.#prev[index] as Index,\n this.#next[index] as Index\n )\n }\n this.#connect(this.#tail, index)\n this.#tail = index\n }\n }\n\n /**\n * Deletes a key out of the cache.\n *\n * Returns true if the key was deleted, false otherwise.\n */\n delete(k: K) {\n return this.#delete(k, 'delete')\n }\n\n #delete(k: K, reason: LRUCache.DisposeReason) {\n let deleted = false\n if (this.#size !== 0) {\n const index = this.#keyMap.get(k)\n if (index !== undefined) {\n deleted = true\n if (this.#size === 1) {\n this.#clear(reason)\n } else {\n this.#removeItemSize(index)\n const v = this.#valList[index]\n if (this.#isBackgroundFetch(v)) {\n v.__abortController.abort(new Error('deleted'))\n } else if (this.#hasDispose || this.#hasDisposeAfter) {\n if (this.#hasDispose) {\n this.#dispose?.(v as V, k, reason)\n }\n if (this.#hasDisposeAfter) {\n this.#disposed?.push([v as V, k, reason])\n }\n }\n this.#keyMap.delete(k)\n this.#keyList[index] = undefined\n this.#valList[index] = undefined\n if (index === this.#tail) {\n this.#tail = this.#prev[index] as Index\n } else if (index === this.#head) {\n this.#head = this.#next[index] as Index\n } else {\n const pi = this.#prev[index] as number\n this.#next[pi] = this.#next[index] as number\n const ni = this.#next[index] as number\n this.#prev[ni] = this.#prev[index] as number\n }\n this.#size--\n this.#free.push(index)\n }\n }\n }\n if (this.#hasDisposeAfter && this.#disposed?.length) {\n const dt = this.#disposed\n let task: DisposeTask<K, V> | undefined\n while ((task = dt?.shift())) {\n this.#disposeAfter?.(...task)\n }\n }\n return deleted\n }\n\n /**\n * Clear the cache entirely, throwing away all values.\n */\n clear() {\n return this.#clear('delete')\n }\n #clear(reason: LRUCache.DisposeReason) {\n for (const index of this.#rindexes({ allowStale: true })) {\n const v = this.#valList[index]\n if (this.#isBackgroundFetch(v)) {\n v.__abortController.abort(new Error('deleted'))\n } else {\n const k = this.#keyList[index]\n if (this.#hasDispose) {\n this.#dispose?.(v as V, k as K, reason)\n }\n if (this.#hasDisposeAfter) {\n this.#disposed?.push([v as V, k as K, reason])\n }\n }\n }\n\n this.#keyMap.clear()\n this.#valList.fill(undefined)\n this.#keyList.fill(undefined)\n if (this.#ttls && this.#starts) {\n this.#ttls.fill(0)\n this.#starts.fill(0)\n }\n if (this.#sizes) {\n this.#sizes.fill(0)\n }\n this.#head = 0 as Index\n this.#tail = 0 as Index\n this.#free.length = 0\n this.#calculatedSize = 0\n this.#size = 0\n if (this.#hasDisposeAfter && this.#disposed) {\n const dt = this.#disposed\n let task: DisposeTask<K, V> | undefined\n while ((task = dt?.shift())) {\n this.#disposeAfter?.(...task)\n }\n }\n }\n}\n","import { LRUCache } from 'lru-cache'\nimport { posix, win32 } from 'node:path'\n\nimport { fileURLToPath } from 'node:url'\n\nimport {\n lstatSync,\n readdir as readdirCB,\n readdirSync,\n readlinkSync,\n realpathSync as rps,\n} from 'fs'\nimport * as actualFS from 'node:fs'\n\nconst realpathSync = rps.native\n// TODO: test perf of fs/promises realpath vs realpathCB,\n// since the promises one uses realpath.native\n\nimport { lstat, readdir, readlink, realpath } from 'node:fs/promises'\n\nimport { Minipass } from 'minipass'\nimport type { Dirent, Stats } from 'node:fs'\n\n/**\n * An object that will be used to override the default `fs`\n * methods. Any methods that are not overridden will use Node's\n * built-in implementations.\n *\n * - lstatSync\n * - readdir (callback `withFileTypes` Dirent variant, used for\n * readdirCB and most walks)\n * - readdirSync\n * - readlinkSync\n * - realpathSync\n * - promises: Object containing the following async methods:\n * - lstat\n * - readdir (Dirent variant only)\n * - readlink\n * - realpath\n */\nexport interface FSOption {\n lstatSync?: (path: string) => Stats\n readdir?: (\n path: string,\n options: { withFileTypes: true },\n cb: (er: NodeJS.ErrnoException | null, entries?: Dirent[]) => any,\n ) => void\n readdirSync?: (\n path: string,\n options: { withFileTypes: true },\n ) => Dirent[]\n readlinkSync?: (path: string) => string\n realpathSync?: (path: string) => string\n promises?: {\n lstat?: (path: string) => Promise<Stats>\n readdir?: (\n path: string,\n options: { withFileTypes: true },\n ) => Promise<Dirent[]>\n readlink?: (path: string) => Promise<string>\n realpath?: (path: string) => Promise<string>\n [k: string]: any\n }\n [k: string]: any\n}\n\ninterface FSValue {\n lstatSync: (path: string) => Stats\n readdir: (\n path: string,\n options: { withFileTypes: true },\n cb: (er: NodeJS.ErrnoException | null, entries?: Dirent[]) => any,\n ) => void\n readdirSync: (path: string, options: { withFileTypes: true }) => Dirent[]\n readlinkSync: (path: string) => string\n realpathSync: (path: string) => string\n promises: {\n lstat: (path: string) => Promise<Stats>\n readdir: (\n path: string,\n options: { withFileTypes: true },\n ) => Promise<Dirent[]>\n readlink: (path: string) => Promise<string>\n realpath: (path: string) => Promise<string>\n [k: string]: any\n }\n [k: string]: any\n}\n\nconst defaultFS: FSValue = {\n lstatSync,\n readdir: readdirCB,\n readdirSync,\n readlinkSync,\n realpathSync,\n promises: {\n lstat,\n readdir,\n readlink,\n realpath,\n },\n}\n\n// if they just gave us require('fs') then use our default\nconst fsFromOption = (fsOption?: FSOption): FSValue =>\n !fsOption || fsOption === defaultFS || fsOption === actualFS ?\n defaultFS\n : {\n ...defaultFS,\n ...fsOption,\n promises: {\n ...defaultFS.promises,\n ...(fsOption.promises || {}),\n },\n }\n\n// turn something like //?/c:/ into c:\\\nconst uncDriveRegexp = /^\\\\\\\\\\?\\\\([a-z]:)\\\\?$/i\nconst uncToDrive = (rootPath: string): string =>\n rootPath.replace(/\\//g, '\\\\').replace(uncDriveRegexp, '$1\\\\')\n\n// windows paths are separated by either / or \\\nconst eitherSep = /[\\\\\\/]/\n\nconst UNKNOWN = 0 // may not even exist, for all we know\nconst IFIFO = 0b0001\nconst IFCHR = 0b0010\nconst IFDIR = 0b0100\nconst IFBLK = 0b0110\nconst IFREG = 0b1000\nconst IFLNK = 0b1010\nconst IFSOCK = 0b1100\nconst IFMT = 0b1111\n\nexport type Type =\n | 'Unknown'\n | 'FIFO'\n | 'CharacterDevice'\n | 'Directory'\n | 'BlockDevice'\n | 'File'\n | 'SymbolicLink'\n | 'Socket'\n\n// mask to unset low 4 bits\nconst IFMT_UNKNOWN = ~IFMT\n\n// set after successfully calling readdir() and getting entries.\nconst READDIR_CALLED = 0b0000_0001_0000\n// set after a successful lstat()\nconst LSTAT_CALLED = 0b0000_0010_0000\n// set if an entry (or one of its parents) is definitely not a dir\nconst ENOTDIR = 0b0000_0100_0000\n// set if an entry (or one of its parents) does not exist\n// (can also be set on lstat errors like EACCES or ENAMETOOLONG)\nconst ENOENT = 0b0000_1000_0000\n// cannot have child entries -- also verify &IFMT is either IFDIR or IFLNK\n// set if we fail to readlink\nconst ENOREADLINK = 0b0001_0000_0000\n// set if we know realpath() will fail\nconst ENOREALPATH = 0b0010_0000_0000\n\nconst ENOCHILD = ENOTDIR | ENOENT | ENOREALPATH\nconst TYPEMASK = 0b0011_1111_1111\n\nconst entToType = (s: Dirent | Stats) =>\n s.isFile() ? IFREG\n : s.isDirectory() ? IFDIR\n : s.isSymbolicLink() ? IFLNK\n : s.isCharacterDevice() ? IFCHR\n : s.isBlockDevice() ? IFBLK\n : s.isSocket() ? IFSOCK\n : s.isFIFO() ? IFIFO\n : UNKNOWN\n\n// normalize unicode path names\nconst normalizeCache = new Map<string, string>()\nconst normalize = (s: string) => {\n const c = normalizeCache.get(s)\n if (c) return c\n const n = s.normalize('NFKD')\n normalizeCache.set(s, n)\n return n\n}\n\nconst normalizeNocaseCache = new Map<string, string>()\nconst normalizeNocase = (s: string) => {\n const c = normalizeNocaseCache.get(s)\n if (c) return c\n const n = normalize(s.toLowerCase())\n normalizeNocaseCache.set(s, n)\n return n\n}\n\n/**\n * Options that may be provided to the Path constructor\n */\nexport interface PathOpts {\n fullpath?: string\n relative?: string\n relativePosix?: string\n parent?: PathBase\n /**\n * See {@link FSOption}\n */\n fs?: FSOption\n}\n\n/**\n * An LRUCache for storing resolved path strings or Path objects.\n * @internal\n */\nexport class ResolveCache extends LRUCache<string, string> {\n constructor() {\n super({ max: 256 })\n }\n}\n\n// In order to prevent blowing out the js heap by allocating hundreds of\n// thousands of Path entries when walking extremely large trees, the \"children\"\n// in this tree are represented by storing an array of Path entries in an\n// LRUCache, indexed by the parent. At any time, Path.children() may return an\n// empty array, indicating that it doesn't know about any of its children, and\n// thus has to rebuild that cache. This is fine, it just means that we don't\n// benefit as much from having the cached entries, but huge directory walks\n// don't blow out the stack, and smaller ones are still as fast as possible.\n//\n//It does impose some complexity when building up the readdir data, because we\n//need to pass a reference to the children array that we started with.\n\n/**\n * an LRUCache for storing child entries.\n * @internal\n */\nexport class ChildrenCache extends LRUCache<PathBase, Children> {\n constructor(maxSize: number = 16 * 1024) {\n super({\n maxSize,\n // parent + children\n sizeCalculation: a => a.length + 1,\n })\n }\n}\n\n/**\n * Array of Path objects, plus a marker indicating the first provisional entry\n *\n * @internal\n */\nexport type Children = PathBase[] & { provisional: number }\n\nconst setAsCwd = Symbol('PathScurry setAsCwd')\n\n/**\n * Path objects are sort of like a super-powered\n * {@link https://nodejs.org/docs/latest/api/fs.html#class-fsdirent fs.Dirent}\n *\n * Each one represents a single filesystem entry on disk, which may or may not\n * exist. It includes methods for reading various types of information via\n * lstat, readlink, and readdir, and caches all information to the greatest\n * degree possible.\n *\n * Note that fs operations that would normally throw will instead return an\n * \"empty\" value. This is in order to prevent excessive overhead from error\n * stack traces.\n */\nexport abstract class PathBase implements Dirent {\n /**\n * the basename of this path\n *\n * **Important**: *always* test the path name against any test string\n * usingthe {@link isNamed} method, and not by directly comparing this\n * string. Otherwise, unicode path strings that the system sees as identical\n * will not be properly treated as the same path, leading to incorrect\n * behavior and possible security issues.\n */\n name: string\n /**\n * the Path entry corresponding to the path root.\n *\n * @internal\n */\n root: PathBase\n /**\n * All roots found within the current PathScurry family\n *\n * @internal\n */\n roots: { [k: string]: PathBase }\n /**\n * a reference to the parent path, or undefined in the case of root entries\n *\n * @internal\n */\n parent?: PathBase\n /**\n * boolean indicating whether paths are compared case-insensitively\n * @internal\n */\n nocase: boolean\n\n /**\n * boolean indicating that this path is the current working directory\n * of the PathScurry collection that contains it.\n */\n isCWD: boolean = false\n\n /**\n * the string or regexp used to split paths. On posix, it is `'/'`, and on\n * windows it is a RegExp matching either `'/'` or `'\\\\'`\n */\n abstract splitSep: string | RegExp\n /**\n * The path separator string to use when joining paths\n */\n abstract sep: string\n\n // potential default fs override\n #fs: FSValue\n\n // Stats fields\n #dev?: number\n get dev() {\n return this.#dev\n }\n #mode?: number\n get mode() {\n return this.#mode\n }\n #nlink?: number\n get nlink() {\n return this.#nlink\n }\n #uid?: number\n get uid() {\n return this.#uid\n }\n #gid?: number\n get gid() {\n return this.#gid\n }\n #rdev?: number\n get rdev() {\n return this.#rdev\n }\n #blksize?: number\n get blksize() {\n return this.#blksize\n }\n #ino?: number\n get ino() {\n return this.#ino\n }\n #size?: number\n get size() {\n return this.#size\n }\n #blocks?: number\n get blocks() {\n return this.#blocks\n }\n #atimeMs?: number\n get atimeMs() {\n return this.#atimeMs\n }\n #mtimeMs?: number\n get mtimeMs() {\n return this.#mtimeMs\n }\n #ctimeMs?: number\n get ctimeMs() {\n return this.#ctimeMs\n }\n #birthtimeMs?: number\n get birthtimeMs() {\n return this.#birthtimeMs\n }\n #atime?: Date\n get atime() {\n return this.#atime\n }\n #mtime?: Date\n get mtime() {\n return this.#mtime\n }\n #ctime?: Date\n get ctime() {\n return this.#ctime\n }\n #birthtime?: Date\n get birthtime() {\n return this.#birthtime\n }\n\n #matchName: string\n #depth?: number\n #fullpath?: string\n #fullpathPosix?: string\n #relative?: string\n #relativePosix?: string\n #type: number\n #children: ChildrenCache\n #linkTarget?: PathBase\n #realpath?: PathBase\n\n /**\n * This property is for compatibility with the Dirent class as of\n * Node v20, where Dirent['parentPath'] refers to the path of the\n * directory that was passed to readdir. For root entries, it's the path\n * to the entry itself.\n */\n get parentPath(): string {\n return (this.parent || this).fullpath()\n }\n\n /**\n * Deprecated alias for Dirent['parentPath'] Somewhat counterintuitively,\n * this property refers to the *parent* path, not the path object itself.\n */\n get path(): string {\n return this.parentPath\n }\n\n /**\n * Do not create new Path objects directly. They should always be accessed\n * via the PathScurry class or other methods on the Path class.\n *\n * @internal\n */\n constructor(\n name: string,\n type: number = UNKNOWN,\n root: PathBase | undefined,\n roots: { [k: string]: PathBase },\n nocase: boolean,\n children: ChildrenCache,\n opts: PathOpts,\n ) {\n this.name = name\n this.#matchName = nocase ? normalizeNocase(name) : normalize(name)\n this.#type = type & TYPEMASK\n this.nocase = nocase\n this.roots = roots\n this.root = root || this\n this.#children = children\n this.#fullpath = opts.fullpath\n this.#relative = opts.relative\n this.#relativePosix = opts.relativePosix\n this.parent = opts.parent\n if (this.parent) {\n this.#fs = this.parent.#fs\n } else {\n this.#fs = fsFromOption(opts.fs)\n }\n }\n\n /**\n * Returns the depth of the Path object from its root.\n *\n * For example, a path at `/foo/bar` would have a depth of 2.\n */\n depth(): number {\n if (this.#depth !== undefined) return this.#depth\n if (!this.parent) return (this.#depth = 0)\n return (this.#depth = this.parent.depth() + 1)\n }\n\n /**\n * @internal\n */\n abstract getRootString(path: string): string\n /**\n * @internal\n */\n abstract getRoot(rootPath: string): PathBase\n /**\n * @internal\n */\n abstract newChild(name: string, type?: number, opts?: PathOpts): PathBase\n\n /**\n * @internal\n */\n childrenCache() {\n return this.#children\n }\n\n /**\n * Get the Path object referenced by the string path, resolved from this Path\n */\n resolve(path?: string): PathBase {\n if (!path) {\n return this\n }\n const rootPath = this.getRootString(path)\n const dir = path.substring(rootPath.length)\n const dirParts = dir.split(this.splitSep)\n const result: PathBase =\n rootPath ?\n this.getRoot(rootPath).#resolveParts(dirParts)\n : this.#resolveParts(dirParts)\n return result\n }\n\n #resolveParts(dirParts: string[]) {\n let p: PathBase = this\n for (const part of dirParts) {\n p = p.child(part)\n }\n return p\n }\n\n /**\n * Returns the cached children Path objects, if still available. If they\n * have fallen out of the cache, then returns an empty array, and resets the\n * READDIR_CALLED bit, so that future calls to readdir() will require an fs\n * lookup.\n *\n * @internal\n */\n children(): Children {\n const cached = this.#children.get(this)\n if (cached) {\n return cached\n }\n const children: Children = Object.assign([], { provisional: 0 })\n this.#children.set(this, children)\n this.#type &= ~READDIR_CALLED\n return children\n }\n\n /**\n * Resolves a path portion and returns or creates the child Path.\n *\n * Returns `this` if pathPart is `''` or `'.'`, or `parent` if pathPart is\n * `'..'`.\n *\n * This should not be called directly. If `pathPart` contains any path\n * separators, it will lead to unsafe undefined behavior.\n *\n * Use `Path.resolve()` instead.\n *\n * @internal\n */\n child(pathPart: string, opts?: PathOpts): PathBase {\n if (pathPart === '' || pathPart === '.') {\n return this\n }\n if (pathPart === '..') {\n return this.parent || this\n }\n\n // find the child\n const children = this.children()\n const name =\n this.nocase ? normalizeNocase(pathPart) : normalize(pathPart)\n for (const p of children) {\n if (p.#matchName === name) {\n return p\n }\n }\n\n // didn't find it, create provisional child, since it might not\n // actually exist. If we know the parent isn't a dir, then\n // in fact it CAN'T exist.\n const s = this.parent ? this.sep : ''\n const fullpath =\n this.#fullpath ? this.#fullpath + s + pathPart : undefined\n const pchild = this.newChild(pathPart, UNKNOWN, {\n ...opts,\n parent: this,\n fullpath,\n })\n\n if (!this.canReaddir()) {\n pchild.#type |= ENOENT\n }\n\n // don't have to update provisional, because if we have real children,\n // then provisional is set to children.length, otherwise a lower number\n children.push(pchild)\n return pchild\n }\n\n /**\n * The relative path from the cwd. If it does not share an ancestor with\n * the cwd, then this ends up being equivalent to the fullpath()\n */\n relative(): string {\n if (this.isCWD) return ''\n if (this.#relative !== undefined) {\n return this.#relative\n }\n const name = this.name\n const p = this.parent\n if (!p) {\n return (this.#relative = this.name)\n }\n const pv = p.relative()\n return pv + (!pv || !p.parent ? '' : this.sep) + name\n }\n\n /**\n * The relative path from the cwd, using / as the path separator.\n * If it does not share an ancestor with\n * the cwd, then this ends up being equivalent to the fullpathPosix()\n * On posix systems, this is identical to relative().\n */\n relativePosix(): string {\n if (this.sep === '/') return this.relative()\n if (this.isCWD) return ''\n if (this.#relativePosix !== undefined) return this.#relativePosix\n const name = this.name\n const p = this.parent\n if (!p) {\n return (this.#relativePosix = this.fullpathPosix())\n }\n const pv = p.relativePosix()\n return pv + (!pv || !p.parent ? '' : '/') + name\n }\n\n /**\n * The fully resolved path string for this Path entry\n */\n fullpath(): string {\n if (this.#fullpath !== undefined) {\n return this.#fullpath\n }\n const name = this.name\n const p = this.parent\n if (!p) {\n return (this.#fullpath = this.name)\n }\n const pv = p.fullpath()\n const fp = pv + (!p.parent ? '' : this.sep) + name\n return (this.#fullpath = fp)\n }\n\n /**\n * On platforms other than windows, this is identical to fullpath.\n *\n * On windows, this is overridden to return the forward-slash form of the\n * full UNC path.\n */\n fullpathPosix(): string {\n if (this.#fullpathPosix !== undefined) return this.#fullpathPosix\n if (this.sep === '/') return (this.#fullpathPosix = this.fullpath())\n if (!this.parent) {\n const p = this.fullpath().replace(/\\\\/g, '/')\n if (/^[a-z]:\\//i.test(p)) {\n return (this.#fullpathPosix = `//?/${p}`)\n } else {\n return (this.#fullpathPosix = p)\n }\n }\n const p = this.parent\n const pfpp = p.fullpathPosix()\n const fpp = pfpp + (!pfpp || !p.parent ? '' : '/') + this.name\n return (this.#fullpathPosix = fpp)\n }\n\n /**\n * Is the Path of an unknown type?\n *\n * Note that we might know *something* about it if there has been a previous\n * filesystem operation, for example that it does not exist, or is not a\n * link, or whether it has child entries.\n */\n isUnknown(): boolean {\n return (this.#type & IFMT) === UNKNOWN\n }\n\n isType(type: Type): boolean {\n return this[`is${type}`]()\n }\n\n getType(): Type {\n return (\n this.isUnknown() ? 'Unknown'\n : this.isDirectory() ? 'Directory'\n : this.isFile() ? 'File'\n : this.isSymbolicLink() ? 'SymbolicLink'\n : this.isFIFO() ? 'FIFO'\n : this.isCharacterDevice() ? 'CharacterDevice'\n : this.isBlockDevice() ? 'BlockDevice'\n : /* c8 ignore start */ this.isSocket() ? 'Socket'\n : 'Unknown'\n )\n /* c8 ignore stop */\n }\n\n /**\n * Is the Path a regular file?\n */\n isFile(): boolean {\n return (this.#type & IFMT) === IFREG\n }\n\n /**\n * Is the Path a directory?\n */\n isDirectory(): boolean {\n return (this.#type & IFMT) === IFDIR\n }\n\n /**\n * Is the path a character device?\n */\n isCharacterDevice(): boolean {\n return (this.#type & IFMT) === IFCHR\n }\n\n /**\n * Is the path a block device?\n */\n isBlockDevice(): boolean {\n return (this.#type & IFMT) === IFBLK\n }\n\n /**\n * Is the path a FIFO pipe?\n */\n isFIFO(): boolean {\n return (this.#type & IFMT) === IFIFO\n }\n\n /**\n * Is the path a socket?\n */\n isSocket(): boolean {\n return (this.#type & IFMT) === IFSOCK\n }\n\n /**\n * Is the path a symbolic link?\n */\n isSymbolicLink(): boolean {\n return (this.#type & IFLNK) === IFLNK\n }\n\n /**\n * Return the entry if it has been subject of a successful lstat, or\n * undefined otherwise.\n *\n * Does not read the filesystem, so an undefined result *could* simply\n * mean that we haven't called lstat on it.\n */\n lstatCached(): PathBase | undefined {\n return this.#type & LSTAT_CALLED ? this : undefined\n }\n\n /**\n * Return the cached link target if the entry has been the subject of a\n * successful readlink, or undefined otherwise.\n *\n * Does not read the filesystem, so an undefined result *could* just mean we\n * don't have any cached data. Only use it if you are very sure that a\n * readlink() has been called at some point.\n */\n readlinkCached(): PathBase | undefined {\n return this.#linkTarget\n }\n\n /**\n * Returns the cached realpath target if the entry has been the subject\n * of a successful realpath, or undefined otherwise.\n *\n * Does not read the filesystem, so an undefined result *could* just mean we\n * don't have any cached data. Only use it if you are very sure that a\n * realpath() has been called at some point.\n */\n realpathCached(): PathBase | undefined {\n return this.#realpath\n }\n\n /**\n * Returns the cached child Path entries array if the entry has been the\n * subject of a successful readdir(), or [] otherwise.\n *\n * Does not read the filesystem, so an empty array *could* just mean we\n * don't have any cached data. Only use it if you are very sure that a\n * readdir() has been called recently enough to still be valid.\n */\n readdirCached(): PathBase[] {\n const children = this.children()\n return children.slice(0, children.provisional)\n }\n\n /**\n * Return true if it's worth trying to readlink. Ie, we don't (yet) have\n * any indication that readlink will definitely fail.\n *\n * Returns false if the path is known to not be a symlink, if a previous\n * readlink failed, or if the entry does not exist.\n */\n canReadlink(): boolean {\n if (this.#linkTarget) return true\n if (!this.parent) return false\n // cases where it cannot possibly succeed\n const ifmt = this.#type & IFMT\n return !(\n (ifmt !== UNKNOWN && ifmt !== IFLNK) ||\n this.#type & ENOREADLINK ||\n this.#type & ENOENT\n )\n }\n\n /**\n * Return true if readdir has previously been successfully called on this\n * path, indicating that cachedReaddir() is likely valid.\n */\n calledReaddir(): boolean {\n return !!(this.#type & READDIR_CALLED)\n }\n\n /**\n * Returns true if the path is known to not exist. That is, a previous lstat\n * or readdir failed to verify its existence when that would have been\n * expected, or a parent entry was marked either enoent or enotdir.\n */\n isENOENT(): boolean {\n return !!(this.#type & ENOENT)\n }\n\n /**\n * Return true if the path is a match for the given path name. This handles\n * case sensitivity and unicode normalization.\n *\n * Note: even on case-sensitive systems, it is **not** safe to test the\n * equality of the `.name` property to determine whether a given pathname\n * matches, due to unicode normalization mismatches.\n *\n * Always use this method instead of testing the `path.name` property\n * directly.\n */\n isNamed(n: string): boolean {\n return !this.nocase ?\n this.#matchName === normalize(n)\n : this.#matchName === normalizeNocase(n)\n }\n\n /**\n * Return the Path object corresponding to the target of a symbolic link.\n *\n * If the Path is not a symbolic link, or if the readlink call fails for any\n * reason, `undefined` is returned.\n *\n * Result is cached, and thus may be outdated if the filesystem is mutated.\n */\n async readlink(): Promise<PathBase | undefined> {\n const target = this.#linkTarget\n if (target) {\n return target\n }\n if (!this.canReadlink()) {\n return undefined\n }\n /* c8 ignore start */\n // already covered by the canReadlink test, here for ts grumples\n if (!this.parent) {\n return undefined\n }\n /* c8 ignore stop */\n try {\n const read = await this.#fs.promises.readlink(this.fullpath())\n const linkTarget = (await this.parent.realpath())?.resolve(read)\n if (linkTarget) {\n return (this.#linkTarget = linkTarget)\n }\n } catch (er) {\n this.#readlinkFail((er as NodeJS.ErrnoException).code)\n return undefined\n }\n }\n\n /**\n * Synchronous {@link PathBase.readlink}\n */\n readlinkSync(): PathBase | undefined {\n const target = this.#linkTarget\n if (target) {\n return target\n }\n if (!this.canReadlink()) {\n return undefined\n }\n /* c8 ignore start */\n // already covered by the canReadlink test, here for ts grumples\n if (!this.parent) {\n return undefined\n }\n /* c8 ignore stop */\n try {\n const read = this.#fs.readlinkSync(this.fullpath())\n const linkTarget = this.parent.realpathSync()?.resolve(read)\n if (linkTarget) {\n return (this.#linkTarget = linkTarget)\n }\n } catch (er) {\n this.#readlinkFail((er as NodeJS.ErrnoException).code)\n return undefined\n }\n }\n\n #readdirSuccess(children: Children) {\n // succeeded, mark readdir called bit\n this.#type |= READDIR_CALLED\n // mark all remaining provisional children as ENOENT\n for (let p = children.provisional; p < children.length; p++) {\n const c = children[p]\n if (c) c.#markENOENT()\n }\n }\n\n #markENOENT() {\n // mark as UNKNOWN and ENOENT\n if (this.#type & ENOENT) return\n this.#type = (this.#type | ENOENT) & IFMT_UNKNOWN\n this.#markChildrenENOENT()\n }\n\n #markChildrenENOENT() {\n // all children are provisional and do not exist\n const children = this.children()\n children.provisional = 0\n for (const p of children) {\n p.#markENOENT()\n }\n }\n\n #markENOREALPATH() {\n this.#type |= ENOREALPATH\n this.#markENOTDIR()\n }\n\n // save the information when we know the entry is not a dir\n #markENOTDIR() {\n // entry is not a directory, so any children can't exist.\n // this *should* be impossible, since any children created\n // after it's been marked ENOTDIR should be marked ENOENT,\n // so it won't even get to this point.\n /* c8 ignore start */\n if (this.#type & ENOTDIR) return\n /* c8 ignore stop */\n let t = this.#type\n // this could happen if we stat a dir, then delete it,\n // then try to read it or one of its children.\n if ((t & IFMT) === IFDIR) t &= IFMT_UNKNOWN\n this.#type = t | ENOTDIR\n this.#markChildrenENOENT()\n }\n\n #readdirFail(code: string = '') {\n // markENOTDIR and markENOENT also set provisional=0\n if (code === 'ENOTDIR' || code === 'EPERM') {\n this.#markENOTDIR()\n } else if (code === 'ENOENT') {\n this.#markENOENT()\n } else {\n this.children().provisional = 0\n }\n }\n\n #lstatFail(code: string = '') {\n // Windows just raises ENOENT in this case, disable for win CI\n /* c8 ignore start */\n if (code === 'ENOTDIR') {\n // already know it has a parent by this point\n const p = this.parent as PathBase\n p.#markENOTDIR()\n } else if (code === 'ENOENT') {\n /* c8 ignore stop */\n this.#markENOENT()\n }\n }\n\n #readlinkFail(code: string = '') {\n let ter = this.#type\n ter |= ENOREADLINK\n if (code === 'ENOENT') ter |= ENOENT\n // windows gets a weird error when you try to readlink a file\n if (code === 'EINVAL' || code === 'UNKNOWN') {\n // exists, but not a symlink, we don't know WHAT it is, so remove\n // all IFMT bits.\n ter &= IFMT_UNKNOWN\n }\n this.#type = ter\n // windows just gets ENOENT in this case. We do cover the case,\n // just disabled because it's impossible on Windows CI\n /* c8 ignore start */\n if (code === 'ENOTDIR' && this.parent) {\n this.parent.#markENOTDIR()\n }\n /* c8 ignore stop */\n }\n\n #readdirAddChild(e: Dirent, c: Children) {\n return (\n this.#readdirMaybePromoteChild(e, c) ||\n this.#readdirAddNewChild(e, c)\n )\n }\n\n #readdirAddNewChild(e: Dirent, c: Children): PathBase {\n // alloc new entry at head, so it's never provisional\n const type = entToType(e)\n const child = this.newChild(e.name, type, { parent: this })\n const ifmt = child.#type & IFMT\n if (ifmt !== IFDIR && ifmt !== IFLNK && ifmt !== UNKNOWN) {\n child.#type |= ENOTDIR\n }\n c.unshift(child)\n c.provisional++\n return child\n }\n\n #readdirMaybePromoteChild(e: Dirent, c: Children): PathBase | undefined {\n for (let p = c.provisional; p < c.length; p++) {\n const pchild = c[p]\n const name =\n this.nocase ? normalizeNocase(e.name) : normalize(e.name)\n if (name !== pchild!.#matchName) {\n continue\n }\n\n return this.#readdirPromoteChild(e, pchild!, p, c)\n }\n }\n\n #readdirPromoteChild(\n e: Dirent,\n p: PathBase,\n index: number,\n c: Children,\n ): PathBase {\n const v = p.name\n // retain any other flags, but set ifmt from dirent\n p.#type = (p.#type & IFMT_UNKNOWN) | entToType(e)\n // case sensitivity fixing when we learn the true name.\n if (v !== e.name) p.name = e.name\n\n // just advance provisional index (potentially off the list),\n // otherwise we have to splice/pop it out and re-insert at head\n if (index !== c.provisional) {\n if (index === c.length - 1) c.pop()\n else c.splice(index, 1)\n c.unshift(p)\n }\n c.provisional++\n return p\n }\n\n /**\n * Call lstat() on this Path, and update all known information that can be\n * determined.\n *\n * Note that unlike `fs.lstat()`, the returned value does not contain some\n * information, such as `mode`, `dev`, `nlink`, and `ino`. If that\n * information is required, you will need to call `fs.lstat` yourself.\n *\n * If the Path refers to a nonexistent file, or if the lstat call fails for\n * any reason, `undefined` is returned. Otherwise the updated Path object is\n * returned.\n *\n * Results are cached, and thus may be out of date if the filesystem is\n * mutated.\n */\n async lstat(): Promise<PathBase | undefined> {\n if ((this.#type & ENOENT) === 0) {\n try {\n this.#applyStat(await this.#fs.promises.lstat(this.fullpath()))\n return this\n } catch (er) {\n this.#lstatFail((er as NodeJS.ErrnoException).code)\n }\n }\n }\n\n /**\n * synchronous {@link PathBase.lstat}\n */\n lstatSync(): PathBase | undefined {\n if ((this.#type & ENOENT) === 0) {\n try {\n this.#applyStat(this.#fs.lstatSync(this.fullpath()))\n return this\n } catch (er) {\n this.#lstatFail((er as NodeJS.ErrnoException).code)\n }\n }\n }\n\n #applyStat(st: Stats) {\n const {\n atime,\n atimeMs,\n birthtime,\n birthtimeMs,\n blksize,\n blocks,\n ctime,\n ctimeMs,\n dev,\n gid,\n ino,\n mode,\n mtime,\n mtimeMs,\n nlink,\n rdev,\n size,\n uid,\n } = st\n this.#atime = atime\n this.#atimeMs = atimeMs\n this.#birthtime = birthtime\n this.#birthtimeMs = birthtimeMs\n this.#blksize = blksize\n this.#blocks = blocks\n this.#ctime = ctime\n this.#ctimeMs = ctimeMs\n this.#dev = dev\n this.#gid = gid\n this.#ino = ino\n this.#mode = mode\n this.#mtime = mtime\n this.#mtimeMs = mtimeMs\n this.#nlink = nlink\n this.#rdev = rdev\n this.#size = size\n this.#uid = uid\n const ifmt = entToType(st)\n // retain any other flags, but set the ifmt\n this.#type = (this.#type & IFMT_UNKNOWN) | ifmt | LSTAT_CALLED\n if (ifmt !== UNKNOWN && ifmt !== IFDIR && ifmt !== IFLNK) {\n this.#type |= ENOTDIR\n }\n }\n\n #onReaddirCB: ((\n er: NodeJS.ErrnoException | null,\n entries: Path[],\n ) => any)[] = []\n #readdirCBInFlight: boolean = false\n #callOnReaddirCB(children: Path[]) {\n this.#readdirCBInFlight = false\n const cbs = this.#onReaddirCB.slice()\n this.#onReaddirCB.length = 0\n cbs.forEach(cb => cb(null, children))\n }\n\n /**\n * Standard node-style callback interface to get list of directory entries.\n *\n * If the Path cannot or does not contain any children, then an empty array\n * is returned.\n *\n * Results are cached, and thus may be out of date if the filesystem is\n * mutated.\n *\n * @param cb The callback called with (er, entries). Note that the `er`\n * param is somewhat extraneous, as all readdir() errors are handled and\n * simply result in an empty set of entries being returned.\n * @param allowZalgo Boolean indicating that immediately known results should\n * *not* be deferred with `queueMicrotask`. Defaults to `false`. Release\n * zalgo at your peril, the dark pony lord is devious and unforgiving.\n */\n readdirCB(\n cb: (er: NodeJS.ErrnoException | null, entries: PathBase[]) => any,\n allowZalgo: boolean = false,\n ): void {\n if (!this.canReaddir()) {\n if (allowZalgo) cb(null, [])\n else queueMicrotask(() => cb(null, []))\n return\n }\n\n const children = this.children()\n if (this.calledReaddir()) {\n const c = children.slice(0, children.provisional)\n if (allowZalgo) cb(null, c)\n else queueMicrotask(() => cb(null, c))\n return\n }\n\n // don't have to worry about zalgo at this point.\n this.#onReaddirCB.push(cb)\n if (this.#readdirCBInFlight) {\n return\n }\n this.#readdirCBInFlight = true\n\n // else read the directory, fill up children\n // de-provisionalize any provisional children.\n const fullpath = this.fullpath()\n this.#fs.readdir(fullpath, { withFileTypes: true }, (er, entries) => {\n if (er) {\n this.#readdirFail((er as NodeJS.ErrnoException).code)\n children.provisional = 0\n } else {\n // if we didn't get an error, we always get entries.\n //@ts-ignore\n for (const e of entries) {\n this.#readdirAddChild(e, children)\n }\n this.#readdirSuccess(children)\n }\n this.#callOnReaddirCB(children.slice(0, children.provisional))\n return\n })\n }\n\n #asyncReaddirInFlight?: Promise<void>\n\n /**\n * Return an array of known child entries.\n *\n * If the Path cannot or does not contain any children, then an empty array\n * is returned.\n *\n * Results are cached, and thus may be out of date if the filesystem is\n * mutated.\n */\n async readdir(): Promise<PathBase[]> {\n if (!this.canReaddir()) {\n return []\n }\n\n const children = this.children()\n if (this.calledReaddir()) {\n return children.slice(0, children.provisional)\n }\n\n // else read the directory, fill up children\n // de-provisionalize any provisional children.\n const fullpath = this.fullpath()\n if (this.#asyncReaddirInFlight) {\n await this.#asyncReaddirInFlight\n } else {\n /* c8 ignore start */\n let resolve: () => void = () => {}\n /* c8 ignore stop */\n this.#asyncReaddirInFlight = new Promise<void>(\n res => (resolve = res),\n )\n try {\n for (const e of await this.#fs.promises.readdir(fullpath, {\n withFileTypes: true,\n })) {\n this.#readdirAddChild(e, children)\n }\n this.#readdirSuccess(children)\n } catch (er) {\n this.#readdirFail((er as NodeJS.ErrnoException).code)\n children.provisional = 0\n }\n this.#asyncReaddirInFlight = undefined\n resolve()\n }\n return children.slice(0, children.provisional)\n }\n\n /**\n * synchronous {@link PathBase.readdir}\n */\n readdirSync(): PathBase[] {\n if (!this.canReaddir()) {\n return []\n }\n\n const children = this.children()\n if (this.calledReaddir()) {\n return children.slice(0, children.provisional)\n }\n\n // else read the directory, fill up children\n // de-provisionalize any provisional children.\n const fullpath = this.fullpath()\n try {\n for (const e of this.#fs.readdirSync(fullpath, {\n withFileTypes: true,\n })) {\n this.#readdirAddChild(e, children)\n }\n this.#readdirSuccess(children)\n } catch (er) {\n this.#readdirFail((er as NodeJS.ErrnoException).code)\n children.provisional = 0\n }\n return children.slice(0, children.provisional)\n }\n\n canReaddir() {\n if (this.#type & ENOCHILD) return false\n const ifmt = IFMT & this.#type\n // we always set ENOTDIR when setting IFMT, so should be impossible\n /* c8 ignore start */\n if (!(ifmt === UNKNOWN || ifmt === IFDIR || ifmt === IFLNK)) {\n return false\n }\n /* c8 ignore stop */\n return true\n }\n\n shouldWalk(\n dirs: Set<PathBase | undefined>,\n walkFilter?: (e: PathBase) => boolean,\n ): boolean {\n return (\n (this.#type & IFDIR) === IFDIR &&\n !(this.#type & ENOCHILD) &&\n !dirs.has(this) &&\n (!walkFilter || walkFilter(this))\n )\n }\n\n /**\n * Return the Path object corresponding to path as resolved\n * by realpath(3).\n *\n * If the realpath call fails for any reason, `undefined` is returned.\n *\n * Result is cached, and thus may be outdated if the filesystem is mutated.\n * On success, returns a Path object.\n */\n async realpath(): Promise<PathBase | undefined> {\n if (this.#realpath) return this.#realpath\n if ((ENOREALPATH | ENOREADLINK | ENOENT) & this.#type) return undefined\n try {\n const rp = await this.#fs.promises.realpath(this.fullpath())\n return (this.#realpath = this.resolve(rp))\n } catch (_) {\n this.#markENOREALPATH()\n }\n }\n\n /**\n * Synchronous {@link realpath}\n */\n realpathSync(): PathBase | undefined {\n if (this.#realpath) return this.#realpath\n if ((ENOREALPATH | ENOREADLINK | ENOENT) & this.#type) return undefined\n try {\n const rp = this.#fs.realpathSync(this.fullpath())\n return (this.#realpath = this.resolve(rp))\n } catch (_) {\n this.#markENOREALPATH()\n }\n }\n\n /**\n * Internal method to mark this Path object as the scurry cwd,\n * called by {@link PathScurry#chdir}\n *\n * @internal\n */\n [setAsCwd](oldCwd: PathBase): void {\n if (oldCwd === this) return\n oldCwd.isCWD = false\n this.isCWD = true\n\n const changed = new Set<PathBase>([])\n let rp = []\n let p: PathBase = this\n while (p && p.parent) {\n changed.add(p)\n p.#relative = rp.join(this.sep)\n p.#relativePosix = rp.join('/')\n p = p.parent\n rp.push('..')\n }\n // now un-memoize parents of old cwd\n p = oldCwd\n while (p && p.parent && !changed.has(p)) {\n p.#relative = undefined\n p.#relativePosix = undefined\n p = p.parent\n }\n }\n}\n\n/**\n * Path class used on win32 systems\n *\n * Uses `'\\\\'` as the path separator for returned paths, either `'\\\\'` or `'/'`\n * as the path separator for parsing paths.\n */\nexport class PathWin32 extends PathBase {\n /**\n * Separator for generating path strings.\n */\n sep: '\\\\' = '\\\\'\n /**\n * Separator for parsing path strings.\n */\n splitSep: RegExp = eitherSep\n\n /**\n * Do not create new Path objects directly. They should always be accessed\n * via the PathScurry class or other methods on the Path class.\n *\n * @internal\n */\n constructor(\n name: string,\n type: number = UNKNOWN,\n root: PathBase | undefined,\n roots: { [k: string]: PathBase },\n nocase: boolean,\n children: ChildrenCache,\n opts: PathOpts,\n ) {\n super(name, type, root, roots, nocase, children, opts)\n }\n\n /**\n * @internal\n */\n newChild(name: string, type: number = UNKNOWN, opts: PathOpts = {}) {\n return new PathWin32(\n name,\n type,\n this.root,\n this.roots,\n this.nocase,\n this.childrenCache(),\n opts,\n )\n }\n\n /**\n * @internal\n */\n getRootString(path: string): string {\n return win32.parse(path).root\n }\n\n /**\n * @internal\n */\n getRoot(rootPath: string): PathBase {\n rootPath = uncToDrive(rootPath.toUpperCase())\n if (rootPath === this.root.name) {\n return this.root\n }\n // ok, not that one, check if it matches another we know about\n for (const [compare, root] of Object.entries(this.roots)) {\n if (this.sameRoot(rootPath, compare)) {\n return (this.roots[rootPath] = root)\n }\n }\n // otherwise, have to create a new one.\n return (this.roots[rootPath] = new PathScurryWin32(\n rootPath,\n this,\n ).root)\n }\n\n /**\n * @internal\n */\n sameRoot(rootPath: string, compare: string = this.root.name): boolean {\n // windows can (rarely) have case-sensitive filesystem, but\n // UNC and drive letters are always case-insensitive, and canonically\n // represented uppercase.\n rootPath = rootPath\n .toUpperCase()\n .replace(/\\//g, '\\\\')\n .replace(uncDriveRegexp, '$1\\\\')\n return rootPath === compare\n }\n}\n\n/**\n * Path class used on all posix systems.\n *\n * Uses `'/'` as the path separator.\n */\nexport class PathPosix extends PathBase {\n /**\n * separator for parsing path strings\n */\n splitSep: '/' = '/'\n /**\n * separator for generating path strings\n */\n sep: '/' = '/'\n\n /**\n * Do not create new Path objects directly. They should always be accessed\n * via the PathScurry class or other methods on the Path class.\n *\n * @internal\n */\n constructor(\n name: string,\n type: number = UNKNOWN,\n root: PathBase | undefined,\n roots: { [k: string]: PathBase },\n nocase: boolean,\n children: ChildrenCache,\n opts: PathOpts,\n ) {\n super(name, type, root, roots, nocase, children, opts)\n }\n\n /**\n * @internal\n */\n getRootString(path: string): string {\n return path.startsWith('/') ? '/' : ''\n }\n\n /**\n * @internal\n */\n getRoot(_rootPath: string): PathBase {\n return this.root\n }\n\n /**\n * @internal\n */\n newChild(name: string, type: number = UNKNOWN, opts: PathOpts = {}) {\n return new PathPosix(\n name,\n type,\n this.root,\n this.roots,\n this.nocase,\n this.childrenCache(),\n opts,\n )\n }\n}\n\n/**\n * Options that may be provided to the PathScurry constructor\n */\nexport interface PathScurryOpts {\n /**\n * perform case-insensitive path matching. Default based on platform\n * subclass.\n */\n nocase?: boolean\n /**\n * Number of Path entries to keep in the cache of Path child references.\n *\n * Setting this higher than 65536 will dramatically increase the data\n * consumption and construction time overhead of each PathScurry.\n *\n * Setting this value to 256 or lower will significantly reduce the data\n * consumption and construction time overhead, but may also reduce resolve()\n * and readdir() performance on large filesystems.\n *\n * Default `16384`.\n */\n childrenCacheSize?: number\n /**\n * An object that overrides the built-in functions from the fs and\n * fs/promises modules.\n *\n * See {@link FSOption}\n */\n fs?: FSOption\n}\n\n/**\n * The base class for all PathScurry classes, providing the interface for path\n * resolution and filesystem operations.\n *\n * Typically, you should *not* instantiate this class directly, but rather one\n * of the platform-specific classes, or the exported {@link PathScurry} which\n * defaults to the current platform.\n */\nexport abstract class PathScurryBase {\n /**\n * The root Path entry for the current working directory of this Scurry\n */\n root: PathBase\n /**\n * The string path for the root of this Scurry's current working directory\n */\n rootPath: string\n /**\n * A collection of all roots encountered, referenced by rootPath\n */\n roots: { [k: string]: PathBase }\n /**\n * The Path entry corresponding to this PathScurry's current working directory.\n */\n cwd: PathBase\n #resolveCache: ResolveCache\n #resolvePosixCache: ResolveCache\n #children: ChildrenCache\n /**\n * Perform path comparisons case-insensitively.\n *\n * Defaults true on Darwin and Windows systems, false elsewhere.\n */\n nocase: boolean\n\n /**\n * The path separator used for parsing paths\n *\n * `'/'` on Posix systems, either `'/'` or `'\\\\'` on Windows\n */\n abstract sep: string | RegExp\n\n #fs: FSValue\n\n /**\n * This class should not be instantiated directly.\n *\n * Use PathScurryWin32, PathScurryDarwin, PathScurryPosix, or PathScurry\n *\n * @internal\n */\n constructor(\n cwd: URL | string = process.cwd(),\n pathImpl: typeof win32 | typeof posix,\n sep: string | RegExp,\n {\n nocase,\n childrenCacheSize = 16 * 1024,\n fs = defaultFS,\n }: PathScurryOpts = {},\n ) {\n this.#fs = fsFromOption(fs)\n if (cwd instanceof URL || cwd.startsWith('file://')) {\n cwd = fileURLToPath(cwd)\n }\n // resolve and split root, and then add to the store.\n // this is the only time we call path.resolve()\n const cwdPath = pathImpl.resolve(cwd)\n this.roots = Object.create(null)\n this.rootPath = this.parseRootPath(cwdPath)\n this.#resolveCache = new ResolveCache()\n this.#resolvePosixCache = new ResolveCache()\n this.#children = new ChildrenCache(childrenCacheSize)\n\n const split = cwdPath.substring(this.rootPath.length).split(sep)\n // resolve('/') leaves '', splits to [''], we don't want that.\n if (split.length === 1 && !split[0]) {\n split.pop()\n }\n /* c8 ignore start */\n if (nocase === undefined) {\n throw new TypeError(\n 'must provide nocase setting to PathScurryBase ctor',\n )\n }\n /* c8 ignore stop */\n this.nocase = nocase\n this.root = this.newRoot(this.#fs)\n this.roots[this.rootPath] = this.root\n let prev: PathBase = this.root\n let len = split.length - 1\n const joinSep = pathImpl.sep\n let abs = this.rootPath\n let sawFirst = false\n for (const part of split) {\n const l = len--\n prev = prev.child(part, {\n relative: new Array(l).fill('..').join(joinSep),\n relativePosix: new Array(l).fill('..').join('/'),\n fullpath: (abs += (sawFirst ? '' : joinSep) + part),\n })\n sawFirst = true\n }\n this.cwd = prev\n }\n\n /**\n * Get the depth of a provided path, string, or the cwd\n */\n depth(path: Path | string = this.cwd): number {\n if (typeof path === 'string') {\n path = this.cwd.resolve(path)\n }\n return path.depth()\n }\n\n /**\n * Parse the root portion of a path string\n *\n * @internal\n */\n abstract parseRootPath(dir: string): string\n /**\n * create a new Path to use as root during construction.\n *\n * @internal\n */\n abstract newRoot(fs: FSValue): PathBase\n /**\n * Determine whether a given path string is absolute\n */\n abstract isAbsolute(p: string): boolean\n\n /**\n * Return the cache of child entries. Exposed so subclasses can create\n * child Path objects in a platform-specific way.\n *\n * @internal\n */\n childrenCache() {\n return this.#children\n }\n\n /**\n * Resolve one or more path strings to a resolved string\n *\n * Same interface as require('path').resolve.\n *\n * Much faster than path.resolve() when called multiple times for the same\n * path, because the resolved Path objects are cached. Much slower\n * otherwise.\n */\n resolve(...paths: string[]): string {\n // first figure out the minimum number of paths we have to test\n // we always start at cwd, but any absolutes will bump the start\n let r = ''\n for (let i = paths.length - 1; i >= 0; i--) {\n const p = paths[i]\n if (!p || p === '.') continue\n r = r ? `${p}/${r}` : p\n if (this.isAbsolute(p)) {\n break\n }\n }\n const cached = this.#resolveCache.get(r)\n if (cached !== undefined) {\n return cached\n }\n const result = this.cwd.resolve(r).fullpath()\n this.#resolveCache.set(r, result)\n return result\n }\n\n /**\n * Resolve one or more path strings to a resolved string, returning\n * the posix path. Identical to .resolve() on posix systems, but on\n * windows will return a forward-slash separated UNC path.\n *\n * Same interface as require('path').resolve.\n *\n * Much faster than path.resolve() when called multiple times for the same\n * path, because the resolved Path objects are cached. Much slower\n * otherwise.\n */\n resolvePosix(...paths: string[]): string {\n // first figure out the minimum number of paths we have to test\n // we always start at cwd, but any absolutes will bump the start\n let r = ''\n for (let i = paths.length - 1; i >= 0; i--) {\n const p = paths[i]\n if (!p || p === '.') continue\n r = r ? `${p}/${r}` : p\n if (this.isAbsolute(p)) {\n break\n }\n }\n const cached = this.#resolvePosixCache.get(r)\n if (cached !== undefined) {\n return cached\n }\n const result = this.cwd.resolve(r).fullpathPosix()\n this.#resolvePosixCache.set(r, result)\n return result\n }\n\n /**\n * find the relative path from the cwd to the supplied path string or entry\n */\n relative(entry: PathBase | string = this.cwd): string {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n }\n return entry.relative()\n }\n\n /**\n * find the relative path from the cwd to the supplied path string or\n * entry, using / as the path delimiter, even on Windows.\n */\n relativePosix(entry: PathBase | string = this.cwd): string {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n }\n return entry.relativePosix()\n }\n\n /**\n * Return the basename for the provided string or Path object\n */\n basename(entry: PathBase | string = this.cwd): string {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n }\n return entry.name\n }\n\n /**\n * Return the dirname for the provided string or Path object\n */\n dirname(entry: PathBase | string = this.cwd): string {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n }\n return (entry.parent || entry).fullpath()\n }\n\n /**\n * Return an array of known child entries.\n *\n * First argument may be either a string, or a Path object.\n *\n * If the Path cannot or does not contain any children, then an empty array\n * is returned.\n *\n * Results are cached, and thus may be out of date if the filesystem is\n * mutated.\n *\n * Unlike `fs.readdir()`, the `withFileTypes` option defaults to `true`. Set\n * `{ withFileTypes: false }` to return strings.\n */\n\n readdir(): Promise<PathBase[]>\n readdir(opts: { withFileTypes: true }): Promise<PathBase[]>\n readdir(opts: { withFileTypes: false }): Promise<string[]>\n readdir(opts: { withFileTypes: boolean }): Promise<PathBase[] | string[]>\n readdir(entry: PathBase | string): Promise<PathBase[]>\n readdir(\n entry: PathBase | string,\n opts: { withFileTypes: true },\n ): Promise<PathBase[]>\n readdir(\n entry: PathBase | string,\n opts: { withFileTypes: false },\n ): Promise<string[]>\n readdir(\n entry: PathBase | string,\n opts: { withFileTypes: boolean },\n ): Promise<PathBase[] | string[]>\n async readdir(\n entry: PathBase | string | { withFileTypes: boolean } = this.cwd,\n opts: { withFileTypes: boolean } = {\n withFileTypes: true,\n },\n ): Promise<PathBase[] | string[]> {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n } else if (!(entry instanceof PathBase)) {\n opts = entry\n entry = this.cwd\n }\n const { withFileTypes } = opts\n if (!entry.canReaddir()) {\n return []\n } else {\n const p = await entry.readdir()\n return withFileTypes ? p : p.map(e => e.name)\n }\n }\n\n /**\n * synchronous {@link PathScurryBase.readdir}\n */\n readdirSync(): PathBase[]\n readdirSync(opts: { withFileTypes: true }): PathBase[]\n readdirSync(opts: { withFileTypes: false }): string[]\n readdirSync(opts: { withFileTypes: boolean }): PathBase[] | string[]\n readdirSync(entry: PathBase | string): PathBase[]\n readdirSync(\n entry: PathBase | string,\n opts: { withFileTypes: true },\n ): PathBase[]\n readdirSync(\n entry: PathBase | string,\n opts: { withFileTypes: false },\n ): string[]\n readdirSync(\n entry: PathBase | string,\n opts: { withFileTypes: boolean },\n ): PathBase[] | string[]\n readdirSync(\n entry: PathBase | string | { withFileTypes: boolean } = this.cwd,\n opts: { withFileTypes: boolean } = {\n withFileTypes: true,\n },\n ): PathBase[] | string[] {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n } else if (!(entry instanceof PathBase)) {\n opts = entry\n entry = this.cwd\n }\n const { withFileTypes = true } = opts\n if (!entry.canReaddir()) {\n return []\n } else if (withFileTypes) {\n return entry.readdirSync()\n } else {\n return entry.readdirSync().map(e => e.name)\n }\n }\n\n /**\n * Call lstat() on the string or Path object, and update all known\n * information that can be determined.\n *\n * Note that unlike `fs.lstat()`, the returned value does not contain some\n * information, such as `mode`, `dev`, `nlink`, and `ino`. If that\n * information is required, you will need to call `fs.lstat` yourself.\n *\n * If the Path refers to a nonexistent file, or if the lstat call fails for\n * any reason, `undefined` is returned. Otherwise the updated Path object is\n * returned.\n *\n * Results are cached, and thus may be out of date if the filesystem is\n * mutated.\n */\n async lstat(\n entry: string | PathBase = this.cwd,\n ): Promise<PathBase | undefined> {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n }\n return entry.lstat()\n }\n\n /**\n * synchronous {@link PathScurryBase.lstat}\n */\n lstatSync(entry: string | PathBase = this.cwd): PathBase | undefined {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n }\n return entry.lstatSync()\n }\n\n /**\n * Return the Path object or string path corresponding to the target of a\n * symbolic link.\n *\n * If the path is not a symbolic link, or if the readlink call fails for any\n * reason, `undefined` is returned.\n *\n * Result is cached, and thus may be outdated if the filesystem is mutated.\n *\n * `{withFileTypes}` option defaults to `false`.\n *\n * On success, returns a Path object if `withFileTypes` option is true,\n * otherwise a string.\n */\n readlink(): Promise<string | undefined>\n readlink(opt: { withFileTypes: false }): Promise<string | undefined>\n readlink(opt: { withFileTypes: true }): Promise<PathBase | undefined>\n readlink(opt: {\n withFileTypes: boolean\n }): Promise<PathBase | string | undefined>\n readlink(\n entry: string | PathBase,\n opt?: { withFileTypes: false },\n ): Promise<string | undefined>\n readlink(\n entry: string | PathBase,\n opt: { withFileTypes: true },\n ): Promise<PathBase | undefined>\n readlink(\n entry: string | PathBase,\n opt: { withFileTypes: boolean },\n ): Promise<string | PathBase | undefined>\n async readlink(\n entry: string | PathBase | { withFileTypes: boolean } = this.cwd,\n { withFileTypes }: { withFileTypes: boolean } = {\n withFileTypes: false,\n },\n ): Promise<string | PathBase | undefined> {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n } else if (!(entry instanceof PathBase)) {\n withFileTypes = entry.withFileTypes\n entry = this.cwd\n }\n const e = await entry.readlink()\n return withFileTypes ? e : e?.fullpath()\n }\n\n /**\n * synchronous {@link PathScurryBase.readlink}\n */\n readlinkSync(): string | undefined\n readlinkSync(opt: { withFileTypes: false }): string | undefined\n readlinkSync(opt: { withFileTypes: true }): PathBase | undefined\n readlinkSync(opt: {\n withFileTypes: boolean\n }): PathBase | string | undefined\n readlinkSync(\n entry: string | PathBase,\n opt?: { withFileTypes: false },\n ): string | undefined\n readlinkSync(\n entry: string | PathBase,\n opt: { withFileTypes: true },\n ): PathBase | undefined\n readlinkSync(\n entry: string | PathBase,\n opt: { withFileTypes: boolean },\n ): string | PathBase | undefined\n readlinkSync(\n entry: string | PathBase | { withFileTypes: boolean } = this.cwd,\n { withFileTypes }: { withFileTypes: boolean } = {\n withFileTypes: false,\n },\n ): string | PathBase | undefined {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n } else if (!(entry instanceof PathBase)) {\n withFileTypes = entry.withFileTypes\n entry = this.cwd\n }\n const e = entry.readlinkSync()\n return withFileTypes ? e : e?.fullpath()\n }\n\n /**\n * Return the Path object or string path corresponding to path as resolved\n * by realpath(3).\n *\n * If the realpath call fails for any reason, `undefined` is returned.\n *\n * Result is cached, and thus may be outdated if the filesystem is mutated.\n *\n * `{withFileTypes}` option defaults to `false`.\n *\n * On success, returns a Path object if `withFileTypes` option is true,\n * otherwise a string.\n */\n realpath(): Promise<string | undefined>\n realpath(opt: { withFileTypes: false }): Promise<string | undefined>\n realpath(opt: { withFileTypes: true }): Promise<PathBase | undefined>\n realpath(opt: {\n withFileTypes: boolean\n }): Promise<PathBase | string | undefined>\n realpath(\n entry: string | PathBase,\n opt?: { withFileTypes: false },\n ): Promise<string | undefined>\n realpath(\n entry: string | PathBase,\n opt: { withFileTypes: true },\n ): Promise<PathBase | undefined>\n realpath(\n entry: string | PathBase,\n opt: { withFileTypes: boolean },\n ): Promise<string | PathBase | undefined>\n async realpath(\n entry: string | PathBase | { withFileTypes: boolean } = this.cwd,\n { withFileTypes }: { withFileTypes: boolean } = {\n withFileTypes: false,\n },\n ): Promise<string | PathBase | undefined> {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n } else if (!(entry instanceof PathBase)) {\n withFileTypes = entry.withFileTypes\n entry = this.cwd\n }\n const e = await entry.realpath()\n return withFileTypes ? e : e?.fullpath()\n }\n\n realpathSync(): string | undefined\n realpathSync(opt: { withFileTypes: false }): string | undefined\n realpathSync(opt: { withFileTypes: true }): PathBase | undefined\n realpathSync(opt: {\n withFileTypes: boolean\n }): PathBase | string | undefined\n realpathSync(\n entry: string | PathBase,\n opt?: { withFileTypes: false },\n ): string | undefined\n realpathSync(\n entry: string | PathBase,\n opt: { withFileTypes: true },\n ): PathBase | undefined\n realpathSync(\n entry: string | PathBase,\n opt: { withFileTypes: boolean },\n ): string | PathBase | undefined\n realpathSync(\n entry: string | PathBase | { withFileTypes: boolean } = this.cwd,\n { withFileTypes }: { withFileTypes: boolean } = {\n withFileTypes: false,\n },\n ): string | PathBase | undefined {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n } else if (!(entry instanceof PathBase)) {\n withFileTypes = entry.withFileTypes\n entry = this.cwd\n }\n const e = entry.realpathSync()\n return withFileTypes ? e : e?.fullpath()\n }\n\n /**\n * Asynchronously walk the directory tree, returning an array of\n * all path strings or Path objects found.\n *\n * Note that this will be extremely memory-hungry on large filesystems.\n * In such cases, it may be better to use the stream or async iterator\n * walk implementation.\n */\n walk(): Promise<PathBase[]>\n walk(\n opts: WalkOptionsWithFileTypesTrue | WalkOptionsWithFileTypesUnset,\n ): Promise<PathBase[]>\n walk(opts: WalkOptionsWithFileTypesFalse): Promise<string[]>\n walk(opts: WalkOptions): Promise<string[] | PathBase[]>\n walk(entry: string | PathBase): Promise<PathBase[]>\n walk(\n entry: string | PathBase,\n opts: WalkOptionsWithFileTypesTrue | WalkOptionsWithFileTypesUnset,\n ): Promise<PathBase[]>\n walk(\n entry: string | PathBase,\n opts: WalkOptionsWithFileTypesFalse,\n ): Promise<string[]>\n walk(\n entry: string | PathBase,\n opts: WalkOptions,\n ): Promise<PathBase[] | string[]>\n async walk(\n entry: string | PathBase | WalkOptions = this.cwd,\n opts: WalkOptions = {},\n ): Promise<PathBase[] | string[]> {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n } else if (!(entry instanceof PathBase)) {\n opts = entry\n entry = this.cwd\n }\n const {\n withFileTypes = true,\n follow = false,\n filter,\n walkFilter,\n } = opts\n const results: (string | PathBase)[] = []\n if (!filter || filter(entry)) {\n results.push(withFileTypes ? entry : entry.fullpath())\n }\n const dirs = new Set<PathBase>()\n const walk = (\n dir: PathBase,\n cb: (er?: NodeJS.ErrnoException) => void,\n ) => {\n dirs.add(dir)\n dir.readdirCB((er, entries) => {\n /* c8 ignore start */\n if (er) {\n return cb(er)\n }\n /* c8 ignore stop */\n let len = entries.length\n if (!len) return cb()\n const next = () => {\n if (--len === 0) {\n cb()\n }\n }\n for (const e of entries) {\n if (!filter || filter(e)) {\n results.push(withFileTypes ? e : e.fullpath())\n }\n if (follow && e.isSymbolicLink()) {\n e.realpath()\n .then(r => (r?.isUnknown() ? r.lstat() : r))\n .then(r =>\n r?.shouldWalk(dirs, walkFilter) ? walk(r, next) : next(),\n )\n } else {\n if (e.shouldWalk(dirs, walkFilter)) {\n walk(e, next)\n } else {\n next()\n }\n }\n }\n }, true) // zalgooooooo\n }\n\n const start = entry\n return new Promise<PathBase[] | string[]>((res, rej) => {\n walk(start, er => {\n /* c8 ignore start */\n if (er) return rej(er)\n /* c8 ignore stop */\n res(results as PathBase[] | string[])\n })\n })\n }\n\n /**\n * Synchronously walk the directory tree, returning an array of\n * all path strings or Path objects found.\n *\n * Note that this will be extremely memory-hungry on large filesystems.\n * In such cases, it may be better to use the stream or async iterator\n * walk implementation.\n */\n walkSync(): PathBase[]\n walkSync(\n opts: WalkOptionsWithFileTypesTrue | WalkOptionsWithFileTypesUnset,\n ): PathBase[]\n walkSync(opts: WalkOptionsWithFileTypesFalse): string[]\n walkSync(opts: WalkOptions): string[] | PathBase[]\n walkSync(entry: string | PathBase): PathBase[]\n walkSync(\n entry: string | PathBase,\n opts: WalkOptionsWithFileTypesUnset | WalkOptionsWithFileTypesTrue,\n ): PathBase[]\n walkSync(\n entry: string | PathBase,\n opts: WalkOptionsWithFileTypesFalse,\n ): string[]\n walkSync(\n entry: string | PathBase,\n opts: WalkOptions,\n ): PathBase[] | string[]\n walkSync(\n entry: string | PathBase | WalkOptions = this.cwd,\n opts: WalkOptions = {},\n ): PathBase[] | string[] {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n } else if (!(entry instanceof PathBase)) {\n opts = entry\n entry = this.cwd\n }\n const {\n withFileTypes = true,\n follow = false,\n filter,\n walkFilter,\n } = opts\n const results: (string | PathBase)[] = []\n if (!filter || filter(entry)) {\n results.push(withFileTypes ? entry : entry.fullpath())\n }\n const dirs = new Set<PathBase>([entry])\n for (const dir of dirs) {\n const entries = dir.readdirSync()\n for (const e of entries) {\n if (!filter || filter(e)) {\n results.push(withFileTypes ? e : e.fullpath())\n }\n let r: PathBase | undefined = e\n if (e.isSymbolicLink()) {\n if (!(follow && (r = e.realpathSync()))) continue\n if (r.isUnknown()) r.lstatSync()\n }\n if (r.shouldWalk(dirs, walkFilter)) {\n dirs.add(r)\n }\n }\n }\n return results as string[] | PathBase[]\n }\n\n /**\n * Support for `for await`\n *\n * Alias for {@link PathScurryBase.iterate}\n *\n * Note: As of Node 19, this is very slow, compared to other methods of\n * walking. Consider using {@link PathScurryBase.stream} if memory overhead\n * and backpressure are concerns, or {@link PathScurryBase.walk} if not.\n */\n [Symbol.asyncIterator]() {\n return this.iterate()\n }\n\n /**\n * Async generator form of {@link PathScurryBase.walk}\n *\n * Note: As of Node 19, this is very slow, compared to other methods of\n * walking, especially if most/all of the directory tree has been previously\n * walked. Consider using {@link PathScurryBase.stream} if memory overhead\n * and backpressure are concerns, or {@link PathScurryBase.walk} if not.\n */\n iterate(): AsyncGenerator<PathBase, void, void>\n iterate(\n opts: WalkOptionsWithFileTypesTrue | WalkOptionsWithFileTypesUnset,\n ): AsyncGenerator<PathBase, void, void>\n iterate(\n opts: WalkOptionsWithFileTypesFalse,\n ): AsyncGenerator<string, void, void>\n iterate(opts: WalkOptions): AsyncGenerator<string | PathBase, void, void>\n iterate(entry: string | PathBase): AsyncGenerator<PathBase, void, void>\n iterate(\n entry: string | PathBase,\n opts: WalkOptionsWithFileTypesTrue | WalkOptionsWithFileTypesUnset,\n ): AsyncGenerator<PathBase, void, void>\n iterate(\n entry: string | PathBase,\n opts: WalkOptionsWithFileTypesFalse,\n ): AsyncGenerator<string, void, void>\n iterate(\n entry: string | PathBase,\n opts: WalkOptions,\n ): AsyncGenerator<PathBase | string, void, void>\n iterate(\n entry: string | PathBase | WalkOptions = this.cwd,\n options: WalkOptions = {},\n ): AsyncGenerator<PathBase | string, void, void> {\n // iterating async over the stream is significantly more performant,\n // especially in the warm-cache scenario, because it buffers up directory\n // entries in the background instead of waiting for a yield for each one.\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n } else if (!(entry instanceof PathBase)) {\n options = entry\n entry = this.cwd\n }\n return this.stream(entry, options)[Symbol.asyncIterator]()\n }\n\n /**\n * Iterating over a PathScurry performs a synchronous walk.\n *\n * Alias for {@link PathScurryBase.iterateSync}\n */\n [Symbol.iterator]() {\n return this.iterateSync()\n }\n\n iterateSync(): Generator<PathBase, void, void>\n iterateSync(\n opts: WalkOptionsWithFileTypesTrue | WalkOptionsWithFileTypesUnset,\n ): Generator<PathBase, void, void>\n iterateSync(\n opts: WalkOptionsWithFileTypesFalse,\n ): Generator<string, void, void>\n iterateSync(opts: WalkOptions): Generator<string | PathBase, void, void>\n iterateSync(entry: string | PathBase): Generator<PathBase, void, void>\n iterateSync(\n entry: string | PathBase,\n opts: WalkOptionsWithFileTypesTrue | WalkOptionsWithFileTypesUnset,\n ): Generator<PathBase, void, void>\n iterateSync(\n entry: string | PathBase,\n opts: WalkOptionsWithFileTypesFalse,\n ): Generator<string, void, void>\n iterateSync(\n entry: string | PathBase,\n opts: WalkOptions,\n ): Generator<PathBase | string, void, void>\n *iterateSync(\n entry: string | PathBase | WalkOptions = this.cwd,\n opts: WalkOptions = {},\n ): Generator<PathBase | string, void, void> {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n } else if (!(entry instanceof PathBase)) {\n opts = entry\n entry = this.cwd\n }\n const {\n withFileTypes = true,\n follow = false,\n filter,\n walkFilter,\n } = opts\n if (!filter || filter(entry)) {\n yield withFileTypes ? entry : entry.fullpath()\n }\n const dirs = new Set<PathBase>([entry])\n for (const dir of dirs) {\n const entries = dir.readdirSync()\n for (const e of entries) {\n if (!filter || filter(e)) {\n yield withFileTypes ? e : e.fullpath()\n }\n let r: PathBase | undefined = e\n if (e.isSymbolicLink()) {\n if (!(follow && (r = e.realpathSync()))) continue\n if (r.isUnknown()) r.lstatSync()\n }\n if (r.shouldWalk(dirs, walkFilter)) {\n dirs.add(r)\n }\n }\n }\n }\n\n /**\n * Stream form of {@link PathScurryBase.walk}\n *\n * Returns a Minipass stream that emits {@link PathBase} objects by default,\n * or strings if `{ withFileTypes: false }` is set in the options.\n */\n stream(): Minipass<PathBase>\n stream(\n opts: WalkOptionsWithFileTypesTrue | WalkOptionsWithFileTypesUnset,\n ): Minipass<PathBase>\n stream(opts: WalkOptionsWithFileTypesFalse): Minipass<string>\n stream(opts: WalkOptions): Minipass<string | PathBase>\n stream(entry: string | PathBase): Minipass<PathBase>\n stream(\n entry: string | PathBase,\n opts: WalkOptionsWithFileTypesUnset | WalkOptionsWithFileTypesTrue,\n ): Minipass<PathBase>\n stream(\n entry: string | PathBase,\n opts: WalkOptionsWithFileTypesFalse,\n ): Minipass<string>\n stream(\n entry: string | PathBase,\n opts: WalkOptions,\n ): Minipass<string> | Minipass<PathBase>\n stream(\n entry: string | PathBase | WalkOptions = this.cwd,\n opts: WalkOptions = {},\n ): Minipass<string> | Minipass<PathBase> {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n } else if (!(entry instanceof PathBase)) {\n opts = entry\n entry = this.cwd\n }\n const {\n withFileTypes = true,\n follow = false,\n filter,\n walkFilter,\n } = opts\n const results = new Minipass<string | PathBase>({ objectMode: true })\n if (!filter || filter(entry)) {\n results.write(withFileTypes ? entry : entry.fullpath())\n }\n const dirs = new Set<PathBase>()\n const queue: PathBase[] = [entry]\n let processing = 0\n const process = () => {\n let paused = false\n while (!paused) {\n const dir = queue.shift()\n if (!dir) {\n if (processing === 0) results.end()\n return\n }\n\n processing++\n dirs.add(dir)\n\n const onReaddir = (\n er: null | NodeJS.ErrnoException,\n entries: PathBase[],\n didRealpaths: boolean = false,\n ) => {\n /* c8 ignore start */\n if (er) return results.emit('error', er)\n /* c8 ignore stop */\n if (follow && !didRealpaths) {\n const promises: Promise<PathBase | undefined>[] = []\n for (const e of entries) {\n if (e.isSymbolicLink()) {\n promises.push(\n e\n .realpath()\n .then((r: PathBase | undefined) =>\n r?.isUnknown() ? r.lstat() : r,\n ),\n )\n }\n }\n if (promises.length) {\n Promise.all(promises).then(() =>\n onReaddir(null, entries, true),\n )\n return\n }\n }\n\n for (const e of entries) {\n if (e && (!filter || filter(e))) {\n if (!results.write(withFileTypes ? e : e.fullpath())) {\n paused = true\n }\n }\n }\n\n processing--\n for (const e of entries) {\n const r = e.realpathCached() || e\n if (r.shouldWalk(dirs, walkFilter)) {\n queue.push(r)\n }\n }\n if (paused && !results.flowing) {\n results.once('drain', process)\n } else if (!sync) {\n process()\n }\n }\n\n // zalgo containment\n let sync = true\n dir.readdirCB(onReaddir, true)\n sync = false\n }\n }\n process()\n return results as Minipass<string> | Minipass<PathBase>\n }\n\n /**\n * Synchronous form of {@link PathScurryBase.stream}\n *\n * Returns a Minipass stream that emits {@link PathBase} objects by default,\n * or strings if `{ withFileTypes: false }` is set in the options.\n *\n * Will complete the walk in a single tick if the stream is consumed fully.\n * Otherwise, will pause as needed for stream backpressure.\n */\n streamSync(): Minipass<PathBase>\n streamSync(\n opts: WalkOptionsWithFileTypesTrue | WalkOptionsWithFileTypesUnset,\n ): Minipass<PathBase>\n streamSync(opts: WalkOptionsWithFileTypesFalse): Minipass<string>\n streamSync(opts: WalkOptions): Minipass<string | PathBase>\n streamSync(entry: string | PathBase): Minipass<PathBase>\n streamSync(\n entry: string | PathBase,\n opts: WalkOptionsWithFileTypesUnset | WalkOptionsWithFileTypesTrue,\n ): Minipass<PathBase>\n streamSync(\n entry: string | PathBase,\n opts: WalkOptionsWithFileTypesFalse,\n ): Minipass<string>\n streamSync(\n entry: string | PathBase,\n opts: WalkOptions,\n ): Minipass<string> | Minipass<PathBase>\n streamSync(\n entry: string | PathBase | WalkOptions = this.cwd,\n opts: WalkOptions = {},\n ): Minipass<string> | Minipass<PathBase> {\n if (typeof entry === 'string') {\n entry = this.cwd.resolve(entry)\n } else if (!(entry instanceof PathBase)) {\n opts = entry\n entry = this.cwd\n }\n const {\n withFileTypes = true,\n follow = false,\n filter,\n walkFilter,\n } = opts\n const results = new Minipass<string | PathBase>({ objectMode: true })\n const dirs = new Set<PathBase>()\n if (!filter || filter(entry)) {\n results.write(withFileTypes ? entry : entry.fullpath())\n }\n const queue: PathBase[] = [entry]\n let processing = 0\n const process = () => {\n let paused = false\n while (!paused) {\n const dir = queue.shift()\n if (!dir) {\n if (processing === 0) results.end()\n return\n }\n processing++\n dirs.add(dir)\n\n const entries = dir.readdirSync()\n for (const e of entries) {\n if (!filter || filter(e)) {\n if (!results.write(withFileTypes ? e : e.fullpath())) {\n paused = true\n }\n }\n }\n processing--\n for (const e of entries) {\n let r: PathBase | undefined = e\n if (e.isSymbolicLink()) {\n if (!(follow && (r = e.realpathSync()))) continue\n if (r.isUnknown()) r.lstatSync()\n }\n if (r.shouldWalk(dirs, walkFilter)) {\n queue.push(r)\n }\n }\n }\n if (paused && !results.flowing) results.once('drain', process)\n }\n process()\n return results as Minipass<string> | Minipass<PathBase>\n }\n\n chdir(path: string | Path = this.cwd) {\n const oldCwd = this.cwd\n this.cwd = typeof path === 'string' ? this.cwd.resolve(path) : path\n this.cwd[setAsCwd](oldCwd)\n }\n}\n\n/**\n * Options provided to all walk methods.\n */\nexport interface WalkOptions {\n /**\n * Return results as {@link PathBase} objects rather than strings.\n * When set to false, results are fully resolved paths, as returned by\n * {@link PathBase.fullpath}.\n * @default true\n */\n withFileTypes?: boolean\n\n /**\n * Attempt to read directory entries from symbolic links. Otherwise, only\n * actual directories are traversed. Regardless of this setting, a given\n * target path will only ever be walked once, meaning that a symbolic link\n * to a previously traversed directory will never be followed.\n *\n * Setting this imposes a slight performance penalty, because `readlink`\n * must be called on all symbolic links encountered, in order to avoid\n * infinite cycles.\n * @default false\n */\n follow?: boolean\n\n /**\n * Only return entries where the provided function returns true.\n *\n * This will not prevent directories from being traversed, even if they do\n * not pass the filter, though it will prevent directories themselves from\n * being included in the result set. See {@link walkFilter}\n *\n * Asynchronous functions are not supported here.\n *\n * By default, if no filter is provided, all entries and traversed\n * directories are included.\n */\n filter?: (entry: PathBase) => boolean\n\n /**\n * Only traverse directories (and in the case of {@link follow} being set to\n * true, symbolic links to directories) if the provided function returns\n * true.\n *\n * This will not prevent directories from being included in the result set,\n * even if they do not pass the supplied filter function. See {@link filter}\n * to do that.\n *\n * Asynchronous functions are not supported here.\n */\n walkFilter?: (entry: PathBase) => boolean\n}\n\nexport type WalkOptionsWithFileTypesUnset = WalkOptions & {\n withFileTypes?: undefined\n}\nexport type WalkOptionsWithFileTypesTrue = WalkOptions & {\n withFileTypes: true\n}\nexport type WalkOptionsWithFileTypesFalse = WalkOptions & {\n withFileTypes: false\n}\n\n/**\n * Windows implementation of {@link PathScurryBase}\n *\n * Defaults to case insensitve, uses `'\\\\'` to generate path strings. Uses\n * {@link PathWin32} for Path objects.\n */\nexport class PathScurryWin32 extends PathScurryBase {\n /**\n * separator for generating path strings\n */\n sep: '\\\\' = '\\\\'\n\n constructor(\n cwd: URL | string = process.cwd(),\n opts: PathScurryOpts = {},\n ) {\n const { nocase = true } = opts\n super(cwd, win32, '\\\\', { ...opts, nocase })\n this.nocase = nocase\n for (let p: PathBase | undefined = this.cwd; p; p = p.parent) {\n p.nocase = this.nocase\n }\n }\n\n /**\n * @internal\n */\n parseRootPath(dir: string): string {\n // if the path starts with a single separator, it's not a UNC, and we'll\n // just get separator as the root, and driveFromUNC will return \\\n // In that case, mount \\ on the root from the cwd.\n return win32.parse(dir).root.toUpperCase()\n }\n\n /**\n * @internal\n */\n newRoot(fs: FSValue) {\n return new PathWin32(\n this.rootPath,\n IFDIR,\n undefined,\n this.roots,\n this.nocase,\n this.childrenCache(),\n { fs },\n )\n }\n\n /**\n * Return true if the provided path string is an absolute path\n */\n isAbsolute(p: string): boolean {\n return (\n p.startsWith('/') || p.startsWith('\\\\') || /^[a-z]:(\\/|\\\\)/i.test(p)\n )\n }\n}\n\n/**\n * {@link PathScurryBase} implementation for all posix systems other than Darwin.\n *\n * Defaults to case-sensitive matching, uses `'/'` to generate path strings.\n *\n * Uses {@link PathPosix} for Path objects.\n */\nexport class PathScurryPosix extends PathScurryBase {\n /**\n * separator for generating path strings\n */\n sep: '/' = '/'\n constructor(\n cwd: URL | string = process.cwd(),\n opts: PathScurryOpts = {},\n ) {\n const { nocase = false } = opts\n super(cwd, posix, '/', { ...opts, nocase })\n this.nocase = nocase\n }\n\n /**\n * @internal\n */\n parseRootPath(_dir: string): string {\n return '/'\n }\n\n /**\n * @internal\n */\n newRoot(fs: FSValue) {\n return new PathPosix(\n this.rootPath,\n IFDIR,\n undefined,\n this.roots,\n this.nocase,\n this.childrenCache(),\n { fs },\n )\n }\n\n /**\n * Return true if the provided path string is an absolute path\n */\n isAbsolute(p: string): boolean {\n return p.startsWith('/')\n }\n}\n\n/**\n * {@link PathScurryBase} implementation for Darwin (macOS) systems.\n *\n * Defaults to case-insensitive matching, uses `'/'` for generating path\n * strings.\n *\n * Uses {@link PathPosix} for Path objects.\n */\nexport class PathScurryDarwin extends PathScurryPosix {\n constructor(\n cwd: URL | string = process.cwd(),\n opts: PathScurryOpts = {},\n ) {\n const { nocase = true } = opts\n super(cwd, { ...opts, nocase })\n }\n}\n\n/**\n * Default {@link PathBase} implementation for the current platform.\n *\n * {@link PathWin32} on Windows systems, {@link PathPosix} on all others.\n */\nexport const Path = process.platform === 'win32' ? PathWin32 : PathPosix\nexport type Path = PathBase | InstanceType<typeof Path>\n\n/**\n * Default {@link PathScurryBase} implementation for the current platform.\n *\n * {@link PathScurryWin32} on Windows systems, {@link PathScurryDarwin} on\n * Darwin (macOS) systems, {@link PathScurryPosix} on all others.\n */\nexport const PathScurry:\n | typeof PathScurryWin32\n | typeof PathScurryDarwin\n | typeof PathScurryPosix =\n process.platform === 'win32' ? PathScurryWin32\n : process.platform === 'darwin' ? PathScurryDarwin\n : PathScurryPosix\nexport type PathScurry = PathScurryBase | InstanceType<typeof PathScurry>\n","const proc =\n typeof process === 'object' && process\n ? process\n : {\n stdout: null,\n stderr: null,\n }\nimport { EventEmitter } from 'node:events'\nimport Stream from 'node:stream'\nimport { StringDecoder } from 'node:string_decoder'\n\n/**\n * Same as StringDecoder, but exposing the `lastNeed` flag on the type\n */\ntype SD = StringDecoder & { lastNeed: boolean }\n\nexport type { SD, Pipe, PipeProxyErrors }\n\n/**\n * Return true if the argument is a Minipass stream, Node stream, or something\n * else that Minipass can interact with.\n */\nexport const isStream = (\n s: any\n): s is Minipass.Readable | Minipass.Writable =>\n !!s &&\n typeof s === 'object' &&\n (s instanceof Minipass ||\n s instanceof Stream ||\n isReadable(s) ||\n isWritable(s))\n\n/**\n * Return true if the argument is a valid {@link Minipass.Readable}\n */\nexport const isReadable = (s: any): s is Minipass.Readable =>\n !!s &&\n typeof s === 'object' &&\n s instanceof EventEmitter &&\n typeof (s as Minipass.Readable).pipe === 'function' &&\n // node core Writable streams have a pipe() method, but it throws\n (s as Minipass.Readable).pipe !== Stream.Writable.prototype.pipe\n\n/**\n * Return true if the argument is a valid {@link Minipass.Writable}\n */\nexport const isWritable = (s: any): s is Minipass.Readable =>\n !!s &&\n typeof s === 'object' &&\n s instanceof EventEmitter &&\n typeof (s as Minipass.Writable).write === 'function' &&\n typeof (s as Minipass.Writable).end === 'function'\n\nconst EOF = Symbol('EOF')\nconst MAYBE_EMIT_END = Symbol('maybeEmitEnd')\nconst EMITTED_END = Symbol('emittedEnd')\nconst EMITTING_END = Symbol('emittingEnd')\nconst EMITTED_ERROR = Symbol('emittedError')\nconst CLOSED = Symbol('closed')\nconst READ = Symbol('read')\nconst FLUSH = Symbol('flush')\nconst FLUSHCHUNK = Symbol('flushChunk')\nconst ENCODING = Symbol('encoding')\nconst DECODER = Symbol('decoder')\nconst FLOWING = Symbol('flowing')\nconst PAUSED = Symbol('paused')\nconst RESUME = Symbol('resume')\nconst BUFFER = Symbol('buffer')\nconst PIPES = Symbol('pipes')\nconst BUFFERLENGTH = Symbol('bufferLength')\nconst BUFFERPUSH = Symbol('bufferPush')\nconst BUFFERSHIFT = Symbol('bufferShift')\nconst OBJECTMODE = Symbol('objectMode')\n// internal event when stream is destroyed\nconst DESTROYED = Symbol('destroyed')\n// internal event when stream has an error\nconst ERROR = Symbol('error')\nconst EMITDATA = Symbol('emitData')\nconst EMITEND = Symbol('emitEnd')\nconst EMITEND2 = Symbol('emitEnd2')\nconst ASYNC = Symbol('async')\nconst ABORT = Symbol('abort')\nconst ABORTED = Symbol('aborted')\nconst SIGNAL = Symbol('signal')\nconst DATALISTENERS = Symbol('dataListeners')\nconst DISCARDED = Symbol('discarded')\n\nconst defer = (fn: (...a: any[]) => any) => Promise.resolve().then(fn)\nconst nodefer = (fn: (...a: any[]) => any) => fn()\n\n// events that mean 'the stream is over'\n// these are treated specially, and re-emitted\n// if they are listened for after emitting.\ntype EndishEvent = 'end' | 'finish' | 'prefinish'\nconst isEndish = (ev: any): ev is EndishEvent =>\n ev === 'end' || ev === 'finish' || ev === 'prefinish'\n\nconst isArrayBufferLike = (b: any): b is ArrayBufferLike =>\n b instanceof ArrayBuffer ||\n (!!b &&\n typeof b === 'object' &&\n b.constructor &&\n b.constructor.name === 'ArrayBuffer' &&\n b.byteLength >= 0)\n\nconst isArrayBufferView = (b: any): b is ArrayBufferView =>\n !Buffer.isBuffer(b) && ArrayBuffer.isView(b)\n\n/**\n * Options that may be passed to stream.pipe()\n */\nexport interface PipeOptions {\n /**\n * end the destination stream when the source stream ends\n */\n end?: boolean\n /**\n * proxy errors from the source stream to the destination stream\n */\n proxyErrors?: boolean\n}\n\n/**\n * Internal class representing a pipe to a destination stream.\n *\n * @internal\n */\nclass Pipe<T extends unknown> {\n src: Minipass<T>\n dest: Minipass<any, T>\n opts: PipeOptions\n ondrain: () => any\n constructor(\n src: Minipass<T>,\n dest: Minipass.Writable,\n opts: PipeOptions\n ) {\n this.src = src\n this.dest = dest as Minipass<any, T>\n this.opts = opts\n this.ondrain = () => src[RESUME]()\n this.dest.on('drain', this.ondrain)\n }\n unpipe() {\n this.dest.removeListener('drain', this.ondrain)\n }\n // only here for the prototype\n /* c8 ignore start */\n proxyErrors(_er: any) {}\n /* c8 ignore stop */\n end() {\n this.unpipe()\n if (this.opts.end) this.dest.end()\n }\n}\n\n/**\n * Internal class representing a pipe to a destination stream where\n * errors are proxied.\n *\n * @internal\n */\nclass PipeProxyErrors<T> extends Pipe<T> {\n unpipe() {\n this.src.removeListener('error', this.proxyErrors)\n super.unpipe()\n }\n constructor(\n src: Minipass<T>,\n dest: Minipass.Writable,\n opts: PipeOptions\n ) {\n super(src, dest, opts)\n this.proxyErrors = er => dest.emit('error', er)\n src.on('error', this.proxyErrors)\n }\n}\n\nexport namespace Minipass {\n /**\n * Encoding used to create a stream that outputs strings rather than\n * Buffer objects.\n */\n export type Encoding = BufferEncoding | 'buffer' | null\n\n /**\n * Any stream that Minipass can pipe into\n */\n export type Writable =\n | Minipass<any, any, any>\n | NodeJS.WriteStream\n | (NodeJS.WriteStream & { fd: number })\n | (EventEmitter & {\n end(): any\n write(chunk: any, ...args: any[]): any\n })\n\n /**\n * Any stream that can be read from\n */\n export type Readable =\n | Minipass<any, any, any>\n | NodeJS.ReadStream\n | (NodeJS.ReadStream & { fd: number })\n | (EventEmitter & {\n pause(): any\n resume(): any\n pipe(...destArgs: any[]): any\n })\n\n /**\n * Utility type that can be iterated sync or async\n */\n export type DualIterable<T> = Iterable<T> & AsyncIterable<T>\n\n type EventArguments = Record<string | symbol, unknown[]>\n\n /**\n * The listing of events that a Minipass class can emit.\n * Extend this when extending the Minipass class, and pass as\n * the third template argument. The key is the name of the event,\n * and the value is the argument list.\n *\n * Any undeclared events will still be allowed, but the handler will get\n * arguments as `unknown[]`.\n */\n export interface Events<RType extends any = Buffer>\n extends EventArguments {\n readable: []\n data: [chunk: RType]\n error: [er: unknown]\n abort: [reason: unknown]\n drain: []\n resume: []\n end: []\n finish: []\n prefinish: []\n close: []\n [DESTROYED]: [er?: unknown]\n [ERROR]: [er: unknown]\n }\n\n /**\n * String or buffer-like data that can be joined and sliced\n */\n export type ContiguousData =\n | Buffer\n | ArrayBufferLike\n | ArrayBufferView\n | string\n export type BufferOrString = Buffer | string\n\n /**\n * Options passed to the Minipass constructor.\n */\n export type SharedOptions = {\n /**\n * Defer all data emission and other events until the end of the\n * current tick, similar to Node core streams\n */\n async?: boolean\n /**\n * A signal which will abort the stream\n */\n signal?: AbortSignal\n /**\n * Output string encoding. Set to `null` or `'buffer'` (or omit) to\n * emit Buffer objects rather than strings.\n *\n * Conflicts with `objectMode`\n */\n encoding?: BufferEncoding | null | 'buffer'\n /**\n * Output data exactly as it was written, supporting non-buffer/string\n * data (such as arbitrary objects, falsey values, etc.)\n *\n * Conflicts with `encoding`\n */\n objectMode?: boolean\n }\n\n /**\n * Options for a string encoded output\n */\n export type EncodingOptions = SharedOptions & {\n encoding: BufferEncoding\n objectMode?: false\n }\n\n /**\n * Options for contiguous data buffer output\n */\n export type BufferOptions = SharedOptions & {\n encoding?: null | 'buffer'\n objectMode?: false\n }\n\n /**\n * Options for objectMode arbitrary output\n */\n export type ObjectModeOptions = SharedOptions & {\n objectMode: true\n encoding?: null\n }\n\n /**\n * Utility type to determine allowed options based on read type\n */\n export type Options<T> =\n | ObjectModeOptions\n | (T extends string\n ? EncodingOptions\n : T extends Buffer\n ? BufferOptions\n : SharedOptions)\n}\n\nconst isObjectModeOptions = (\n o: Minipass.SharedOptions\n): o is Minipass.ObjectModeOptions => !!o.objectMode\n\nconst isEncodingOptions = (\n o: Minipass.SharedOptions\n): o is Minipass.EncodingOptions =>\n !o.objectMode && !!o.encoding && o.encoding !== 'buffer'\n\n/**\n * Main export, the Minipass class\n *\n * `RType` is the type of data emitted, defaults to Buffer\n *\n * `WType` is the type of data to be written, if RType is buffer or string,\n * then any {@link Minipass.ContiguousData} is allowed.\n *\n * `Events` is the set of event handler signatures that this object\n * will emit, see {@link Minipass.Events}\n */\nexport class Minipass<\n RType extends unknown = Buffer,\n WType extends unknown = RType extends Minipass.BufferOrString\n ? Minipass.ContiguousData\n : RType,\n Events extends Minipass.Events<RType> = Minipass.Events<RType>\n >\n extends EventEmitter\n implements Minipass.DualIterable<RType>\n{\n [FLOWING]: boolean = false;\n [PAUSED]: boolean = false;\n [PIPES]: Pipe<RType>[] = [];\n [BUFFER]: RType[] = [];\n [OBJECTMODE]: boolean;\n [ENCODING]: BufferEncoding | null;\n [ASYNC]: boolean;\n [DECODER]: SD | null;\n [EOF]: boolean = false;\n [EMITTED_END]: boolean = false;\n [EMITTING_END]: boolean = false;\n [CLOSED]: boolean = false;\n [EMITTED_ERROR]: unknown = null;\n [BUFFERLENGTH]: number = 0;\n [DESTROYED]: boolean = false;\n [SIGNAL]?: AbortSignal;\n [ABORTED]: boolean = false;\n [DATALISTENERS]: number = 0;\n [DISCARDED]: boolean = false\n\n /**\n * true if the stream can be written\n */\n writable: boolean = true\n /**\n * true if the stream can be read\n */\n readable: boolean = true\n\n /**\n * If `RType` is Buffer, then options do not need to be provided.\n * Otherwise, an options object must be provided to specify either\n * {@link Minipass.SharedOptions.objectMode} or\n * {@link Minipass.SharedOptions.encoding}, as appropriate.\n */\n constructor(\n ...args:\n | [Minipass.ObjectModeOptions]\n | (RType extends Buffer\n ? [] | [Minipass.Options<RType>]\n : [Minipass.Options<RType>])\n ) {\n const options: Minipass.Options<RType> = (args[0] ||\n {}) as Minipass.Options<RType>\n super()\n if (options.objectMode && typeof options.encoding === 'string') {\n throw new TypeError(\n 'Encoding and objectMode may not be used together'\n )\n }\n if (isObjectModeOptions(options)) {\n this[OBJECTMODE] = true\n this[ENCODING] = null\n } else if (isEncodingOptions(options)) {\n this[ENCODING] = options.encoding\n this[OBJECTMODE] = false\n } else {\n this[OBJECTMODE] = false\n this[ENCODING] = null\n }\n this[ASYNC] = !!options.async\n this[DECODER] = this[ENCODING]\n ? (new StringDecoder(this[ENCODING]) as SD)\n : null\n\n //@ts-ignore - private option for debugging and testing\n if (options && options.debugExposeBuffer === true) {\n Object.defineProperty(this, 'buffer', { get: () => this[BUFFER] })\n }\n //@ts-ignore - private option for debugging and testing\n if (options && options.debugExposePipes === true) {\n Object.defineProperty(this, 'pipes', { get: () => this[PIPES] })\n }\n\n const { signal } = options\n if (signal) {\n this[SIGNAL] = signal\n if (signal.aborted) {\n this[ABORT]()\n } else {\n signal.addEventListener('abort', () => this[ABORT]())\n }\n }\n }\n\n /**\n * The amount of data stored in the buffer waiting to be read.\n *\n * For Buffer strings, this will be the total byte length.\n * For string encoding streams, this will be the string character length,\n * according to JavaScript's `string.length` logic.\n * For objectMode streams, this is a count of the items waiting to be\n * emitted.\n */\n get bufferLength() {\n return this[BUFFERLENGTH]\n }\n\n /**\n * The `BufferEncoding` currently in use, or `null`\n */\n get encoding() {\n return this[ENCODING]\n }\n\n /**\n * @deprecated - This is a read only property\n */\n set encoding(_enc) {\n throw new Error('Encoding must be set at instantiation time')\n }\n\n /**\n * @deprecated - Encoding may only be set at instantiation time\n */\n setEncoding(_enc: Minipass.Encoding) {\n throw new Error('Encoding must be set at instantiation time')\n }\n\n /**\n * True if this is an objectMode stream\n */\n get objectMode() {\n return this[OBJECTMODE]\n }\n\n /**\n * @deprecated - This is a read-only property\n */\n set objectMode(_om) {\n throw new Error('objectMode must be set at instantiation time')\n }\n\n /**\n * true if this is an async stream\n */\n get ['async'](): boolean {\n return this[ASYNC]\n }\n /**\n * Set to true to make this stream async.\n *\n * Once set, it cannot be unset, as this would potentially cause incorrect\n * behavior. Ie, a sync stream can be made async, but an async stream\n * cannot be safely made sync.\n */\n set ['async'](a: boolean) {\n this[ASYNC] = this[ASYNC] || !!a\n }\n\n // drop everything and get out of the flow completely\n [ABORT]() {\n this[ABORTED] = true\n this.emit('abort', this[SIGNAL]?.reason)\n this.destroy(this[SIGNAL]?.reason)\n }\n\n /**\n * True if the stream has been aborted.\n */\n get aborted() {\n return this[ABORTED]\n }\n /**\n * No-op setter. Stream aborted status is set via the AbortSignal provided\n * in the constructor options.\n */\n set aborted(_) {}\n\n /**\n * Write data into the stream\n *\n * If the chunk written is a string, and encoding is not specified, then\n * `utf8` will be assumed. If the stream encoding matches the encoding of\n * a written string, and the state of the string decoder allows it, then\n * the string will be passed through to either the output or the internal\n * buffer without any processing. Otherwise, it will be turned into a\n * Buffer object for processing into the desired encoding.\n *\n * If provided, `cb` function is called immediately before return for\n * sync streams, or on next tick for async streams, because for this\n * base class, a chunk is considered \"processed\" once it is accepted\n * and either emitted or buffered. That is, the callback does not indicate\n * that the chunk has been eventually emitted, though of course child\n * classes can override this function to do whatever processing is required\n * and call `super.write(...)` only once processing is completed.\n */\n write(chunk: WType, cb?: () => void): boolean\n write(\n chunk: WType,\n encoding?: Minipass.Encoding,\n cb?: () => void\n ): boolean\n write(\n chunk: WType,\n encoding?: Minipass.Encoding | (() => void),\n cb?: () => void\n ): boolean {\n if (this[ABORTED]) return false\n if (this[EOF]) throw new Error('write after end')\n\n if (this[DESTROYED]) {\n this.emit(\n 'error',\n Object.assign(\n new Error('Cannot call write after a stream was destroyed'),\n { code: 'ERR_STREAM_DESTROYED' }\n )\n )\n return true\n }\n\n if (typeof encoding === 'function') {\n cb = encoding\n encoding = 'utf8'\n }\n\n if (!encoding) encoding = 'utf8'\n\n const fn = this[ASYNC] ? defer : nodefer\n\n // convert array buffers and typed array views into buffers\n // at some point in the future, we may want to do the opposite!\n // leave strings and buffers as-is\n // anything is only allowed if in object mode, so throw\n if (!this[OBJECTMODE] && !Buffer.isBuffer(chunk)) {\n if (isArrayBufferView(chunk)) {\n //@ts-ignore - sinful unsafe type changing\n chunk = Buffer.from(\n chunk.buffer,\n chunk.byteOffset,\n chunk.byteLength\n )\n } else if (isArrayBufferLike(chunk)) {\n //@ts-ignore - sinful unsafe type changing\n chunk = Buffer.from(chunk)\n } else if (typeof chunk !== 'string') {\n throw new Error(\n 'Non-contiguous data written to non-objectMode stream'\n )\n }\n }\n\n // handle object mode up front, since it's simpler\n // this yields better performance, fewer checks later.\n if (this[OBJECTMODE]) {\n // maybe impossible?\n /* c8 ignore start */\n if (this[FLOWING] && this[BUFFERLENGTH] !== 0) this[FLUSH](true)\n /* c8 ignore stop */\n\n if (this[FLOWING]) this.emit('data', chunk as unknown as RType)\n else this[BUFFERPUSH](chunk as unknown as RType)\n\n if (this[BUFFERLENGTH] !== 0) this.emit('readable')\n\n if (cb) fn(cb)\n\n return this[FLOWING]\n }\n\n // at this point the chunk is a buffer or string\n // don't buffer it up or send it to the decoder\n if (!(chunk as Minipass.BufferOrString).length) {\n if (this[BUFFERLENGTH] !== 0) this.emit('readable')\n if (cb) fn(cb)\n return this[FLOWING]\n }\n\n // fast-path writing strings of same encoding to a stream with\n // an empty buffer, skipping the buffer/decoder dance\n if (\n typeof chunk === 'string' &&\n // unless it is a string already ready for us to use\n !(encoding === this[ENCODING] && !this[DECODER]?.lastNeed)\n ) {\n //@ts-ignore - sinful unsafe type change\n chunk = Buffer.from(chunk, encoding)\n }\n\n if (Buffer.isBuffer(chunk) && this[ENCODING]) {\n //@ts-ignore - sinful unsafe type change\n chunk = this[DECODER].write(chunk)\n }\n\n // Note: flushing CAN potentially switch us into not-flowing mode\n if (this[FLOWING] && this[BUFFERLENGTH] !== 0) this[FLUSH](true)\n\n if (this[FLOWING]) this.emit('data', chunk as unknown as RType)\n else this[BUFFERPUSH](chunk as unknown as RType)\n\n if (this[BUFFERLENGTH] !== 0) this.emit('readable')\n\n if (cb) fn(cb)\n\n return this[FLOWING]\n }\n\n /**\n * Low-level explicit read method.\n *\n * In objectMode, the argument is ignored, and one item is returned if\n * available.\n *\n * `n` is the number of bytes (or in the case of encoding streams,\n * characters) to consume. If `n` is not provided, then the entire buffer\n * is returned, or `null` is returned if no data is available.\n *\n * If `n` is greater that the amount of data in the internal buffer,\n * then `null` is returned.\n */\n read(n?: number | null): RType | null {\n if (this[DESTROYED]) return null\n this[DISCARDED] = false\n\n if (\n this[BUFFERLENGTH] === 0 ||\n n === 0 ||\n (n && n > this[BUFFERLENGTH])\n ) {\n this[MAYBE_EMIT_END]()\n return null\n }\n\n if (this[OBJECTMODE]) n = null\n\n if (this[BUFFER].length > 1 && !this[OBJECTMODE]) {\n // not object mode, so if we have an encoding, then RType is string\n // otherwise, must be Buffer\n this[BUFFER] = [\n (this[ENCODING]\n ? this[BUFFER].join('')\n : Buffer.concat(\n this[BUFFER] as Buffer[],\n this[BUFFERLENGTH]\n )) as RType,\n ]\n }\n\n const ret = this[READ](n || null, this[BUFFER][0] as RType)\n this[MAYBE_EMIT_END]()\n return ret\n }\n\n [READ](n: number | null, chunk: RType) {\n if (this[OBJECTMODE]) this[BUFFERSHIFT]()\n else {\n const c = chunk as Minipass.BufferOrString\n if (n === c.length || n === null) this[BUFFERSHIFT]()\n else if (typeof c === 'string') {\n this[BUFFER][0] = c.slice(n) as RType\n chunk = c.slice(0, n) as RType\n this[BUFFERLENGTH] -= n\n } else {\n this[BUFFER][0] = c.subarray(n) as RType\n chunk = c.subarray(0, n) as RType\n this[BUFFERLENGTH] -= n\n }\n }\n\n this.emit('data', chunk)\n\n if (!this[BUFFER].length && !this[EOF]) this.emit('drain')\n\n return chunk\n }\n\n /**\n * End the stream, optionally providing a final write.\n *\n * See {@link Minipass#write} for argument descriptions\n */\n end(cb?: () => void): this\n end(chunk: WType, cb?: () => void): this\n end(chunk: WType, encoding?: Minipass.Encoding, cb?: () => void): this\n end(\n chunk?: WType | (() => void),\n encoding?: Minipass.Encoding | (() => void),\n cb?: () => void\n ): this {\n if (typeof chunk === 'function') {\n cb = chunk as () => void\n chunk = undefined\n }\n if (typeof encoding === 'function') {\n cb = encoding\n encoding = 'utf8'\n }\n if (chunk !== undefined) this.write(chunk, encoding)\n if (cb) this.once('end', cb)\n this[EOF] = true\n this.writable = false\n\n // if we haven't written anything, then go ahead and emit,\n // even if we're not reading.\n // we'll re-emit if a new 'end' listener is added anyway.\n // This makes MP more suitable to write-only use cases.\n if (this[FLOWING] || !this[PAUSED]) this[MAYBE_EMIT_END]()\n return this\n }\n\n // don't let the internal resume be overwritten\n [RESUME]() {\n if (this[DESTROYED]) return\n\n if (!this[DATALISTENERS] && !this[PIPES].length) {\n this[DISCARDED] = true\n }\n this[PAUSED] = false\n this[FLOWING] = true\n this.emit('resume')\n if (this[BUFFER].length) this[FLUSH]()\n else if (this[EOF]) this[MAYBE_EMIT_END]()\n else this.emit('drain')\n }\n\n /**\n * Resume the stream if it is currently in a paused state\n *\n * If called when there are no pipe destinations or `data` event listeners,\n * this will place the stream in a \"discarded\" state, where all data will\n * be thrown away. The discarded state is removed if a pipe destination or\n * data handler is added, if pause() is called, or if any synchronous or\n * asynchronous iteration is started.\n */\n resume() {\n return this[RESUME]()\n }\n\n /**\n * Pause the stream\n */\n pause() {\n this[FLOWING] = false\n this[PAUSED] = true\n this[DISCARDED] = false\n }\n\n /**\n * true if the stream has been forcibly destroyed\n */\n get destroyed() {\n return this[DESTROYED]\n }\n\n /**\n * true if the stream is currently in a flowing state, meaning that\n * any writes will be immediately emitted.\n */\n get flowing() {\n return this[FLOWING]\n }\n\n /**\n * true if the stream is currently in a paused state\n */\n get paused() {\n return this[PAUSED]\n }\n\n [BUFFERPUSH](chunk: RType) {\n if (this[OBJECTMODE]) this[BUFFERLENGTH] += 1\n else this[BUFFERLENGTH] += (chunk as Minipass.BufferOrString).length\n this[BUFFER].push(chunk)\n }\n\n [BUFFERSHIFT](): RType {\n if (this[OBJECTMODE]) this[BUFFERLENGTH] -= 1\n else\n this[BUFFERLENGTH] -= (\n this[BUFFER][0] as Minipass.BufferOrString\n ).length\n return this[BUFFER].shift() as RType\n }\n\n [FLUSH](noDrain: boolean = false) {\n do {} while (\n this[FLUSHCHUNK](this[BUFFERSHIFT]()) &&\n this[BUFFER].length\n )\n\n if (!noDrain && !this[BUFFER].length && !this[EOF]) this.emit('drain')\n }\n\n [FLUSHCHUNK](chunk: RType) {\n this.emit('data', chunk)\n return this[FLOWING]\n }\n\n /**\n * Pipe all data emitted by this stream into the destination provided.\n *\n * Triggers the flow of data.\n */\n pipe<W extends Minipass.Writable>(dest: W, opts?: PipeOptions): W {\n if (this[DESTROYED]) return dest\n this[DISCARDED] = false\n\n const ended = this[EMITTED_END]\n opts = opts || {}\n if (dest === proc.stdout || dest === proc.stderr) opts.end = false\n else opts.end = opts.end !== false\n opts.proxyErrors = !!opts.proxyErrors\n\n // piping an ended stream ends immediately\n if (ended) {\n if (opts.end) dest.end()\n } else {\n // \"as\" here just ignores the WType, which pipes don't care about,\n // since they're only consuming from us, and writing to the dest\n this[PIPES].push(\n !opts.proxyErrors\n ? new Pipe<RType>(this as Minipass<RType>, dest, opts)\n : new PipeProxyErrors<RType>(this as Minipass<RType>, dest, opts)\n )\n if (this[ASYNC]) defer(() => this[RESUME]())\n else this[RESUME]()\n }\n\n return dest\n }\n\n /**\n * Fully unhook a piped destination stream.\n *\n * If the destination stream was the only consumer of this stream (ie,\n * there are no other piped destinations or `'data'` event listeners)\n * then the flow of data will stop until there is another consumer or\n * {@link Minipass#resume} is explicitly called.\n */\n unpipe<W extends Minipass.Writable>(dest: W) {\n const p = this[PIPES].find(p => p.dest === dest)\n if (p) {\n if (this[PIPES].length === 1) {\n if (this[FLOWING] && this[DATALISTENERS] === 0) {\n this[FLOWING] = false\n }\n this[PIPES] = []\n } else this[PIPES].splice(this[PIPES].indexOf(p), 1)\n p.unpipe()\n }\n }\n\n /**\n * Alias for {@link Minipass#on}\n */\n addListener<Event extends keyof Events>(\n ev: Event,\n handler: (...args: Events[Event]) => any\n ): this {\n return this.on(ev, handler)\n }\n\n /**\n * Mostly identical to `EventEmitter.on`, with the following\n * behavior differences to prevent data loss and unnecessary hangs:\n *\n * - Adding a 'data' event handler will trigger the flow of data\n *\n * - Adding a 'readable' event handler when there is data waiting to be read\n * will cause 'readable' to be emitted immediately.\n *\n * - Adding an 'endish' event handler ('end', 'finish', etc.) which has\n * already passed will cause the event to be emitted immediately and all\n * handlers removed.\n *\n * - Adding an 'error' event handler after an error has been emitted will\n * cause the event to be re-emitted immediately with the error previously\n * raised.\n */\n on<Event extends keyof Events>(\n ev: Event,\n handler: (...args: Events[Event]) => any\n ): this {\n const ret = super.on(\n ev as string | symbol,\n handler as (...a: any[]) => any\n )\n if (ev === 'data') {\n this[DISCARDED] = false\n this[DATALISTENERS]++\n if (!this[PIPES].length && !this[FLOWING]) {\n this[RESUME]()\n }\n } else if (ev === 'readable' && this[BUFFERLENGTH] !== 0) {\n super.emit('readable')\n } else if (isEndish(ev) && this[EMITTED_END]) {\n super.emit(ev)\n this.removeAllListeners(ev)\n } else if (ev === 'error' && this[EMITTED_ERROR]) {\n const h = handler as (...a: Events['error']) => any\n if (this[ASYNC]) defer(() => h.call(this, this[EMITTED_ERROR]))\n else h.call(this, this[EMITTED_ERROR])\n }\n return ret\n }\n\n /**\n * Alias for {@link Minipass#off}\n */\n removeListener<Event extends keyof Events>(\n ev: Event,\n handler: (...args: Events[Event]) => any\n ) {\n return this.off(ev, handler)\n }\n\n /**\n * Mostly identical to `EventEmitter.off`\n *\n * If a 'data' event handler is removed, and it was the last consumer\n * (ie, there are no pipe destinations or other 'data' event listeners),\n * then the flow of data will stop until there is another consumer or\n * {@link Minipass#resume} is explicitly called.\n */\n off<Event extends keyof Events>(\n ev: Event,\n handler: (...args: Events[Event]) => any\n ) {\n const ret = super.off(\n ev as string | symbol,\n handler as (...a: any[]) => any\n )\n // if we previously had listeners, and now we don't, and we don't\n // have any pipes, then stop the flow, unless it's been explicitly\n // put in a discarded flowing state via stream.resume().\n if (ev === 'data') {\n this[DATALISTENERS] = this.listeners('data').length\n if (\n this[DATALISTENERS] === 0 &&\n !this[DISCARDED] &&\n !this[PIPES].length\n ) {\n this[FLOWING] = false\n }\n }\n return ret\n }\n\n /**\n * Mostly identical to `EventEmitter.removeAllListeners`\n *\n * If all 'data' event handlers are removed, and they were the last consumer\n * (ie, there are no pipe destinations), then the flow of data will stop\n * until there is another consumer or {@link Minipass#resume} is explicitly\n * called.\n */\n removeAllListeners<Event extends keyof Events>(ev?: Event) {\n const ret = super.removeAllListeners(ev as string | symbol | undefined)\n if (ev === 'data' || ev === undefined) {\n this[DATALISTENERS] = 0\n if (!this[DISCARDED] && !this[PIPES].length) {\n this[FLOWING] = false\n }\n }\n return ret\n }\n\n /**\n * true if the 'end' event has been emitted\n */\n get emittedEnd() {\n return this[EMITTED_END]\n }\n\n [MAYBE_EMIT_END]() {\n if (\n !this[EMITTING_END] &&\n !this[EMITTED_END] &&\n !this[DESTROYED] &&\n this[BUFFER].length === 0 &&\n this[EOF]\n ) {\n this[EMITTING_END] = true\n this.emit('end')\n this.emit('prefinish')\n this.emit('finish')\n if (this[CLOSED]) this.emit('close')\n this[EMITTING_END] = false\n }\n }\n\n /**\n * Mostly identical to `EventEmitter.emit`, with the following\n * behavior differences to prevent data loss and unnecessary hangs:\n *\n * If the stream has been destroyed, and the event is something other\n * than 'close' or 'error', then `false` is returned and no handlers\n * are called.\n *\n * If the event is 'end', and has already been emitted, then the event\n * is ignored. If the stream is in a paused or non-flowing state, then\n * the event will be deferred until data flow resumes. If the stream is\n * async, then handlers will be called on the next tick rather than\n * immediately.\n *\n * If the event is 'close', and 'end' has not yet been emitted, then\n * the event will be deferred until after 'end' is emitted.\n *\n * If the event is 'error', and an AbortSignal was provided for the stream,\n * and there are no listeners, then the event is ignored, matching the\n * behavior of node core streams in the presense of an AbortSignal.\n *\n * If the event is 'finish' or 'prefinish', then all listeners will be\n * removed after emitting the event, to prevent double-firing.\n */\n emit<Event extends keyof Events>(\n ev: Event,\n ...args: Events[Event]\n ): boolean {\n const data = args[0]\n // error and close are only events allowed after calling destroy()\n if (\n ev !== 'error' &&\n ev !== 'close' &&\n ev !== DESTROYED &&\n this[DESTROYED]\n ) {\n return false\n } else if (ev === 'data') {\n return !this[OBJECTMODE] && !data\n ? false\n : this[ASYNC]\n ? (defer(() => this[EMITDATA](data as RType)), true)\n : this[EMITDATA](data as RType)\n } else if (ev === 'end') {\n return this[EMITEND]()\n } else if (ev === 'close') {\n this[CLOSED] = true\n // don't emit close before 'end' and 'finish'\n if (!this[EMITTED_END] && !this[DESTROYED]) return false\n const ret = super.emit('close')\n this.removeAllListeners('close')\n return ret\n } else if (ev === 'error') {\n this[EMITTED_ERROR] = data\n super.emit(ERROR, data)\n const ret =\n !this[SIGNAL] || this.listeners('error').length\n ? super.emit('error', data)\n : false\n this[MAYBE_EMIT_END]()\n return ret\n } else if (ev === 'resume') {\n const ret = super.emit('resume')\n this[MAYBE_EMIT_END]()\n return ret\n } else if (ev === 'finish' || ev === 'prefinish') {\n const ret = super.emit(ev)\n this.removeAllListeners(ev)\n return ret\n }\n\n // Some other unknown event\n const ret = super.emit(ev as string, ...args)\n this[MAYBE_EMIT_END]()\n return ret\n }\n\n [EMITDATA](data: RType) {\n for (const p of this[PIPES]) {\n if (p.dest.write(data as RType) === false) this.pause()\n }\n const ret = this[DISCARDED] ? false : super.emit('data', data)\n this[MAYBE_EMIT_END]()\n return ret\n }\n\n [EMITEND]() {\n if (this[EMITTED_END]) return false\n\n this[EMITTED_END] = true\n this.readable = false\n return this[ASYNC]\n ? (defer(() => this[EMITEND2]()), true)\n : this[EMITEND2]()\n }\n\n [EMITEND2]() {\n if (this[DECODER]) {\n const data = this[DECODER].end()\n if (data) {\n for (const p of this[PIPES]) {\n p.dest.write(data as RType)\n }\n if (!this[DISCARDED]) super.emit('data', data)\n }\n }\n\n for (const p of this[PIPES]) {\n p.end()\n }\n const ret = super.emit('end')\n this.removeAllListeners('end')\n return ret\n }\n\n /**\n * Return a Promise that resolves to an array of all emitted data once\n * the stream ends.\n */\n async collect(): Promise<RType[] & { dataLength: number }> {\n const buf: RType[] & { dataLength: number } = Object.assign([], {\n dataLength: 0,\n })\n if (!this[OBJECTMODE]) buf.dataLength = 0\n // set the promise first, in case an error is raised\n // by triggering the flow here.\n const p = this.promise()\n this.on('data', c => {\n buf.push(c)\n if (!this[OBJECTMODE])\n buf.dataLength += (c as Minipass.BufferOrString).length\n })\n await p\n return buf\n }\n\n /**\n * Return a Promise that resolves to the concatenation of all emitted data\n * once the stream ends.\n *\n * Not allowed on objectMode streams.\n */\n async concat(): Promise<RType> {\n if (this[OBJECTMODE]) {\n throw new Error('cannot concat in objectMode')\n }\n const buf = await this.collect()\n return (\n this[ENCODING]\n ? buf.join('')\n : Buffer.concat(buf as Buffer[], buf.dataLength)\n ) as RType\n }\n\n /**\n * Return a void Promise that resolves once the stream ends.\n */\n async promise(): Promise<void> {\n return new Promise<void>((resolve, reject) => {\n this.on(DESTROYED, () => reject(new Error('stream destroyed')))\n this.on('error', er => reject(er))\n this.on('end', () => resolve())\n })\n }\n\n /**\n * Asynchronous `for await of` iteration.\n *\n * This will continue emitting all chunks until the stream terminates.\n */\n [Symbol.asyncIterator](): AsyncGenerator<RType, void, void> {\n // set this up front, in case the consumer doesn't call next()\n // right away.\n this[DISCARDED] = false\n let stopped = false\n const stop = async (): Promise<IteratorReturnResult<void>> => {\n this.pause()\n stopped = true\n return { value: undefined, done: true }\n }\n const next = (): Promise<IteratorResult<RType, void>> => {\n if (stopped) return stop()\n const res = this.read()\n if (res !== null) return Promise.resolve({ done: false, value: res })\n\n if (this[EOF]) return stop()\n\n let resolve!: (res: IteratorResult<RType>) => void\n let reject!: (er: unknown) => void\n const onerr = (er: unknown) => {\n this.off('data', ondata)\n this.off('end', onend)\n this.off(DESTROYED, ondestroy)\n stop()\n reject(er)\n }\n const ondata = (value: RType) => {\n this.off('error', onerr)\n this.off('end', onend)\n this.off(DESTROYED, ondestroy)\n this.pause()\n resolve({ value, done: !!this[EOF] })\n }\n const onend = () => {\n this.off('error', onerr)\n this.off('data', ondata)\n this.off(DESTROYED, ondestroy)\n stop()\n resolve({ done: true, value: undefined })\n }\n const ondestroy = () => onerr(new Error('stream destroyed'))\n return new Promise<IteratorResult<RType>>((res, rej) => {\n reject = rej\n resolve = res\n this.once(DESTROYED, ondestroy)\n this.once('error', onerr)\n this.once('end', onend)\n this.once('data', ondata)\n })\n }\n\n return {\n next,\n throw: stop,\n return: stop,\n [Symbol.asyncIterator]() {\n return this\n },\n }\n }\n\n /**\n * Synchronous `for of` iteration.\n *\n * The iteration will terminate when the internal buffer runs out, even\n * if the stream has not yet terminated.\n */\n [Symbol.iterator](): Generator<RType, void, void> {\n // set this up front, in case the consumer doesn't call next()\n // right away.\n this[DISCARDED] = false\n let stopped = false\n const stop = (): IteratorReturnResult<void> => {\n this.pause()\n this.off(ERROR, stop)\n this.off(DESTROYED, stop)\n this.off('end', stop)\n stopped = true\n return { done: true, value: undefined }\n }\n\n const next = (): IteratorResult<RType, void> => {\n if (stopped) return stop()\n const value = this.read()\n return value === null ? stop() : { done: false, value }\n }\n\n this.once('end', stop)\n this.once(ERROR, stop)\n this.once(DESTROYED, stop)\n\n return {\n next,\n throw: stop,\n return: stop,\n [Symbol.iterator]() {\n return this\n },\n }\n }\n\n /**\n * Destroy a stream, preventing it from being used for any further purpose.\n *\n * If the stream has a `close()` method, then it will be called on\n * destruction.\n *\n * After destruction, any attempt to write data, read data, or emit most\n * events will be ignored.\n *\n * If an error argument is provided, then it will be emitted in an\n * 'error' event.\n */\n destroy(er?: unknown) {\n if (this[DESTROYED]) {\n if (er) this.emit('error', er)\n else this.emit(DESTROYED)\n return this\n }\n\n this[DESTROYED] = true\n this[DISCARDED] = true\n\n // throw away all buffered data, it's never coming out\n this[BUFFER].length = 0\n this[BUFFERLENGTH] = 0\n\n const wc = this as Minipass<RType, WType, Events> & {\n close?: () => void\n }\n if (typeof wc.close === 'function' && !this[CLOSED]) wc.close()\n\n if (er) this.emit('error', er)\n // if no error to emit, still reject pending promises\n else this.emit(DESTROYED)\n\n return this\n }\n\n /**\n * Alias for {@link isStream}\n *\n * Former export location, maintained for backwards compatibility.\n *\n * @deprecated\n */\n static get isStream() {\n return isStream\n }\n}\n","// this is just a very light wrapper around 2 arrays with an offset index\n\nimport { GLOBSTAR } from 'minimatch'\nexport type MMPattern = string | RegExp | typeof GLOBSTAR\n\n// an array of length >= 1\nexport type PatternList = [p: MMPattern, ...rest: MMPattern[]]\nexport type UNCPatternList = [\n p0: '',\n p1: '',\n p2: string,\n p3: string,\n ...rest: MMPattern[],\n]\nexport type DrivePatternList = [p0: string, ...rest: MMPattern[]]\nexport type AbsolutePatternList = [p0: '', ...rest: MMPattern[]]\nexport type GlobList = [p: string, ...rest: string[]]\n\nconst isPatternList = (pl: MMPattern[]): pl is PatternList =>\n pl.length >= 1\nconst isGlobList = (gl: string[]): gl is GlobList => gl.length >= 1\n\n/**\n * An immutable-ish view on an array of glob parts and their parsed\n * results\n */\nexport class Pattern {\n readonly #patternList: PatternList\n readonly #globList: GlobList\n readonly #index: number\n readonly length: number\n readonly #platform: NodeJS.Platform\n #rest?: Pattern | null\n #globString?: string\n #isDrive?: boolean\n #isUNC?: boolean\n #isAbsolute?: boolean\n #followGlobstar: boolean = true\n\n constructor(\n patternList: MMPattern[],\n globList: string[],\n index: number,\n platform: NodeJS.Platform,\n ) {\n if (!isPatternList(patternList)) {\n throw new TypeError('empty pattern list')\n }\n if (!isGlobList(globList)) {\n throw new TypeError('empty glob list')\n }\n if (globList.length !== patternList.length) {\n throw new TypeError('mismatched pattern list and glob list lengths')\n }\n this.length = patternList.length\n if (index < 0 || index >= this.length) {\n throw new TypeError('index out of range')\n }\n this.#patternList = patternList\n this.#globList = globList\n this.#index = index\n this.#platform = platform\n\n // normalize root entries of absolute patterns on initial creation.\n if (this.#index === 0) {\n // c: => ['c:/']\n // C:/ => ['C:/']\n // C:/x => ['C:/', 'x']\n // //host/share => ['//host/share/']\n // //host/share/ => ['//host/share/']\n // //host/share/x => ['//host/share/', 'x']\n // /etc => ['/', 'etc']\n // / => ['/']\n if (this.isUNC()) {\n // '' / '' / 'host' / 'share'\n const [p0, p1, p2, p3, ...prest] = this.#patternList\n const [g0, g1, g2, g3, ...grest] = this.#globList\n if (prest[0] === '') {\n // ends in /\n prest.shift()\n grest.shift()\n }\n const p = [p0, p1, p2, p3, ''].join('/')\n const g = [g0, g1, g2, g3, ''].join('/')\n this.#patternList = [p, ...prest]\n this.#globList = [g, ...grest]\n this.length = this.#patternList.length\n } else if (this.isDrive() || this.isAbsolute()) {\n const [p1, ...prest] = this.#patternList\n const [g1, ...grest] = this.#globList\n if (prest[0] === '') {\n // ends in /\n prest.shift()\n grest.shift()\n }\n const p = (p1 as string) + '/'\n const g = g1 + '/'\n this.#patternList = [p, ...prest]\n this.#globList = [g, ...grest]\n this.length = this.#patternList.length\n }\n }\n }\n\n /**\n * The first entry in the parsed list of patterns\n */\n pattern(): MMPattern {\n return this.#patternList[this.#index] as MMPattern\n }\n\n /**\n * true of if pattern() returns a string\n */\n isString(): boolean {\n return typeof this.#patternList[this.#index] === 'string'\n }\n /**\n * true of if pattern() returns GLOBSTAR\n */\n isGlobstar(): boolean {\n return this.#patternList[this.#index] === GLOBSTAR\n }\n /**\n * true if pattern() returns a regexp\n */\n isRegExp(): boolean {\n return this.#patternList[this.#index] instanceof RegExp\n }\n\n /**\n * The /-joined set of glob parts that make up this pattern\n */\n globString(): string {\n return (this.#globString =\n this.#globString ||\n (this.#index === 0 ?\n this.isAbsolute() ?\n this.#globList[0] + this.#globList.slice(1).join('/')\n : this.#globList.join('/')\n : this.#globList.slice(this.#index).join('/')))\n }\n\n /**\n * true if there are more pattern parts after this one\n */\n hasMore(): boolean {\n return this.length > this.#index + 1\n }\n\n /**\n * The rest of the pattern after this part, or null if this is the end\n */\n rest(): Pattern | null {\n if (this.#rest !== undefined) return this.#rest\n if (!this.hasMore()) return (this.#rest = null)\n this.#rest = new Pattern(\n this.#patternList,\n this.#globList,\n this.#index + 1,\n this.#platform,\n )\n this.#rest.#isAbsolute = this.#isAbsolute\n this.#rest.#isUNC = this.#isUNC\n this.#rest.#isDrive = this.#isDrive\n return this.#rest\n }\n\n /**\n * true if the pattern represents a //unc/path/ on windows\n */\n isUNC(): boolean {\n const pl = this.#patternList\n return this.#isUNC !== undefined ?\n this.#isUNC\n : (this.#isUNC =\n this.#platform === 'win32' &&\n this.#index === 0 &&\n pl[0] === '' &&\n pl[1] === '' &&\n typeof pl[2] === 'string' &&\n !!pl[2] &&\n typeof pl[3] === 'string' &&\n !!pl[3])\n }\n\n // pattern like C:/...\n // split = ['C:', ...]\n // XXX: would be nice to handle patterns like `c:*` to test the cwd\n // in c: for *, but I don't know of a way to even figure out what that\n // cwd is without actually chdir'ing into it?\n /**\n * True if the pattern starts with a drive letter on Windows\n */\n isDrive(): boolean {\n const pl = this.#patternList\n return this.#isDrive !== undefined ?\n this.#isDrive\n : (this.#isDrive =\n this.#platform === 'win32' &&\n this.#index === 0 &&\n this.length > 1 &&\n typeof pl[0] === 'string' &&\n /^[a-z]:$/i.test(pl[0]))\n }\n\n // pattern = '/' or '/...' or '/x/...'\n // split = ['', ''] or ['', ...] or ['', 'x', ...]\n // Drive and UNC both considered absolute on windows\n /**\n * True if the pattern is rooted on an absolute path\n */\n isAbsolute(): boolean {\n const pl = this.#patternList\n return this.#isAbsolute !== undefined ?\n this.#isAbsolute\n : (this.#isAbsolute =\n (pl[0] === '' && pl.length > 1) ||\n this.isDrive() ||\n this.isUNC())\n }\n\n /**\n * consume the root of the pattern, and return it\n */\n root(): string {\n const p = this.#patternList[0]\n return (\n typeof p === 'string' && this.isAbsolute() && this.#index === 0\n ) ?\n p\n : ''\n }\n\n /**\n * Check to see if the current globstar pattern is allowed to follow\n * a symbolic link.\n */\n checkFollowGlobstar(): boolean {\n return !(\n this.#index === 0 ||\n !this.isGlobstar() ||\n !this.#followGlobstar\n )\n }\n\n /**\n * Mark that the current globstar pattern is following a symbolic link\n */\n markFollowGlobstar(): boolean {\n if (this.#index === 0 || !this.isGlobstar() || !this.#followGlobstar)\n return false\n this.#followGlobstar = false\n return true\n }\n}\n","// give it a pattern, and it'll be able to tell you if\n// a given path should be ignored.\n// Ignoring a path ignores its children if the pattern ends in /**\n// Ignores are always parsed in dot:true mode\n\nimport { Minimatch, MinimatchOptions } from 'minimatch'\nimport { Path } from 'path-scurry'\nimport { Pattern } from './pattern.js'\nimport { GlobWalkerOpts } from './walker.js'\n\nexport interface IgnoreLike {\n ignored?: (p: Path) => boolean\n childrenIgnored?: (p: Path) => boolean\n add?: (ignore: string) => void\n}\n\nconst defaultPlatform: NodeJS.Platform =\n (\n typeof process === 'object' &&\n process &&\n typeof process.platform === 'string'\n ) ?\n process.platform\n : 'linux'\n\n/**\n * Class used to process ignored patterns\n */\nexport class Ignore implements IgnoreLike {\n relative: Minimatch[]\n relativeChildren: Minimatch[]\n absolute: Minimatch[]\n absoluteChildren: Minimatch[]\n platform: NodeJS.Platform\n mmopts: MinimatchOptions\n\n constructor(\n ignored: string[],\n {\n nobrace,\n nocase,\n noext,\n noglobstar,\n platform = defaultPlatform,\n }: GlobWalkerOpts,\n ) {\n this.relative = []\n this.absolute = []\n this.relativeChildren = []\n this.absoluteChildren = []\n this.platform = platform\n this.mmopts = {\n dot: true,\n nobrace,\n nocase,\n noext,\n noglobstar,\n optimizationLevel: 2,\n platform,\n nocomment: true,\n nonegate: true,\n }\n for (const ign of ignored) this.add(ign)\n }\n\n add(ign: string) {\n // this is a little weird, but it gives us a clean set of optimized\n // minimatch matchers, without getting tripped up if one of them\n // ends in /** inside a brace section, and it's only inefficient at\n // the start of the walk, not along it.\n // It'd be nice if the Pattern class just had a .test() method, but\n // handling globstars is a bit of a pita, and that code already lives\n // in minimatch anyway.\n // Another way would be if maybe Minimatch could take its set/globParts\n // as an option, and then we could at least just use Pattern to test\n // for absolute-ness.\n // Yet another way, Minimatch could take an array of glob strings, and\n // a cwd option, and do the right thing.\n const mm = new Minimatch(ign, this.mmopts)\n for (let i = 0; i < mm.set.length; i++) {\n const parsed = mm.set[i]\n const globParts = mm.globParts[i]\n /* c8 ignore start */\n if (!parsed || !globParts) {\n throw new Error('invalid pattern object')\n }\n // strip off leading ./ portions\n // https://github.com/isaacs/node-glob/issues/570\n while (parsed[0] === '.' && globParts[0] === '.') {\n parsed.shift()\n globParts.shift()\n }\n /* c8 ignore stop */\n const p = new Pattern(parsed, globParts, 0, this.platform)\n const m = new Minimatch(p.globString(), this.mmopts)\n const children = globParts[globParts.length - 1] === '**'\n const absolute = p.isAbsolute()\n if (absolute) this.absolute.push(m)\n else this.relative.push(m)\n if (children) {\n if (absolute) this.absoluteChildren.push(m)\n else this.relativeChildren.push(m)\n }\n }\n }\n\n ignored(p: Path): boolean {\n const fullpath = p.fullpath()\n const fullpaths = `${fullpath}/`\n const relative = p.relative() || '.'\n const relatives = `${relative}/`\n for (const m of this.relative) {\n if (m.match(relative) || m.match(relatives)) return true\n }\n for (const m of this.absolute) {\n if (m.match(fullpath) || m.match(fullpaths)) return true\n }\n return false\n }\n\n childrenIgnored(p: Path): boolean {\n const fullpath = p.fullpath() + '/'\n const relative = (p.relative() || '.') + '/'\n for (const m of this.relativeChildren) {\n if (m.match(relative)) return true\n }\n for (const m of this.absoluteChildren) {\n if (m.match(fullpath)) return true\n }\n return false\n }\n}\n","// synchronous utility for filtering entries and calculating subwalks\n\nimport { GLOBSTAR, MMRegExp } from 'minimatch'\nimport { Path } from 'path-scurry'\nimport { MMPattern, Pattern } from './pattern.js'\nimport { GlobWalkerOpts } from './walker.js'\n\n/**\n * A cache of which patterns have been processed for a given Path\n */\nexport class HasWalkedCache {\n store: Map<string, Set<string>>\n constructor(store: Map<string, Set<string>> = new Map()) {\n this.store = store\n }\n copy() {\n return new HasWalkedCache(new Map(this.store))\n }\n hasWalked(target: Path, pattern: Pattern) {\n return this.store.get(target.fullpath())?.has(pattern.globString())\n }\n storeWalked(target: Path, pattern: Pattern) {\n const fullpath = target.fullpath()\n const cached = this.store.get(fullpath)\n if (cached) cached.add(pattern.globString())\n else this.store.set(fullpath, new Set([pattern.globString()]))\n }\n}\n\n/**\n * A record of which paths have been matched in a given walk step,\n * and whether they only are considered a match if they are a directory,\n * and whether their absolute or relative path should be returned.\n */\nexport class MatchRecord {\n store: Map<Path, number> = new Map()\n add(target: Path, absolute: boolean, ifDir: boolean) {\n const n = (absolute ? 2 : 0) | (ifDir ? 1 : 0)\n const current = this.store.get(target)\n this.store.set(target, current === undefined ? n : n & current)\n }\n // match, absolute, ifdir\n entries(): [Path, boolean, boolean][] {\n return [...this.store.entries()].map(([path, n]) => [\n path,\n !!(n & 2),\n !!(n & 1),\n ])\n }\n}\n\n/**\n * A collection of patterns that must be processed in a subsequent step\n * for a given path.\n */\nexport class SubWalks {\n store: Map<Path, Pattern[]> = new Map()\n add(target: Path, pattern: Pattern) {\n if (!target.canReaddir()) {\n return\n }\n const subs = this.store.get(target)\n if (subs) {\n if (!subs.find(p => p.globString() === pattern.globString())) {\n subs.push(pattern)\n }\n } else this.store.set(target, [pattern])\n }\n get(target: Path): Pattern[] {\n const subs = this.store.get(target)\n /* c8 ignore start */\n if (!subs) {\n throw new Error('attempting to walk unknown path')\n }\n /* c8 ignore stop */\n return subs\n }\n entries(): [Path, Pattern[]][] {\n return this.keys().map(k => [k, this.store.get(k) as Pattern[]])\n }\n keys(): Path[] {\n return [...this.store.keys()].filter(t => t.canReaddir())\n }\n}\n\n/**\n * The class that processes patterns for a given path.\n *\n * Handles child entry filtering, and determining whether a path's\n * directory contents must be read.\n */\nexport class Processor {\n hasWalkedCache: HasWalkedCache\n matches = new MatchRecord()\n subwalks = new SubWalks()\n patterns?: Pattern[]\n follow: boolean\n dot: boolean\n opts: GlobWalkerOpts\n\n constructor(opts: GlobWalkerOpts, hasWalkedCache?: HasWalkedCache) {\n this.opts = opts\n this.follow = !!opts.follow\n this.dot = !!opts.dot\n this.hasWalkedCache =\n hasWalkedCache ? hasWalkedCache.copy() : new HasWalkedCache()\n }\n\n processPatterns(target: Path, patterns: Pattern[]) {\n this.patterns = patterns\n const processingSet: [Path, Pattern][] = patterns.map(p => [target, p])\n\n // map of paths to the magic-starting subwalks they need to walk\n // first item in patterns is the filter\n\n for (let [t, pattern] of processingSet) {\n this.hasWalkedCache.storeWalked(t, pattern)\n\n const root = pattern.root()\n const absolute = pattern.isAbsolute() && this.opts.absolute !== false\n\n // start absolute patterns at root\n if (root) {\n t = t.resolve(\n root === '/' && this.opts.root !== undefined ?\n this.opts.root\n : root,\n )\n const rest = pattern.rest()\n if (!rest) {\n this.matches.add(t, true, false)\n continue\n } else {\n pattern = rest\n }\n }\n\n if (t.isENOENT()) continue\n\n let p: MMPattern\n let rest: Pattern | null\n let changed = false\n while (\n typeof (p = pattern.pattern()) === 'string' &&\n (rest = pattern.rest())\n ) {\n const c = t.resolve(p)\n t = c\n pattern = rest\n changed = true\n }\n p = pattern.pattern()\n rest = pattern.rest()\n if (changed) {\n if (this.hasWalkedCache.hasWalked(t, pattern)) continue\n this.hasWalkedCache.storeWalked(t, pattern)\n }\n\n // now we have either a final string for a known entry,\n // more strings for an unknown entry,\n // or a pattern starting with magic, mounted on t.\n if (typeof p === 'string') {\n // must not be final entry, otherwise we would have\n // concatenated it earlier.\n const ifDir = p === '..' || p === '' || p === '.'\n this.matches.add(t.resolve(p), absolute, ifDir)\n continue\n } else if (p === GLOBSTAR) {\n // if no rest, match and subwalk pattern\n // if rest, process rest and subwalk pattern\n // if it's a symlink, but we didn't get here by way of a\n // globstar match (meaning it's the first time THIS globstar\n // has traversed a symlink), then we follow it. Otherwise, stop.\n if (\n !t.isSymbolicLink() ||\n this.follow ||\n pattern.checkFollowGlobstar()\n ) {\n this.subwalks.add(t, pattern)\n }\n const rp = rest?.pattern()\n const rrest = rest?.rest()\n if (!rest || ((rp === '' || rp === '.') && !rrest)) {\n // only HAS to be a dir if it ends in **/ or **/.\n // but ending in ** will match files as well.\n this.matches.add(t, absolute, rp === '' || rp === '.')\n } else {\n if (rp === '..') {\n // this would mean you're matching **/.. at the fs root,\n // and no thanks, I'm not gonna test that specific case.\n /* c8 ignore start */\n const tp = t.parent || t\n /* c8 ignore stop */\n if (!rrest) this.matches.add(tp, absolute, true)\n else if (!this.hasWalkedCache.hasWalked(tp, rrest)) {\n this.subwalks.add(tp, rrest)\n }\n }\n }\n } else if (p instanceof RegExp) {\n this.subwalks.add(t, pattern)\n }\n }\n\n return this\n }\n\n subwalkTargets(): Path[] {\n return this.subwalks.keys()\n }\n\n child() {\n return new Processor(this.opts, this.hasWalkedCache)\n }\n\n // return a new Processor containing the subwalks for each\n // child entry, and a set of matches, and\n // a hasWalkedCache that's a copy of this one\n // then we're going to call\n filterEntries(parent: Path, entries: Path[]): Processor {\n const patterns = this.subwalks.get(parent)\n // put matches and entry walks into the results processor\n const results = this.child()\n for (const e of entries) {\n for (const pattern of patterns) {\n const absolute = pattern.isAbsolute()\n const p = pattern.pattern()\n const rest = pattern.rest()\n if (p === GLOBSTAR) {\n results.testGlobstar(e, pattern, rest, absolute)\n } else if (p instanceof RegExp) {\n results.testRegExp(e, p, rest, absolute)\n } else {\n results.testString(e, p, rest, absolute)\n }\n }\n }\n return results\n }\n\n testGlobstar(\n e: Path,\n pattern: Pattern,\n rest: Pattern | null,\n absolute: boolean,\n ) {\n if (this.dot || !e.name.startsWith('.')) {\n if (!pattern.hasMore()) {\n this.matches.add(e, absolute, false)\n }\n if (e.canReaddir()) {\n // if we're in follow mode or it's not a symlink, just keep\n // testing the same pattern. If there's more after the globstar,\n // then this symlink consumes the globstar. If not, then we can\n // follow at most ONE symlink along the way, so we mark it, which\n // also checks to ensure that it wasn't already marked.\n if (this.follow || !e.isSymbolicLink()) {\n this.subwalks.add(e, pattern)\n } else if (e.isSymbolicLink()) {\n if (rest && pattern.checkFollowGlobstar()) {\n this.subwalks.add(e, rest)\n } else if (pattern.markFollowGlobstar()) {\n this.subwalks.add(e, pattern)\n }\n }\n }\n }\n // if the NEXT thing matches this entry, then also add\n // the rest.\n if (rest) {\n const rp = rest.pattern()\n if (\n typeof rp === 'string' &&\n // dots and empty were handled already\n rp !== '..' &&\n rp !== '' &&\n rp !== '.'\n ) {\n this.testString(e, rp, rest.rest(), absolute)\n } else if (rp === '..') {\n /* c8 ignore start */\n const ep = e.parent || e\n /* c8 ignore stop */\n this.subwalks.add(ep, rest)\n } else if (rp instanceof RegExp) {\n this.testRegExp(e, rp, rest.rest(), absolute)\n }\n }\n }\n\n testRegExp(\n e: Path,\n p: MMRegExp,\n rest: Pattern | null,\n absolute: boolean,\n ) {\n if (!p.test(e.name)) return\n if (!rest) {\n this.matches.add(e, absolute, false)\n } else {\n this.subwalks.add(e, rest)\n }\n }\n\n testString(e: Path, p: string, rest: Pattern | null, absolute: boolean) {\n // should never happen?\n if (!e.isNamed(p)) return\n if (!rest) {\n this.matches.add(e, absolute, false)\n } else {\n this.subwalks.add(e, rest)\n }\n }\n}\n","/**\n * Single-use utility classes to provide functionality to the {@link Glob}\n * methods.\n *\n * @module\n */\nimport { Minipass } from 'minipass'\nimport { Path } from 'path-scurry'\nimport { Ignore, IgnoreLike } from './ignore.js'\n\n// XXX can we somehow make it so that it NEVER processes a given path more than\n// once, enough that the match set tracking is no longer needed? that'd speed\n// things up a lot. Or maybe bring back nounique, and skip it in that case?\n\n// a single minimatch set entry with 1 or more parts\nimport { Pattern } from './pattern.js'\nimport { Processor } from './processor.js'\n\nexport interface GlobWalkerOpts {\n absolute?: boolean\n allowWindowsEscape?: boolean\n cwd?: string | URL\n dot?: boolean\n dotRelative?: boolean\n follow?: boolean\n ignore?: string | string[] | IgnoreLike\n mark?: boolean\n matchBase?: boolean\n // Note: maxDepth here means \"maximum actual Path.depth()\",\n // not \"maximum depth beyond cwd\"\n maxDepth?: number\n nobrace?: boolean\n nocase?: boolean\n nodir?: boolean\n noext?: boolean\n noglobstar?: boolean\n platform?: NodeJS.Platform\n posix?: boolean\n realpath?: boolean\n root?: string\n stat?: boolean\n signal?: AbortSignal\n windowsPathsNoEscape?: boolean\n withFileTypes?: boolean\n includeChildMatches?: boolean\n}\n\nexport type GWOFileTypesTrue = GlobWalkerOpts & {\n withFileTypes: true\n}\nexport type GWOFileTypesFalse = GlobWalkerOpts & {\n withFileTypes: false\n}\nexport type GWOFileTypesUnset = GlobWalkerOpts & {\n withFileTypes?: undefined\n}\n\nexport type Result<O extends GlobWalkerOpts> =\n O extends GWOFileTypesTrue ? Path\n : O extends GWOFileTypesFalse ? string\n : O extends GWOFileTypesUnset ? string\n : Path | string\n\nexport type Matches<O extends GlobWalkerOpts> =\n O extends GWOFileTypesTrue ? Set<Path>\n : O extends GWOFileTypesFalse ? Set<string>\n : O extends GWOFileTypesUnset ? Set<string>\n : Set<Path | string>\n\nexport type MatchStream<O extends GlobWalkerOpts> = Minipass<\n Result<O>,\n Result<O>\n>\n\nconst makeIgnore = (\n ignore: string | string[] | IgnoreLike,\n opts: GlobWalkerOpts,\n): IgnoreLike =>\n typeof ignore === 'string' ? new Ignore([ignore], opts)\n : Array.isArray(ignore) ? new Ignore(ignore, opts)\n : ignore\n\n/**\n * basic walking utilities that all the glob walker types use\n */\nexport abstract class GlobUtil<O extends GlobWalkerOpts = GlobWalkerOpts> {\n path: Path\n patterns: Pattern[]\n opts: O\n seen: Set<Path> = new Set<Path>()\n paused: boolean = false\n aborted: boolean = false\n #onResume: (() => any)[] = []\n #ignore?: IgnoreLike\n #sep: '\\\\' | '/'\n signal?: AbortSignal\n maxDepth: number\n includeChildMatches: boolean\n\n constructor(patterns: Pattern[], path: Path, opts: O)\n constructor(patterns: Pattern[], path: Path, opts: O) {\n this.patterns = patterns\n this.path = path\n this.opts = opts\n this.#sep = !opts.posix && opts.platform === 'win32' ? '\\\\' : '/'\n this.includeChildMatches = opts.includeChildMatches !== false\n if (opts.ignore || !this.includeChildMatches) {\n this.#ignore = makeIgnore(opts.ignore ?? [], opts)\n if (\n !this.includeChildMatches &&\n typeof this.#ignore.add !== 'function'\n ) {\n const m = 'cannot ignore child matches, ignore lacks add() method.'\n throw new Error(m)\n }\n }\n // ignore, always set with maxDepth, but it's optional on the\n // GlobOptions type\n /* c8 ignore start */\n this.maxDepth = opts.maxDepth || Infinity\n /* c8 ignore stop */\n if (opts.signal) {\n this.signal = opts.signal\n this.signal.addEventListener('abort', () => {\n this.#onResume.length = 0\n })\n }\n }\n\n #ignored(path: Path): boolean {\n return this.seen.has(path) || !!this.#ignore?.ignored?.(path)\n }\n #childrenIgnored(path: Path): boolean {\n return !!this.#ignore?.childrenIgnored?.(path)\n }\n\n // backpressure mechanism\n pause() {\n this.paused = true\n }\n resume() {\n /* c8 ignore start */\n if (this.signal?.aborted) return\n /* c8 ignore stop */\n this.paused = false\n let fn: (() => any) | undefined = undefined\n while (!this.paused && (fn = this.#onResume.shift())) {\n fn()\n }\n }\n onResume(fn: () => any) {\n if (this.signal?.aborted) return\n /* c8 ignore start */\n if (!this.paused) {\n fn()\n } else {\n /* c8 ignore stop */\n this.#onResume.push(fn)\n }\n }\n\n // do the requisite realpath/stat checking, and return the path\n // to add or undefined to filter it out.\n async matchCheck(e: Path, ifDir: boolean): Promise<Path | undefined> {\n if (ifDir && this.opts.nodir) return undefined\n let rpc: Path | undefined\n if (this.opts.realpath) {\n rpc = e.realpathCached() || (await e.realpath())\n if (!rpc) return undefined\n e = rpc\n }\n const needStat = e.isUnknown() || this.opts.stat\n const s = needStat ? await e.lstat() : e\n if (this.opts.follow && this.opts.nodir && s?.isSymbolicLink()) {\n const target = await s.realpath()\n /* c8 ignore start */\n if (target && (target.isUnknown() || this.opts.stat)) {\n await target.lstat()\n }\n /* c8 ignore stop */\n }\n return this.matchCheckTest(s, ifDir)\n }\n\n matchCheckTest(e: Path | undefined, ifDir: boolean): Path | undefined {\n return (\n e &&\n (this.maxDepth === Infinity || e.depth() <= this.maxDepth) &&\n (!ifDir || e.canReaddir()) &&\n (!this.opts.nodir || !e.isDirectory()) &&\n (!this.opts.nodir ||\n !this.opts.follow ||\n !e.isSymbolicLink() ||\n !e.realpathCached()?.isDirectory()) &&\n !this.#ignored(e)\n ) ?\n e\n : undefined\n }\n\n matchCheckSync(e: Path, ifDir: boolean): Path | undefined {\n if (ifDir && this.opts.nodir) return undefined\n let rpc: Path | undefined\n if (this.opts.realpath) {\n rpc = e.realpathCached() || e.realpathSync()\n if (!rpc) return undefined\n e = rpc\n }\n const needStat = e.isUnknown() || this.opts.stat\n const s = needStat ? e.lstatSync() : e\n if (this.opts.follow && this.opts.nodir && s?.isSymbolicLink()) {\n const target = s.realpathSync()\n if (target && (target?.isUnknown() || this.opts.stat)) {\n target.lstatSync()\n }\n }\n return this.matchCheckTest(s, ifDir)\n }\n\n abstract matchEmit(p: Result<O>): void\n abstract matchEmit(p: string | Path): void\n\n matchFinish(e: Path, absolute: boolean) {\n if (this.#ignored(e)) return\n // we know we have an ignore if this is false, but TS doesn't\n if (!this.includeChildMatches && this.#ignore?.add) {\n const ign = `${e.relativePosix()}/**`\n this.#ignore.add(ign)\n }\n const abs =\n this.opts.absolute === undefined ? absolute : this.opts.absolute\n this.seen.add(e)\n const mark = this.opts.mark && e.isDirectory() ? this.#sep : ''\n // ok, we have what we need!\n if (this.opts.withFileTypes) {\n this.matchEmit(e)\n } else if (abs) {\n const abs = this.opts.posix ? e.fullpathPosix() : e.fullpath()\n this.matchEmit(abs + mark)\n } else {\n const rel = this.opts.posix ? e.relativePosix() : e.relative()\n const pre =\n this.opts.dotRelative && !rel.startsWith('..' + this.#sep) ?\n '.' + this.#sep\n : ''\n this.matchEmit(!rel ? '.' + mark : pre + rel + mark)\n }\n }\n\n async match(e: Path, absolute: boolean, ifDir: boolean): Promise<void> {\n const p = await this.matchCheck(e, ifDir)\n if (p) this.matchFinish(p, absolute)\n }\n\n matchSync(e: Path, absolute: boolean, ifDir: boolean): void {\n const p = this.matchCheckSync(e, ifDir)\n if (p) this.matchFinish(p, absolute)\n }\n\n walkCB(target: Path, patterns: Pattern[], cb: () => any) {\n /* c8 ignore start */\n if (this.signal?.aborted) cb()\n /* c8 ignore stop */\n this.walkCB2(target, patterns, new Processor(this.opts), cb)\n }\n\n walkCB2(\n target: Path,\n patterns: Pattern[],\n processor: Processor,\n cb: () => any,\n ) {\n if (this.#childrenIgnored(target)) return cb()\n if (this.signal?.aborted) cb()\n if (this.paused) {\n this.onResume(() => this.walkCB2(target, patterns, processor, cb))\n return\n }\n processor.processPatterns(target, patterns)\n\n // done processing. all of the above is sync, can be abstracted out.\n // subwalks is a map of paths to the entry filters they need\n // matches is a map of paths to [absolute, ifDir] tuples.\n let tasks = 1\n const next = () => {\n if (--tasks === 0) cb()\n }\n\n for (const [m, absolute, ifDir] of processor.matches.entries()) {\n if (this.#ignored(m)) continue\n tasks++\n this.match(m, absolute, ifDir).then(() => next())\n }\n\n for (const t of processor.subwalkTargets()) {\n if (this.maxDepth !== Infinity && t.depth() >= this.maxDepth) {\n continue\n }\n tasks++\n const childrenCached = t.readdirCached()\n if (t.calledReaddir())\n this.walkCB3(t, childrenCached, processor, next)\n else {\n t.readdirCB(\n (_, entries) => this.walkCB3(t, entries, processor, next),\n true,\n )\n }\n }\n\n next()\n }\n\n walkCB3(\n target: Path,\n entries: Path[],\n processor: Processor,\n cb: () => any,\n ) {\n processor = processor.filterEntries(target, entries)\n\n let tasks = 1\n const next = () => {\n if (--tasks === 0) cb()\n }\n\n for (const [m, absolute, ifDir] of processor.matches.entries()) {\n if (this.#ignored(m)) continue\n tasks++\n this.match(m, absolute, ifDir).then(() => next())\n }\n for (const [target, patterns] of processor.subwalks.entries()) {\n tasks++\n this.walkCB2(target, patterns, processor.child(), next)\n }\n\n next()\n }\n\n walkCBSync(target: Path, patterns: Pattern[], cb: () => any) {\n /* c8 ignore start */\n if (this.signal?.aborted) cb()\n /* c8 ignore stop */\n this.walkCB2Sync(target, patterns, new Processor(this.opts), cb)\n }\n\n walkCB2Sync(\n target: Path,\n patterns: Pattern[],\n processor: Processor,\n cb: () => any,\n ) {\n if (this.#childrenIgnored(target)) return cb()\n if (this.signal?.aborted) cb()\n if (this.paused) {\n this.onResume(() =>\n this.walkCB2Sync(target, patterns, processor, cb),\n )\n return\n }\n processor.processPatterns(target, patterns)\n\n // done processing. all of the above is sync, can be abstracted out.\n // subwalks is a map of paths to the entry filters they need\n // matches is a map of paths to [absolute, ifDir] tuples.\n let tasks = 1\n const next = () => {\n if (--tasks === 0) cb()\n }\n\n for (const [m, absolute, ifDir] of processor.matches.entries()) {\n if (this.#ignored(m)) continue\n this.matchSync(m, absolute, ifDir)\n }\n\n for (const t of processor.subwalkTargets()) {\n if (this.maxDepth !== Infinity && t.depth() >= this.maxDepth) {\n continue\n }\n tasks++\n const children = t.readdirSync()\n this.walkCB3Sync(t, children, processor, next)\n }\n\n next()\n }\n\n walkCB3Sync(\n target: Path,\n entries: Path[],\n processor: Processor,\n cb: () => any,\n ) {\n processor = processor.filterEntries(target, entries)\n\n let tasks = 1\n const next = () => {\n if (--tasks === 0) cb()\n }\n\n for (const [m, absolute, ifDir] of processor.matches.entries()) {\n if (this.#ignored(m)) continue\n this.matchSync(m, absolute, ifDir)\n }\n for (const [target, patterns] of processor.subwalks.entries()) {\n tasks++\n this.walkCB2Sync(target, patterns, processor.child(), next)\n }\n\n next()\n }\n}\n\nexport class GlobWalker<\n O extends GlobWalkerOpts = GlobWalkerOpts,\n> extends GlobUtil<O> {\n matches = new Set<Result<O>>()\n\n constructor(patterns: Pattern[], path: Path, opts: O) {\n super(patterns, path, opts)\n }\n\n matchEmit(e: Result<O>): void {\n this.matches.add(e)\n }\n\n async walk(): Promise<Set<Result<O>>> {\n if (this.signal?.aborted) throw this.signal.reason\n if (this.path.isUnknown()) {\n await this.path.lstat()\n }\n await new Promise((res, rej) => {\n this.walkCB(this.path, this.patterns, () => {\n if (this.signal?.aborted) {\n rej(this.signal.reason)\n } else {\n res(this.matches)\n }\n })\n })\n return this.matches\n }\n\n walkSync(): Set<Result<O>> {\n if (this.signal?.aborted) throw this.signal.reason\n if (this.path.isUnknown()) {\n this.path.lstatSync()\n }\n // nothing for the callback to do, because this never pauses\n this.walkCBSync(this.path, this.patterns, () => {\n if (this.signal?.aborted) throw this.signal.reason\n })\n return this.matches\n }\n}\n\nexport class GlobStream<\n O extends GlobWalkerOpts = GlobWalkerOpts,\n> extends GlobUtil<O> {\n results: Minipass<Result<O>, Result<O>>\n\n constructor(patterns: Pattern[], path: Path, opts: O) {\n super(patterns, path, opts)\n this.results = new Minipass<Result<O>, Result<O>>({\n signal: this.signal,\n objectMode: true,\n })\n this.results.on('drain', () => this.resume())\n this.results.on('resume', () => this.resume())\n }\n\n matchEmit(e: Result<O>): void {\n this.results.write(e)\n if (!this.results.flowing) this.pause()\n }\n\n stream(): MatchStream<O> {\n const target = this.path\n if (target.isUnknown()) {\n target.lstat().then(() => {\n this.walkCB(target, this.patterns, () => this.results.end())\n })\n } else {\n this.walkCB(target, this.patterns, () => this.results.end())\n }\n return this.results\n }\n\n streamSync(): MatchStream<O> {\n if (this.path.isUnknown()) {\n this.path.lstatSync()\n }\n this.walkCBSync(this.path, this.patterns, () => this.results.end())\n return this.results\n }\n}\n","import { Minimatch } from 'minimatch'\nimport { GlobOptions } from './glob.js'\n\n/**\n * Return true if the patterns provided contain any magic glob characters,\n * given the options provided.\n *\n * Brace expansion is not considered \"magic\" unless the `magicalBraces` option\n * is set, as brace expansion just turns one string into an array of strings.\n * So a pattern like `'x{a,b}y'` would return `false`, because `'xay'` and\n * `'xby'` both do not contain any magic glob characters, and it's treated the\n * same as if you had called it on `['xay', 'xby']`. When `magicalBraces:true`\n * is in the options, brace expansion _is_ treated as a pattern having magic.\n */\nexport const hasMagic = (\n pattern: string | string[],\n options: GlobOptions = {},\n): boolean => {\n if (!Array.isArray(pattern)) {\n pattern = [pattern]\n }\n for (const p of pattern) {\n if (new Minimatch(p, options).hasMagic()) return true\n }\n return false\n}\n","import { escape, unescape } from 'minimatch'\nimport { Minipass } from 'minipass'\nimport { Path } from 'path-scurry'\nimport type {\n GlobOptions,\n GlobOptionsWithFileTypesFalse,\n GlobOptionsWithFileTypesTrue,\n GlobOptionsWithFileTypesUnset,\n} from './glob.js'\nimport { Glob } from './glob.js'\nimport { hasMagic } from './has-magic.js'\n\nexport { escape, unescape } from 'minimatch'\nexport type {\n FSOption,\n Path,\n WalkOptions,\n WalkOptionsWithFileTypesTrue,\n WalkOptionsWithFileTypesUnset,\n} from 'path-scurry'\nexport { Glob } from './glob.js'\nexport type {\n GlobOptions,\n GlobOptionsWithFileTypesFalse,\n GlobOptionsWithFileTypesTrue,\n GlobOptionsWithFileTypesUnset,\n} from './glob.js'\nexport { hasMagic } from './has-magic.js'\nexport { Ignore } from './ignore.js'\nexport type { IgnoreLike } from './ignore.js'\nexport type { MatchStream } from './walker.js'\n\n/**\n * Syncronous form of {@link globStream}. Will read all the matches as fast as\n * you consume them, even all in a single tick if you consume them immediately,\n * but will still respond to backpressure if they're not consumed immediately.\n */\nexport function globStreamSync(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesTrue,\n): Minipass<Path, Path>\nexport function globStreamSync(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesFalse,\n): Minipass<string, string>\nexport function globStreamSync(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesUnset,\n): Minipass<string, string>\nexport function globStreamSync(\n pattern: string | string[],\n options: GlobOptions,\n): Minipass<Path, Path> | Minipass<string, string>\nexport function globStreamSync(\n pattern: string | string[],\n options: GlobOptions = {},\n) {\n return new Glob(pattern, options).streamSync()\n}\n\n/**\n * Return a stream that emits all the strings or `Path` objects and\n * then emits `end` when completed.\n */\nexport function globStream(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesFalse,\n): Minipass<string, string>\nexport function globStream(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesTrue,\n): Minipass<Path, Path>\nexport function globStream(\n pattern: string | string[],\n options?: GlobOptionsWithFileTypesUnset | undefined,\n): Minipass<string, string>\nexport function globStream(\n pattern: string | string[],\n options: GlobOptions,\n): Minipass<Path, Path> | Minipass<string, string>\nexport function globStream(\n pattern: string | string[],\n options: GlobOptions = {},\n) {\n return new Glob(pattern, options).stream()\n}\n\n/**\n * Synchronous form of {@link glob}\n */\nexport function globSync(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesFalse,\n): string[]\nexport function globSync(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesTrue,\n): Path[]\nexport function globSync(\n pattern: string | string[],\n options?: GlobOptionsWithFileTypesUnset | undefined,\n): string[]\nexport function globSync(\n pattern: string | string[],\n options: GlobOptions,\n): Path[] | string[]\nexport function globSync(\n pattern: string | string[],\n options: GlobOptions = {},\n) {\n return new Glob(pattern, options).walkSync()\n}\n\n/**\n * Perform an asynchronous glob search for the pattern(s) specified. Returns\n * [Path](https://isaacs.github.io/path-scurry/classes/PathBase) objects if the\n * {@link withFileTypes} option is set to `true`. See {@link GlobOptions} for\n * full option descriptions.\n */\nasync function glob_(\n pattern: string | string[],\n options?: GlobOptionsWithFileTypesUnset | undefined,\n): Promise<string[]>\nasync function glob_(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesTrue,\n): Promise<Path[]>\nasync function glob_(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesFalse,\n): Promise<string[]>\nasync function glob_(\n pattern: string | string[],\n options: GlobOptions,\n): Promise<Path[] | string[]>\nasync function glob_(\n pattern: string | string[],\n options: GlobOptions = {},\n) {\n return new Glob(pattern, options).walk()\n}\n\n/**\n * Return a sync iterator for walking glob pattern matches.\n */\nexport function globIterateSync(\n pattern: string | string[],\n options?: GlobOptionsWithFileTypesUnset | undefined,\n): Generator<string, void, void>\nexport function globIterateSync(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesTrue,\n): Generator<Path, void, void>\nexport function globIterateSync(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesFalse,\n): Generator<string, void, void>\nexport function globIterateSync(\n pattern: string | string[],\n options: GlobOptions,\n): Generator<Path, void, void> | Generator<string, void, void>\nexport function globIterateSync(\n pattern: string | string[],\n options: GlobOptions = {},\n) {\n return new Glob(pattern, options).iterateSync()\n}\n\n/**\n * Return an async iterator for walking glob pattern matches.\n */\nexport function globIterate(\n pattern: string | string[],\n options?: GlobOptionsWithFileTypesUnset | undefined,\n): AsyncGenerator<string, void, void>\nexport function globIterate(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesTrue,\n): AsyncGenerator<Path, void, void>\nexport function globIterate(\n pattern: string | string[],\n options: GlobOptionsWithFileTypesFalse,\n): AsyncGenerator<string, void, void>\nexport function globIterate(\n pattern: string | string[],\n options: GlobOptions,\n): AsyncGenerator<Path, void, void> | AsyncGenerator<string, void, void>\nexport function globIterate(\n pattern: string | string[],\n options: GlobOptions = {},\n) {\n return new Glob(pattern, options).iterate()\n}\n\n// aliases: glob.sync.stream() glob.stream.sync() glob.sync() etc\nexport const streamSync = globStreamSync\nexport const stream = Object.assign(globStream, { sync: globStreamSync })\nexport const iterateSync = globIterateSync\nexport const iterate = Object.assign(globIterate, {\n sync: globIterateSync,\n})\nexport const sync = Object.assign(globSync, {\n stream: globStreamSync,\n iterate: globIterateSync,\n})\n\nexport const glob = Object.assign(glob_, {\n glob: glob_,\n globSync,\n sync,\n globStream,\n stream,\n globStreamSync,\n streamSync,\n globIterate,\n iterate,\n globIterateSync,\n iterateSync,\n Glob,\n hasMagic,\n escape,\n unescape,\n})\nglob.glob = glob\n","/**\n * Feature ID generation utilities\n * Generates short, opaque identifiers (8-char hashes) for features that comply with IAM tag constraints\n * \n * No human-readable words - just pure technical identifiers for IAM ABAC conditions\n */\n\nimport { nanoid } from 'nanoid'\n\n/**\n * Generate a short, unique feature ID\n * Returns a nanoid-generated identifier - completely opaque and unique\n * \n * @returns Short feature ID (8 chars, IAM tag compliant)\n */\nexport function generateFeatureId(): string {\n // Generate 8-character nanoid - URL-safe, collision-resistant\n return nanoid(8)\n}\n\n\n/**\n * Validate that a feature ID meets IAM tag constraints\n */\nexport function validateFeatureId(featureId: string): boolean {\n // Feature IDs should be exactly 8 characters (nanoid)\n if (featureId.length !== 8) {\n return false\n }\n \n // nanoid uses URL-safe characters: A-Z, a-z, 0-9, -, _\n if (!/^[A-Za-z0-9_-]+$/.test(featureId)) {\n return false\n }\n \n return true\n}\n\n/**\n * Generate a batch of feature IDs ensuring uniqueness within the batch\n */\nexport function generateFeatureIdBatch(featureNames: string[]): Record<string, string> {\n const featureIds: Record<string, string> = {}\n const usedIds = new Set<string>()\n \n for (const featureName of featureNames) {\n let featureId = generateFeatureId()\n let counter = 0\n \n // Ensure uniqueness within batch (extremely rare with nanoid but add safety)\n while (usedIds.has(featureId)) {\n featureId = generateFeatureId()\n counter++\n \n // Prevent infinite loop (should never happen with nanoid)\n if (counter > 100) {\n throw new Error(`Could not generate unique feature ID for: ${featureName}`)\n }\n }\n \n usedIds.add(featureId)\n featureIds[featureName] = featureId\n \n // Validate the generated ID\n if (!validateFeatureId(featureId)) {\n throw new Error(`Generated invalid feature ID: ${featureId} for feature: ${featureName}`)\n }\n }\n \n return featureIds\n}\n","/**\n * ABAC Role Directory Scanner\n * Scans abac-roles/* subdirectories and extracts role-feature mappings from YAML files\n */\n\nimport fs from 'fs'\nimport path from 'path'\nimport { glob } from 'glob'\nimport { parseCloudFormationYaml } from '@hyperdrive.bot/serverless-utils'\nimport type { AbacRoleMetadata, Logger, RegistryFeatureEntry } from '../types.js'\n\nexport class AbacRoleScanner {\n private servicePath: string\n private logger: Logger\n private featureRegistry: Map<string, RegistryFeatureEntry>\n\n constructor(servicePath: string, logger: Logger, featureRegistry: Map<string, RegistryFeatureEntry>) {\n this.servicePath = servicePath\n this.logger = logger\n this.featureRegistry = featureRegistry\n }\n\n /**\n * Scan a module's abac-roles directory and extract role-feature mappings\n * @param modulePath - Absolute path to the module directory\n * @returns Array of AbacRoleMetadata objects\n */\n async scanModuleAbacRoles(modulePath: string): Promise<AbacRoleMetadata[]> {\n const abacRolesDir = path.join(modulePath, 'abac-roles')\n\n // Check if abac-roles directory exists\n if (!fs.existsSync(abacRolesDir)) {\n // Gracefully skip modules without abac-roles\n return []\n }\n\n this.logger.info(`📋 Scanning ABAC roles in: ${path.basename(modulePath)}`)\n\n const results: AbacRoleMetadata[] = []\n\n // Discover all subdirectories (role names)\n const entries = fs.readdirSync(abacRolesDir, { withFileTypes: true })\n const roleDirectories = entries\n .filter(entry => entry.isDirectory())\n .map(entry => entry.name)\n\n if (roleDirectories.length === 0) {\n this.logger.info(' No role directories found in abac-roles/')\n return []\n }\n\n this.logger.info(` Found ${roleDirectories.length} role(s): ${roleDirectories.join(', ')}`)\n\n // Process each role directory\n for (const roleName of roleDirectories) {\n const roleDir = path.join(abacRolesDir, roleName)\n const roleMetadata = await this.processRoleDirectory(roleName, roleDir)\n results.push(...roleMetadata)\n }\n\n return this.sortResults(results)\n }\n\n /**\n * Process a single role directory and extract all module-feature mappings\n */\n private async processRoleDirectory(roleName: string, roleDir: string): Promise<AbacRoleMetadata[]> {\n const results: AbacRoleMetadata[] = []\n\n // Find all YAML files in the role directory\n const yamlFiles = await glob(path.join(roleDir, '*.{yml,yaml}').replace(/\\\\/g, '/'))\n\n if (yamlFiles.length === 0) {\n this.logger.warning(`⚠️ No YAML files found in abac-roles/${roleName}/`)\n return results\n }\n\n this.logger.info(` Processing role '${roleName}' with ${yamlFiles.length} module(s)`)\n\n // Process each YAML file (each file = one module)\n for (const yamlFile of yamlFiles) {\n const moduleName = path.basename(yamlFile, path.extname(yamlFile))\n const metadata = await this.processRoleModuleFile(roleName, moduleName, yamlFile)\n\n if (metadata) {\n results.push(metadata)\n }\n }\n\n return results\n }\n\n /**\n * Process a single role-module YAML file\n */\n private async processRoleModuleFile(\n roleName: string,\n moduleName: string,\n yamlFile: string\n ): Promise<AbacRoleMetadata | null> {\n try {\n // Parse YAML file\n const yamlContent = fs.readFileSync(yamlFile, 'utf8')\n const data = parseCloudFormationYaml(yamlContent) as any\n\n // Extract features array\n if (!data.features) {\n this.logger.warning(`⚠️ No 'features' field in ${roleName}/${moduleName}.yml, skipping`)\n return null\n }\n\n if (!Array.isArray(data.features)) {\n this.logger.warning(`⚠️ 'features' field is not an array in ${roleName}/${moduleName}.yml, skipping`)\n return null\n }\n\n if (data.features.length === 0) {\n this.logger.warning(`⚠️ Empty 'features' array in ${roleName}/${moduleName}.yml, skipping`)\n return null\n }\n\n const featureIds = data.features as string[]\n\n // Validate all feature IDs against registry\n this.validateFeatureIds(featureIds, roleName, moduleName)\n\n this.logger.info(` ✓ ${moduleName}: ${featureIds.length} feature(s)`)\n\n return {\n roleName,\n moduleName,\n featureIds\n }\n } catch (error) {\n // Re-throw validation errors (invalid feature IDs should fail deployment)\n if ((error as Error).message.includes('Invalid feature ID(s)')) {\n throw error\n }\n // Log and skip other errors (YAML parse errors, etc.)\n this.logger.error(`❌ Error processing ${yamlFile}: ${(error as Error).message}`)\n return null\n }\n }\n\n /**\n * Validate that all feature IDs exist in the feature registry\n */\n private validateFeatureIds(featureIds: string[], roleName: string, moduleName: string): void {\n const invalidIds: string[] = []\n const validIds: string[] = []\n\n for (const featureId of featureIds) {\n if (this.featureRegistry.has(featureId)) {\n validIds.push(featureId)\n } else {\n invalidIds.push(featureId)\n }\n }\n\n // If invalid IDs found, throw error with helpful message\n if (invalidIds.length > 0) {\n const availableFeatures = Array.from(this.featureRegistry.keys()).sort()\n const errorMessage = [\n `❌ Invalid feature ID in abac-roles/${roleName}/${moduleName}.yml`,\n '',\n `Invalid feature ID(s): ${invalidIds.join(', ')}`,\n '',\n 'These feature IDs were not found in registry/features/',\n '',\n `Available features (${availableFeatures.length} total):`,\n availableFeatures.join(', '),\n '',\n '💡 Fix: Ensure the feature ID exists in the registry/features/ directory',\n ' Each feature file should have a \\'featureId\\' field matching the ID in this YAML.'\n ].join('\\n')\n\n this.logger.error(errorMessage)\n throw new Error(`Invalid feature ID(s) in abac-roles/${roleName}/${moduleName}.yml: ${invalidIds.join(', ')}`)\n }\n }\n\n /**\n * Sort results by roleName, then moduleName\n */\n private sortResults(results: AbacRoleMetadata[]): AbacRoleMetadata[] {\n return results.sort((a, b) => {\n // First sort by roleName\n const roleCompare = a.roleName.localeCompare(b.roleName)\n if (roleCompare !== 0) return roleCompare\n\n // Then sort by moduleName\n return a.moduleName.localeCompare(b.moduleName)\n })\n }\n\n /**\n * Build feature registry map from feature entries\n * Helper method to create the registry Map required by constructor\n */\n static buildFeatureRegistryMap(featureEntries: RegistryFeatureEntry[]): Map<string, RegistryFeatureEntry> {\n const map = new Map<string, RegistryFeatureEntry>()\n for (const entry of featureEntries) {\n map.set(entry.featureId, entry)\n }\n return map\n }\n}\n","/**\n * Registry Generation Command\n */\n\nimport fs from 'fs'\nimport path from 'path'\nimport { dumpCloudFormationYaml } from '@hyperdrive.bot/serverless-utils'\nimport { AIRegistryGenerator } from '../ai/generator.js'\nimport type { Logger, GenerateRegistryOptions } from '../types.js'\n\nexport class RegistryGenerateCommand {\n private servicePath: string\n private aiGenerator: AIRegistryGenerator\n private logger: Logger\n\n constructor(servicePath: string, logger: Logger) {\n this.servicePath = servicePath\n this.logger = logger\n this.aiGenerator = new AIRegistryGenerator(logger)\n }\n\n /**\n * Execute the registry generation command\n */\n async execute(options: GenerateRegistryOptions): Promise<void> {\n const { module: moduleToGenerate, force = false } = options\n\n if (!moduleToGenerate) {\n throw new Error('Module name is required. Use --module <moduleName> or -m <moduleName>')\n }\n\n try {\n this.logger.info(`Starting AI-powered registry generation for module: ${moduleToGenerate}`)\n this.logger.info(`Force overwrite: ${force ? 'YES' : 'NO'}`)\n\n // Validate module directory exists\n const moduleDir = path.resolve(this.servicePath, 'serverless/modules', moduleToGenerate)\n if (!fs.existsSync(moduleDir)) {\n throw new Error(`Module directory not found: ${moduleDir}`)\n }\n\n const registryDir = path.join(moduleDir, 'registry')\n const featuresDir = path.join(registryDir, 'features')\n\n // Check if registry already exists\n if (fs.existsSync(registryDir) && !force) {\n this.logger.warning(`Registry already exists for module '${moduleToGenerate}'`)\n this.logger.info('Use --force to replace existing registry files')\n return\n }\n\n // Analyze module structure\n this.logger.info('Analyzing module structure...')\n const moduleAnalysis = await this.aiGenerator.analyzeModuleStructure(moduleToGenerate, moduleDir)\n\n // Generate registry using AI\n this.logger.info('Generating registry definitions using AI...')\n const registryData = await this.aiGenerator.generateRegistryWithAI(moduleToGenerate, moduleAnalysis)\n\n // Create registry files\n this.logger.info('Creating registry directory structure...')\n fs.mkdirSync(registryDir, { recursive: true })\n fs.mkdirSync(featuresDir, { recursive: true })\n\n // Write module.yml\n const moduleYmlPath = path.join(registryDir, 'module.yml')\n fs.writeFileSync(moduleYmlPath, dumpCloudFormationYaml(registryData.module))\n this.logger.info(`Generated: ${path.relative(this.servicePath, moduleYmlPath)}`)\n\n // Write feature files\n const generatedFeatures: string[] = []\n for (const [featureName, featureData] of Object.entries(registryData.features)) {\n const featureYmlPath = path.join(featuresDir, `${featureName}.yml`)\n fs.writeFileSync(featureYmlPath, dumpCloudFormationYaml(featureData))\n generatedFeatures.push(path.relative(this.servicePath, featureYmlPath))\n this.logger.info(`Generated: ${path.relative(this.servicePath, featureYmlPath)}`)\n }\n\n // Success summary\n this.logger.info(`Registry generation complete for module '${moduleToGenerate}'!`)\n this.logger.info('Generated files:')\n this.logger.info(` 📄 ${path.relative(this.servicePath, moduleYmlPath)}`)\n generatedFeatures.forEach(featurePath => {\n this.logger.info(` 📄 ${featurePath}`)\n })\n\n this.logger.info('Next steps:')\n this.logger.info(' 1. Review the generated registry files')\n this.logger.info(' 2. Customize the descriptions and metadata as needed')\n this.logger.info(' 3. Deploy your module to register it in DynamoDB')\n } catch (error) {\n this.logger.error(`Registry generation failed: ${(error as Error).message}`)\n throw error\n }\n }\n}","/**\n * AI-powered Registry Generation\n */\n\nimport fs from 'fs'\nimport path from 'path'\nimport { glob } from 'glob'\nimport yaml from 'js-yaml'\nimport OpenAI from 'openai'\nimport { cloudFormationSchema } from '@hyperdrive.bot/serverless-utils'\nimport { generateFeatureIdBatch, validateFeatureId } from '../utils/feature-id.js'\nimport { normalizePath } from '../utils/path-utils.js'\n// Define types locally to avoid build issues with module exports\ninterface Logger {\n info: (message: string) => void\n warning: (message: string) => void\n error: (message: string) => void\n}\n\ninterface FunctionAnalysis {\n handler: string\n events: any[]\n description: string\n endpoints: string[]\n}\n\ninterface ResourceAnalysis {\n type: string\n description: string\n}\n\ninterface ModuleAnalysis {\n moduleName: string\n description: string\n functions: Record<string, FunctionAnalysis>\n resources: Record<string, ResourceAnalysis>\n endpoints: string[]\n dependencies: string[]\n}\n\ninterface RegistryGenerationResult {\n module: {\n name: string\n version: string\n description: string\n maintainer?: string\n category?: string\n tags?: string[]\n }\n features: Record<string, any>\n}\n\nexport interface AIRegistryGeneratorConfig {\n openaiApiKey?: string\n openaiBaseUrl?: string\n openaiModel?: string\n defaultMaintainer?: string\n defaultTags?: string[]\n temperature?: number\n}\n\nexport class AIRegistryGenerator {\n private logger: Logger\n private openai: OpenAI | null = null\n private config: AIRegistryGeneratorConfig\n\n constructor(logger: Logger, config: AIRegistryGeneratorConfig = {}) {\n this.logger = logger\n this.config = {\n openaiBaseUrl: 'https://openrouter.ai/api/v1',\n openaiModel: 'anthropic/claude-3.5-sonnet:beta',\n defaultMaintainer: 'DevSquad Team',\n defaultTags: ['serverless'],\n temperature: 0.1,\n ...config\n }\n this.initializeOpenAI()\n }\n\n private initializeOpenAI(): void {\n const apiKey = this.config.openaiApiKey || process.env.OPENROUTER_API_KEY\n if (apiKey) {\n this.openai = new OpenAI({\n apiKey: apiKey,\n baseURL: this.config.openaiBaseUrl\n })\n }\n }\n\n /**\n * Analyze module structure to gather information for AI generation\n */\n async analyzeModuleStructure(moduleName: string, moduleDir: string): Promise<ModuleAnalysis> {\n const analysis: ModuleAnalysis = {\n moduleName,\n functions: {},\n resources: {},\n endpoints: [],\n dependencies: [],\n description: `Auto-generated registry for ${moduleName} module`\n }\n\n // Analyze functions\n await this.analyzeFunctions(moduleDir, analysis)\n\n // Analyze resources\n await this.analyzeResources(moduleDir, analysis)\n\n return analysis\n }\n\n /**\n * Analyze module functions\n */\n private async analyzeFunctions(moduleDir: string, analysis: ModuleAnalysis): Promise<void> {\n const functionsDir = path.join(moduleDir, 'functions')\n if (!fs.existsSync(functionsDir)) return\n\n const functionFiles = await glob(path.join(functionsDir, '**/*.{yml,yaml}').replace(/\\\\/g, '/'))\n\n for (const functionFile of functionFiles) {\n try {\n const functionData = yaml.load(fs.readFileSync(functionFile, 'utf8'), {\n schema: cloudFormationSchema\n }) as any\n\n if (functionData && typeof functionData === 'object') {\n Object.entries(functionData).forEach(([functionName, functionConfig]: [string, any]) => {\n if (functionConfig && functionConfig.handler) {\n // Validate required description field\n if (!functionConfig.description) {\n throw new Error(\n `Function '${functionName}' in ${functionFile} is missing required 'description' field. ` +\n 'All functions must have descriptions for registry generation.'\n )\n }\n\n // Extract HTTP endpoints with aws_iam authorizer\n const functionEndpoints: string[] = []\n if (functionConfig.events) {\n functionConfig.events.forEach((event: any) => {\n if (event.http && event.http.authorizer === 'aws_iam') {\n const method = event.http.method?.toUpperCase() || 'ANY'\n let path = event.http.path || '/'\n \n // Normalize path to ensure it starts with /\n path = normalizePath(path)\n \n // Normalize path parameters to wildcard patterns\n // Convert {paramName} to * for registry consistency\n path = path.replace(/\\{[^}]+\\}/g, '*')\n \n const endpoint = `${method} - ${path}`\n functionEndpoints.push(endpoint)\n analysis.endpoints.push(endpoint)\n }\n })\n }\n\n analysis.functions[functionName] = {\n handler: functionConfig.handler,\n events: functionConfig.events || [],\n description: functionConfig.description,\n endpoints: functionEndpoints\n }\n }\n })\n }\n } catch (error) {\n this.logger.warning(`Could not parse function file: ${functionFile}`)\n this.logger.warning(`Error: ${(error as Error).message}`)\n }\n }\n }\n\n /**\n * Analyze module resources\n */\n private async analyzeResources(moduleDir: string, analysis: ModuleAnalysis): Promise<void> {\n const resourcesDir = path.join(moduleDir, 'resources')\n if (!fs.existsSync(resourcesDir)) return\n\n const resourceFiles = await glob(path.join(resourcesDir, '**/*.{yml,yaml}').replace(/\\\\/g, '/'))\n\n for (const resourceFile of resourceFiles) {\n try {\n const resourceData = yaml.load(fs.readFileSync(resourceFile, 'utf8'), {\n schema: cloudFormationSchema\n }) as any\n\n if (resourceData && typeof resourceData === 'object') {\n Object.entries(resourceData).forEach(([resourceName, resourceConfig]: [string, any]) => {\n if (resourceConfig && resourceConfig.Type) {\n analysis.resources[resourceName] = {\n type: resourceConfig.Type,\n description: `Resource: ${resourceName} (${resourceConfig.Type})`\n }\n }\n })\n }\n } catch (error) {\n this.logger.warning(`Could not parse resource file: ${resourceFile}`)\n }\n }\n }\n\n\n /**\n * Generate registry data using AI\n */\n async generateRegistryWithAI(\n moduleName: string,\n moduleAnalysis: ModuleAnalysis\n ): Promise<RegistryGenerationResult> {\n if (!this.openai) {\n throw new Error('OPENROUTER_API_KEY environment variable is required for AI generation')\n }\n\n const prompt = this.buildRegistryPrompt(moduleName, moduleAnalysis)\n\n try {\n const result = await this.openai.chat.completions.create({\n model: this.config.openaiModel!,\n messages: [\n {\n role: 'system',\n content: `You are an expert at creating module registry files for serverless applications.\nYou analyze serverless module structures and generate comprehensive registry definitions that describe the module's features, endpoints, and capabilities.\n\nYour output must be valid JSON that can be parsed and converted to YAML files.`\n },\n {\n role: 'user',\n content: prompt\n }\n ],\n max_tokens: 4000,\n temperature: this.config.temperature!\n })\n\n const aiResponse = result.choices[0]?.message?.content\n if (!aiResponse) {\n throw new Error('No response from AI')\n }\n\n // Parse the AI response\n let registryData\n try {\n registryData = JSON.parse(aiResponse)\n } catch (parseError) {\n // Try to extract JSON from the response\n const jsonMatch = aiResponse.match(/\\{[\\s\\S]*\\}/)\n if (jsonMatch) {\n registryData = JSON.parse(jsonMatch[0])\n } else {\n throw new Error('AI response is not valid JSON')\n }\n }\n\n // Validate and enhance the generated data\n return this.validateAndEnhanceRegistryData(registryData, moduleAnalysis)\n } catch (error) {\n this.logger.warning(`AI generation failed: ${(error as Error).message}`)\n throw error\n }\n }\n\n /**\n * Build the comprehensive prompt for AI registry generation\n */\n private buildRegistryPrompt(moduleName: string, analysis: ModuleAnalysis): string {\n const functionsWithEndpoints = Object.entries(analysis.functions)\n .filter(([, func]) => func.endpoints && func.endpoints.length > 0)\n\n return `Analyze this serverless module and create registry files for it.\n\n**Module Name:** ${moduleName}\n**Description:** ${analysis.description}\n\n**Functions with API Endpoints (${functionsWithEndpoints.length}):**\n${functionsWithEndpoints\n .map(([name, func]) => {\n return `- ${name}: ${func.description}\\n Endpoints: ${func.endpoints.join(', ')}`\n })\n .join('\\n')}\n\nPlease analyze the module's functions and endpoints to infer the appropriate category and generate a registry structure with ULTRA-GRANULAR, ATOMIC, COMPOSABLE features following enterprise role-based access control patterns.\n\n**CRITICAL GRANULARITY REQUIREMENTS:**\n\n**1. FEATURE ATOMICITY - Each feature must be:**\n- ONE SINGLE business capability that cannot be subdivided further\n- ONE checkbox in admin interface = ONE specific business action\n- NEVER combine multiple business operations (create AND update = separate features)\n- NEVER group different data types (users AND organizations = separate features)\n- NEVER merge different risk levels (view AND delete = separate features)\n\n**2. ULTRA-GRANULAR FEATURE NAMING PATTERN:**\n\\`{verb}-{noun}-{qualifier}-{sensitivity}\\`\n\n**Verb Categories (be extremely specific):**\n- **Read Operations:** \\`visualizar\\`, \\`listar\\`, \\`buscar\\`, \\`consultar\\`, \\`exportar\\`, \\`baixar\\`\n- **Create Operations:** \\`criar\\`, \\`registrar\\`, \\`adicionar\\`, \\`iniciar\\`, \\`gerar\\`\n- **Update Operations:** \\`editar\\`, \\`atualizar\\`, \\`modificar\\`, \\`alterar\\`, \\`configurar\\`\n- **Delete Operations:** \\`excluir\\`, \\`remover\\`, \\`cancelar\\`, \\`desativar\\`, \\`arquivar\\`\n- **Admin Operations:** \\`gerenciar\\`, \\`administrar\\`, \\`configurar-sistema\\`, \\`aprovar\\`, \\`rejeitar\\`\n- **Process Operations:** \\`processar\\`, \\`executar\\`, \\`calcular\\`, \\`sincronizar\\`, \\`validar\\`\n\n**Noun Categories (be ultra-specific):**\n- Break down by exact data type: \\`usuarios\\`, \\`organizacoes\\`, \\`transacoes\\`, \\`relatorios\\`, \\`configuracoes\\`\n- Include data subtypes: \\`usuarios-ativos\\`, \\`transacoes-financeiras\\`, \\`relatorios-auditoria\\`\n- Separate by business context: \\`dados-pessoais\\`, \\`dados-financeiros\\`, \\`dados-operacionais\\`\n\n**Qualifier Examples:**\n- \\`proprios\\` (own data), \\`equipe\\` (team data), \\`organizacao\\` (org data), \\`sistema\\` (system data)\n- \\`basicos\\` (basic info), \\`completos\\` (full info), \\`sensiveis\\` (sensitive info)\n- \\`ativos\\` (active), \\`historicos\\` (historical), \\`arquivados\\` (archived)\n\n**3. HYPER-GRANULAR ENDPOINT GROUPING RULES:**\n- **DEFAULT: ONE endpoint per feature** for maximum granularity\n- **ONLY group 2 endpoints** if they are TRULY COMPLEMENTARY (e.g., GET /users/:id + PUT /users/:id for the exact same data scope and business permission)\n- **NEVER group different HTTP methods** unless they represent the exact same business permission with identical risk profile\n- **SPLIT by data sensitivity:** viewing basic info ≠ viewing sensitive info\n- **SPLIT by scope:** own data ≠ team data ≠ organization data\n- **SPLIT by business impact:** operational changes ≠ financial changes ≠ compliance changes\n- **SPLIT by approval requirements:** self-service ≠ manager approval ≠ admin approval\n- **COMPLEMENTARY EXAMPLES:**\n - ✅ GET /user/profile + PUT /user/profile (same data, same scope)\n - ✅ POST /document + GET /document/:id (create and immediately view own creation)\n - ❌ GET /users + POST /users (list all vs create new - different scopes)\n - ❌ GET /reports + DELETE /reports (read vs delete - different risk levels)\n\n**4. ULTRA-GRANULAR RISK ASSESSMENT MATRIX:**\n\n**CRITICAL RISK (requires multiple approvals):**\n- Delete any user data, financial records, or audit logs\n- System configuration changes affecting multiple users\n- Access to payment processing, banking data, or financial controls\n- User role/permission management and privilege escalation\n- Bulk operations affecting 100+ records\n- Integration with external financial/payment systems\n\n**HIGH RISK (requires manager approval):**\n- Create/update financial transactions or accounting data\n- Access to personally identifiable information (PII)\n- Modify user accounts, roles, or organizational structure\n- Generate compliance reports or export sensitive data\n- Configure business rules, workflows, or approval processes\n- Access to audit logs or security-related data\n\n**MEDIUM RISK (requires team lead approval):**\n- Create/update operational data affecting multiple users\n- Modify team-level configurations or processes\n- Access to aggregated business metrics or reports\n- Process workflows that change business state\n- Import/export non-sensitive business data\n- Configure notifications or user preferences\n\n**LOW RISK (self-service):**\n- View own data, basic lists, or public information\n- Update own profile or non-critical personal settings\n- Basic search and filtering operations\n- View aggregated, anonymized statistics\n- Generate non-sensitive reports for own data\n\n**5. FEATURE SPLITTING DECISION TREE:**\nAsk these questions for EVERY endpoint group:\n- Can these endpoints be used by different job roles? → SPLIT\n- Do they access different types of sensitive data? → SPLIT \n- Do they require different approval levels? → SPLIT\n- Do they have different business impact levels? → SPLIT\n- Would an admin want to grant one without the other? → SPLIT\n- Do they serve different organizational workflows? → SPLIT\n\n**6. REAL-WORLD GRANULARITY EXAMPLES:**\n\n**BAD (too broad):**\n- \\`gerenciar-usuarios\\` (manages all user operations)\n- \\`acessar-relatorios\\` (accesses all reports)\n\n**GOOD (ultra-granular):**\n- \\`visualizar-usuarios-basicos-proprios\\`\n- \\`editar-usuarios-dados-pessoais-equipe\\`\n- \\`excluir-usuarios-inativos-organizacao\\`\n- \\`gerar-relatorios-financeiros-mensais-equipe\\`\n- \\`exportar-dados-auditoria-sensiveis-sistema\\`\n\n**7. OUTPUT REQUIREMENTS:**\n- **MAXIMUM: 19 features per module** (safety limit for IAM policy constraints)\n- Minimum 8-15 features per module (unless genuinely simple)\n- **DEFAULT: 1 endpoint per feature** (maximum granularity)\n- **MAXIMUM: 5 endpoints per feature** ONLY if truly complementary\n- Each feature description must explain the exact business capability\n- Feature names must be intuitive to non-technical administrators\n- All text must be in Portuguese (pt-BR)\n- Features must be ordered by risk level (critical → low)\n\n**CRITICAL: The super-admin feature will be auto-added, making the total 20 features. Generate exactly 19 features maximum to stay within IAM policy limits.**\n\n**Output Format:**\nReturn valid JSON in this exact structure:\n\\`\\`\\`json\n{\n \"module\": {\n \"name\": \"module-name\",\n \"description\": \"Clear, detailed description of what this module does\",\n \"category\": \"inferred-category-from-analysis\",\n \"tags\": [\"inferred-tags\", \"based-on-functionality\"]\n },\n \"features\": {\n \"ultra-granular-feature-key\": {\n \"name\": \"Nome Específico da Capacidade\",\n \"description\": \"Descrição detalhada e intuitiva do que essa capacidade permite fazer no negócio\",\n \"endpoints\": [\"GET - /specific/path\"]\n }\n }\n}\n\\`\\`\\`\n\nGenerate ultra-granular, atomic features that enable precise role composition for complex organizational hierarchies and compliance requirements.`\n }\n\n /**\n * Generate super-admin feature for apigateway module\n */\n private generateSuperAdminFeature(): any {\n return {\n name: 'Super Administrador do Módulo',\n version: '1.0.0',\n description: 'Acesso completo e irrestrito a todas as funcionalidades do módulo, incluindo operações de criação, leitura, atualização e exclusão',\n endpoints: [\n 'GET - /*',\n 'POST - /*',\n 'DELETE - /*',\n 'PATCH - /*',\n 'PUT - /*'\n ],\n customPolicies: []\n }\n }\n\n /**\n * Validate and enhance AI-generated registry data\n */\n private validateAndEnhanceRegistryData(\n registryData: any,\n analysis: ModuleAnalysis\n ): RegistryGenerationResult {\n // Ensure required structure exists\n if (!registryData.module) registryData.module = {}\n if (!registryData.features) registryData.features = {}\n\n // Validate and fill module data\n registryData.module = {\n name: registryData.module.name || analysis.moduleName,\n version: registryData.module.version || '1.0.0',\n description: registryData.module.description || analysis.description,\n maintainer: registryData.module.maintainer || this.config.defaultMaintainer,\n category: registryData.module.category, // AI-inferred category\n tags: registryData.module.tags || [...this.config.defaultTags!].filter(Boolean)\n }\n\n // Clean up and validate features - remove redundant module prefixes\n const cleanedFeatures: Record<string, any> = {}\n const featureNames: string[] = []\n \n // First pass: collect and clean feature keys\n Object.keys(registryData.features).forEach(featureKey => {\n const feature = registryData.features[featureKey]\n \n // Skip features without endpoints\n if (!feature.endpoints || !Array.isArray(feature.endpoints) || feature.endpoints.length === 0) {\n this.logger.warning(`Skipping feature '${featureKey}' - no endpoints defined`)\n return\n }\n \n // Remove redundant module prefix from feature key\n const cleanFeatureKey = this.cleanFeatureKey(featureKey, analysis.moduleName)\n featureNames.push(cleanFeatureKey)\n \n // Update feature data\n cleanedFeatures[cleanFeatureKey] = {\n name: feature.name || cleanFeatureKey,\n version: feature.version || '1.0.0',\n description: feature.description || `${cleanFeatureKey} feature`,\n endpoints: feature.endpoints,\n customPolicies: []\n }\n })\n \n // Add super-admin feature for module\n this.logger.info('Adding super-admin feature for module')\n const superAdminKey = 'super-admin'\n featureNames.push(superAdminKey)\n cleanedFeatures[superAdminKey] = this.generateSuperAdminFeature()\n this.logger.info('Added super-admin feature for module')\n \n // Generate feature IDs for all features\n this.logger.info(`Generating feature IDs for ${featureNames.length} features`)\n const featureIds = generateFeatureIdBatch(featureNames)\n \n // Add feature IDs to all features\n Object.keys(cleanedFeatures).forEach(featureKey => {\n const featureId = featureIds[featureKey]\n if (featureId) {\n cleanedFeatures[featureKey].featureId = featureId\n this.logger.info(` ✓ ${featureKey} → ${featureId}`)\n \n // Validate the feature ID\n if (!validateFeatureId(featureId)) {\n throw new Error(`Invalid feature ID generated: ${featureId} for feature: ${featureKey}`)\n }\n } else {\n this.logger.warning(`No feature ID generated for: ${featureKey}`)\n }\n })\n \n registryData.features = cleanedFeatures\n return registryData\n }\n\n /**\n * Clean feature key by removing redundant module prefixes\n */\n private cleanFeatureKey(featureKey: string, moduleName: string): string {\n // Remove module prefix if it exists (case-insensitive)\n const modulePrefix = moduleName.toLowerCase()\n const lowerFeatureKey = featureKey.toLowerCase()\n \n if (lowerFeatureKey.startsWith(`${modulePrefix}-`)) {\n return featureKey.substring(modulePrefix.length + 1)\n }\n \n return featureKey\n }\n}","/**\n * Service Package Generation Command\n */\n\nimport fs from 'fs'\nimport path from 'path'\nimport type { ModuleRegistryConfig, Logger, GeneratePackageOptions } from '../types.js'\n\nexport class PackageGenerateCommand {\n private config: ModuleRegistryConfig\n private logger: Logger\n\n constructor(config: ModuleRegistryConfig, logger: Logger) {\n this.config = config\n this.logger = logger\n }\n\n /**\n * Execute the package generation command\n */\n async execute(options: GeneratePackageOptions = {}): Promise<void> {\n const { force = false, verbose = false } = options\n\n this.logger.info('Generating Module Registry service package...')\n this.logger.info(`Force regeneration: ${force ? 'YES' : 'NO'}`)\n\n try {\n // Check if files already exist and force is not set\n const moduleRegistryDir = path.join(process.cwd(), 'node_modules', 'module-registry')\n const servicePath = path.join(moduleRegistryDir, this.config.servicePath!)\n\n if (fs.existsSync(servicePath) && !force) {\n this.logger.warning(`Service package already exists at ${servicePath}`)\n this.logger.info('Use --force to regenerate existing files')\n this.logger.info('Service package generation skipped')\n return\n }\n\n // Generate the service files\n await this.generateServiceFile()\n await this.generateEnvironmentFile()\n\n this.logger.info('Module Registry service package generated successfully!')\n this.logger.info('You can now use: import * as ModuleRegistryService from \"module-registry/service\"')\n this.logger.info(`Generated files in: ${moduleRegistryDir}`)\n this.logger.info(' 📄 service.js - Service functions with auto-config')\n this.logger.info(' 📄 service.d.ts - TypeScript definitions')\n this.logger.info(' 📄 env.js - Environment configuration')\n this.logger.info(' 📄 package.json - Package manifest')\n this.logger.info('')\n this.logger.info('Configuration automatically set:')\n this.logger.info(` 📋 Table: ${this.config.tableName}`)\n this.logger.info(` 🌍 Region: ${this.config.region}`)\n this.logger.info(' 🔧 No manual setup required!')\n } catch (error: any) {\n this.logger.error(`Failed to generate service package: ${error.message}`)\n throw error\n }\n }\n\n /**\n * Generate service file with re-exports from the actual service module\n */\n private async generateServiceFile(): Promise<void> {\n if (!this.config.generateService) {\n this.logger.info('Service file generation is disabled')\n return\n }\n\n const serverlessDir = process.cwd()\n const nodeModulesDir = path.join(serverlessDir, 'node_modules')\n const moduleRegistryDir = path.join(nodeModulesDir, 'module-registry')\n\n // Ensure the node_modules/module-registry directory exists\n if (!fs.existsSync(moduleRegistryDir)) {\n fs.mkdirSync(moduleRegistryDir, { recursive: true })\n }\n\n const servicePath = path.join(moduleRegistryDir, this.config.servicePath!)\n\n // Generate the service file content\n const serviceContent = this.generateServiceFileContent()\n const serviceTypesContent = this.generateServiceTypesContent()\n\n // Write the JavaScript file\n fs.writeFileSync(servicePath, serviceContent, 'utf8')\n\n // Generate and write the TypeScript definitions file\n const serviceTypesPath = servicePath.replace('.js', '.d.ts')\n fs.writeFileSync(serviceTypesPath, serviceTypesContent, 'utf8')\n\n // Generate the actual service.js file with injected configuration\n const actualServicePath = path.join(moduleRegistryDir, 'service.js')\n const actualServiceContent = this.generateActualServiceFile()\n fs.writeFileSync(actualServicePath, actualServiceContent, 'utf8')\n\n // Copy the service type definitions\n await this.copyServiceTypes(moduleRegistryDir)\n\n // Create/update package.json for the module-registry directory\n await this.generatePackageJson(moduleRegistryDir)\n\n this.logger.info(`Generated service files: ${this.config.servicePath} and TypeScript definitions`)\n }\n\n /**\n * Generate service file content with re-exports\n */\n private generateServiceFileContent(): string {\n return `// Auto-generated by Module Registry Plugin\n// This file is generated during serverless deployment and should not be manually edited\n\n// Import environment configuration to ensure proper setup\nimport './env.js'\n\nexport {\n listAllModules,\n getModuleFeatures,\n getFeatureDetails,\n getFeatureById,\n getModuleMetadata,\n getAllEndpoints,\n createModuleRegistryLogger\n} from './service.js'\n\n// Default export for compatibility\nimport * as service from './service.js'\nexport default service\n`\n }\n\n /**\n * Generate the actual service.js file with configuration injected\n */\n private generateActualServiceFile(): string {\n // Read the template service.js file\n let templatePath = path.join(__dirname, '..', 'service.js')\n if (!fs.existsSync(templatePath)) {\n templatePath = path.join(__dirname, '..', 'src', 'service.js')\n }\n\n const templateContent = fs.readFileSync(templatePath, 'utf8')\n\n // Inject the resolved configuration values\n const injectedContent = templateContent.replace(\n 'placeholder-table-name',\n this.config.tableName || 'api-dev-module-registry'\n )\n\n return `// Auto-generated by Module Registry Plugin\n// Service functions with build-time configuration injection\n// DO NOT EDIT MANUALLY - Generated from service.js template\n\n${injectedContent}\n`\n }\n\n /**\n * Generate TypeScript definitions for service functions\n */\n private generateServiceTypesContent(): string {\n return `// Auto-generated type definitions for Module Registry service\n// This file provides TypeScript type definitions for the service exports\n\nexport declare function listAllModules(): Promise<ModuleInfo[]>\nexport declare function getModuleFeatures(moduleName: string): Promise<FeatureInfo[]>\nexport declare function getFeatureDetails(moduleName: string, featureId: string): Promise<FeatureDetails | null>\nexport declare function getFeatureById(featureId: string): Promise<FeatureDetails | null>\nexport declare function getModuleMetadata(moduleName: string): Promise<ModuleInfo | null>\nexport declare function getAllEndpoints(): Promise<EndpointInfo[]>\nexport declare function createModuleRegistryLogger(context: string): Logger\n\n// Type exports - matching actual service.ts types\nexport interface ModuleInfo {\n moduleName: string\n description: string\n version: string\n maintainer?: string\n tags: string[]\n lastUpdated: string\n}\n\nexport interface FeatureInfo {\n featureName: string\n description: string\n version: string\n endpoints: string[]\n customPolicies: any[]\n endpointCount: number\n lastUpdated: string\n}\n\nexport interface FeatureDetails extends FeatureInfo {\n moduleName: string\n policyCount: number\n}\n\nexport interface EndpointInfo {\n endpoint: string\n moduleName: string\n featureName: string\n featureDescription: string\n version: string\n}\n\nexport interface Logger {\n info: (message: string, data?: any) => void\n warn: (message: string, data?: any) => void\n error: (message: string, error?: any) => void\n}\n\n// Default export interface\nexport interface ModuleRegistryService {\n listAllModules: typeof listAllModules\n getModuleFeatures: typeof getModuleFeatures\n getFeatureDetails: typeof getFeatureDetails\n getFeatureById: typeof getFeatureById\n getModuleMetadata: typeof getModuleMetadata\n getAllEndpoints: typeof getAllEndpoints\n createModuleRegistryLogger: typeof createModuleRegistryLogger\n}\n\ndeclare const moduleRegistryService: ModuleRegistryService\nexport default moduleRegistryService\n`\n }\n\n /**\n * Copy service type definitions\n */\n private async copyServiceTypes(moduleRegistryDir: string): Promise<void> {\n const sourceServiceTypesPath = path.join(__dirname, '..', 'service.d.ts')\n const targetServiceTypesPath = path.join(moduleRegistryDir, 'service.d.ts')\n\n if (fs.existsSync(sourceServiceTypesPath)) {\n fs.copyFileSync(sourceServiceTypesPath, targetServiceTypesPath)\n } else {\n // Try to read from src directory\n const srcServiceTypesPath = path.join(__dirname, '..', 'src', 'service.d.ts')\n if (fs.existsSync(srcServiceTypesPath)) {\n fs.copyFileSync(srcServiceTypesPath, targetServiceTypesPath)\n }\n }\n }\n\n /**\n * Generate package.json for the virtual module\n */\n private async generatePackageJson(moduleRegistryDir: string): Promise<void> {\n const packageJsonPath = path.join(moduleRegistryDir, 'package.json')\n const packageJsonContent = JSON.stringify(\n {\n name: 'module-registry',\n version: '1.0.0',\n description: 'Generated Module Registry service functions',\n main: 'service.js',\n types: 'service.d.ts',\n exports: {\n '.': {\n types: './service.d.ts',\n default: './service.js'\n },\n './service': {\n types: './service.d.ts',\n default: './service.js'\n },\n './env': {\n default: './env.js'\n }\n }\n },\n null,\n 2\n )\n fs.writeFileSync(packageJsonPath, packageJsonContent, 'utf8')\n }\n\n /**\n * Generate environment configuration file for runtime service usage\n */\n private async generateEnvironmentFile(): Promise<void> {\n if (!this.config.generateService) {\n return // Skip if service generation is disabled\n }\n\n try {\n const moduleRegistryDir = path.join(process.cwd(), 'node_modules', 'module-registry')\n const envPath = path.join(moduleRegistryDir, 'env.js')\n\n const envContent = `// Auto-generated environment configuration for Module Registry\n// This file provides runtime configuration for service functions\n\n// Set environment variables for service functions\nprocess.env.MODULE_REGISTRY_TABLE_NAME = '${this.config.tableName}'\nprocess.env.AWS_REGION = process.env.AWS_REGION || '${this.config.region}'\n\n// Export configuration for direct use\nexport const moduleRegistryConfig = {\n tableName: '${this.config.tableName}',\n region: '${this.config.region}',\n policyPrefix: '${this.config.policyPrefix}'\n}\n\nexport default moduleRegistryConfig\n`\n\n fs.writeFileSync(envPath, envContent, 'utf8')\n this.logger.info('Generated environment configuration: env.js')\n } catch (error: any) {\n this.logger.error(`Failed to generate environment file: ${error.message}`)\n // Don't throw - this is optional\n }\n }\n}","/**\n * Plugin logging and debugging utilities\n */\n\nimport type { Logger, ModuleRegistryConfig, RegistryModuleEntry, RegistryFeatureEntry } from '../types.js'\n\n// Debug: Track plugin instances and hook calls\nlet pluginInstanceCounter = 0\nconst hookCallCounter = new Map<string, number>()\n\nexport class PluginLogger {\n private logger: Logger\n private instanceId: number\n\n constructor(logger: Logger) {\n this.logger = logger\n this.instanceId = ++pluginInstanceCounter\n console.log(`[module-registry] Plugin instance ${this.instanceId} created`)\n }\n\n getInstanceId(): number {\n return this.instanceId\n }\n\n trackHookCall(hookName: string): void {\n const currentCount = (hookCallCounter.get(hookName) || 0) + 1\n hookCallCounter.set(hookName, currentCount)\n console.log(`[module-registry] Hook ${hookName} called ${currentCount} time(s) from instance ${this.instanceId}`)\n }\n\n logInitialConfiguration(config: ModuleRegistryConfig, isModuleDeployment: boolean): void {\n this.logger.info('Module Registry plugin initialized:')\n this.logger.info(` Table: ${config.tableName}`)\n this.logger.info(` Region: ${config.region}`)\n this.logger.info(` Skip DynamoDB: ${config.skipDynamoDB}`)\n this.logger.info(` Skip Policies: ${config.skipPolicies}`)\n this.logger.info(` Policy Prefix: ${config.policyPrefix}`)\n this.logger.info(` Generate Service: ${config.generateService ? '✅ ENABLED' : '❌ DISABLED'}`)\n this.logger.info(` Service Path: ${config.servicePath}`)\n this.logger.info(` Strict Mode: ${config.strict ? '🔒 ENABLED' : '✅ DISABLED'}`)\n this.logger.info(` Module Deployment: ${isModuleDeployment ? '🎯 TARGETED (single module)' : '🌐 FULL (all modules)'}`)\n this.logger.info(' 📋 Variables will be resolved during processing...')\n }\n\n displayRegistrySummary(\n config: ModuleRegistryConfig,\n moduleEntries: RegistryModuleEntry[],\n featureEntries: RegistryFeatureEntry[]\n ): void {\n if (moduleEntries.length === 0 && featureEntries.length === 0) {\n return\n }\n\n // Module summary\n if (moduleEntries.length > 0) {\n const module = moduleEntries[0]\n this.logger.info('┌─ Module Registry ─')\n this.logger.info(`│ Module: ${module.moduleName}`)\n this.logger.info(`│ Version: ${module.version}`)\n this.logger.info(`│ Description: ${module.description}`)\n this.logger.info(`│ Table: ${config.tableName}`)\n this.logger.info('└─')\n }\n\n // Features summary\n if (featureEntries.length > 0) {\n this.logger.info(`┌─ Features (${featureEntries.length}) ─`)\n\n // Group features by module for cleaner display\n const featuresByModule: Record<string, RegistryFeatureEntry[]> = {}\n featureEntries.forEach(feature => {\n if (!featuresByModule[feature.moduleName]) {\n featuresByModule[feature.moduleName] = []\n }\n featuresByModule[feature.moduleName].push(feature)\n })\n\n Object.entries(featuresByModule).forEach(([, features]) => {\n features.forEach(feature => {\n this.logger.info(`│ ${feature.featureName}: (${feature.endpoints.length} endpoints)`)\n })\n })\n\n this.logger.info('└─')\n }\n\n // Registry stats\n this.logger.info('┌─ Registry Statistics ─')\n this.logger.info(`│ Modules: ${moduleEntries.length}`)\n this.logger.info(`│ Features: ${featureEntries.length}`)\n this.logger.info(`│ Total Endpoints: ${featureEntries.reduce((sum, feature) => sum + feature.endpoints.length, 0)}`)\n\n // Count features with policy ARNs\n const featuresWithPolicies = featureEntries.filter(f => f.policyArn).length\n const featuresWithResolvedArns = featureEntries.filter(f =>\n typeof f.policyArn === 'string' && f.policyArn.startsWith('arn:aws:iam::')\n ).length\n\n this.logger.info(`│ Generated Policies: ${featuresWithPolicies}`)\n if (featuresWithResolvedArns > 0) {\n this.logger.info(`│ Resolved Policy ARNs: ${featuresWithResolvedArns}`)\n }\n this.logger.info(`│ Service Package: ${config.generateService ? '✅ Generated' : '❌ Disabled'}`)\n this.logger.info(`│ DynamoDB Table: ${config.tableName}`)\n this.logger.info(`│ Region: ${config.region}`)\n this.logger.info('└─')\n }\n}","/**\n * ABAC Metadata DynamoDB Writer\n * Writes role+module+feature mappings to DynamoDB with correct schema for Epic 3 consumption\n */\n\nimport type { AbacRoleMetadata, Logger, DynamoDBItem, BatchOperation } from '../types.js'\nimport type { DynamoDBManager } from './dynamodb.js'\n\nexport class AbacMetadataWriter {\n private dynamodbManager: DynamoDBManager\n private logger: Logger\n\n constructor(dynamodbManager: DynamoDBManager, logger: Logger) {\n this.dynamodbManager = dynamodbManager\n this.logger = logger\n }\n\n /**\n * Write ABAC metadata to DynamoDB\n * @param metadata - Array of AbacRoleMetadata from AbacRoleScanner\n */\n async writeAbacMetadata(metadata: AbacRoleMetadata[]): Promise<void> {\n // Return early if metadata array is empty (no-op)\n if (metadata.length === 0) {\n this.logger.info('No ABAC metadata to write')\n return\n }\n\n this.logger.info(`Processing ${metadata.length} ABAC metadata object(s)`)\n\n // Transform metadata to DynamoDB items\n const items = this.transformToDynamoDBItems(metadata)\n this.logger.info(`Transformed to ${items.length} DynamoDB item(s)`)\n\n // Clean up stale items before writing new ones\n await this.cleanupStaleAbacItems(metadata)\n\n // Write items to DynamoDB\n const batchCount = Math.ceil(items.length / 25)\n this.logger.info(`Writing ${items.length} item(s) in ${batchCount} batch(es)`)\n\n const operations: BatchOperation[] = items.map(item => ({\n operation: 'put',\n item\n }))\n\n await this.dynamodbManager.batchWrite(operations)\n\n this.logger.info('ABAC metadata written successfully')\n }\n\n /**\n * Transform AbacRoleMetadata array to DynamoDB items\n * One metadata object with N featureIds creates N DynamoDB items\n */\n private transformToDynamoDBItems(metadata: AbacRoleMetadata[]): DynamoDBItem[] {\n const items: DynamoDBItem[] = []\n\n for (const meta of metadata) {\n // Create one item per featureId\n for (const featureId of meta.featureIds) {\n items.push({\n pk: `ROLE#${meta.roleName}`,\n sk: `MODULE#${meta.moduleName}#FEATURE#${featureId}`,\n gsi1pk: 'ROLES',\n gsi1sk: `ROLE#${meta.roleName}`,\n itemType: 'abac-role' as const\n })\n }\n }\n\n return items\n }\n\n /**\n * Clean up stale ABAC items that no longer exist in current metadata\n */\n private async cleanupStaleAbacItems(currentMetadata: AbacRoleMetadata[]): Promise<void> {\n // Get all unique role names from current metadata\n const currentRoleNames = [...new Set(currentMetadata.map(m => m.roleName))]\n\n // Build set of current items for quick lookup\n const currentItemKeys = new Set<string>()\n for (const meta of currentMetadata) {\n for (const featureId of meta.featureIds) {\n const key = `ROLE#${meta.roleName}::MODULE#${meta.moduleName}#FEATURE#${featureId}`\n currentItemKeys.add(key)\n }\n }\n\n // Query existing items for each role and identify stale ones\n const itemsToDelete: DynamoDBItem[] = []\n\n for (const roleName of currentRoleNames) {\n const existingItems = await this.dynamodbManager.queryAbacRoleEntries(roleName)\n\n for (const existingItem of existingItems) {\n const itemKey = `${existingItem.pk}::${existingItem.sk}`\n\n if (!currentItemKeys.has(itemKey)) {\n itemsToDelete.push({\n pk: existingItem.pk,\n sk: existingItem.sk,\n itemType: 'abac-role' as const\n })\n }\n }\n }\n\n // Delete stale items\n if (itemsToDelete.length > 0) {\n this.logger.info(`Cleaning up ${itemsToDelete.length} stale ABAC metadata item(s)`)\n const deleteOperations: BatchOperation[] = itemsToDelete.map(item => ({\n operation: 'delete',\n item\n }))\n await this.dynamodbManager.batchWrite(deleteOperations)\n }\n }\n}\n","/**\n * Plugin orchestration logic\n */\n\nimport type { ModuleRegistryConfig, RegistryModuleEntry, RegistryFeatureEntry, AbacRoleMetadata, Logger } from '../types.js'\nimport { DynamoDBManager } from './dynamodb.js'\nimport { PolicyGenerator } from './policy-generator.js'\nimport { AbacMetadataWriter } from './abac-metadata-writer.js'\nimport { PackageGenerateCommand } from '../commands/package-generate.js'\nimport { CloudFormationClient, DescribeStackResourcesCommand } from '@aws-sdk/client-cloudformation'\nimport { APIGatewayClient, GetRestApisCommand } from '@aws-sdk/client-api-gateway'\n\nexport class PluginOrchestrator {\n private dynamoManager: DynamoDBManager | null = null\n private policyGenerator: PolicyGenerator | null = null\n private abacMetadataWriter: AbacMetadataWriter | null = null\n\n constructor(\n private serverless: any,\n private config: ModuleRegistryConfig,\n private logger: Logger\n ) {}\n\n /**\n * Initialize dependent managers\n */\n initializeManagers(): void {\n this.dynamoManager = new DynamoDBManager(this.config, this.logger)\n this.policyGenerator = new PolicyGenerator(this.config.policyPrefix!, this.logger)\n this.abacMetadataWriter = new AbacMetadataWriter(this.dynamoManager, this.logger)\n }\n\n /**\n * Generate CloudFormation policy resources\n */\n generatePolicyResources(featureEntries: RegistryFeatureEntry[], stage: string): Record<string, any> {\n if (!this.policyGenerator || this.config.skipPolicies || featureEntries.length === 0) {\n this.logger.info('⏭️ Skipping policy resource generation')\n return {}\n }\n\n try {\n return this.policyGenerator.generatePolicyResources(featureEntries, stage)\n } catch (error) {\n this.logger.error('❌ Error generating policy resources', error)\n return {}\n }\n }\n\n /**\n * Update DynamoDB registry data\n */\n async updateRegistryData(moduleEntries: RegistryModuleEntry[], featureEntries: RegistryFeatureEntry[]): Promise<void> {\n if (!this.dynamoManager) {\n throw new Error('DynamoDB manager not initialized')\n }\n await this.dynamoManager.updateRegistryData(moduleEntries, featureEntries)\n }\n\n /**\n * Write ABAC metadata to DynamoDB\n */\n async writeAbacMetadata(abacRoleMetadata: AbacRoleMetadata[]): Promise<void> {\n if (!this.abacMetadataWriter) {\n throw new Error('ABAC metadata writer not initialized')\n }\n await this.abacMetadataWriter.writeAbacMetadata(abacRoleMetadata)\n }\n\n /**\n * Discover API Gateway RestApiId from CloudFormation stack or AWS API Gateway\n */\n async discoverApiGatewayId(stackName: string, serviceName: string, region: string): Promise<string | undefined> {\n try {\n // Primary method: Query CloudFormation stack for API Gateway resources\n const cfnClient = new CloudFormationClient({ region })\n\n try {\n const describeCommand = new DescribeStackResourcesCommand({\n StackName: stackName\n })\n\n const response = await cfnClient.send(describeCommand)\n\n // Find API Gateway REST API resource\n const apiResource = response.StackResources?.find(\n (resource: any) => resource.ResourceType === 'AWS::ApiGateway::RestApi'\n )\n\n if (apiResource?.PhysicalResourceId) {\n this.logger.info(`✅ Discovered API Gateway ID from CloudFormation: ${apiResource.PhysicalResourceId}`)\n return apiResource.PhysicalResourceId\n }\n } catch (cfnError: any) {\n this.logger.warning(`CloudFormation API Gateway lookup failed: ${cfnError.message}`)\n }\n\n // Fallback method: Use API Gateway client to list and match by name\n const apiGatewayClient = new APIGatewayClient({ region })\n\n try {\n const apisCommand = new GetRestApisCommand({})\n const apisResponse = await apiGatewayClient.send(apisCommand)\n\n // Match by service name (common pattern in Serverless Framework)\n const matchingApi = apisResponse.items?.find(\n api => api.name === serviceName || api.name?.includes(serviceName)\n )\n\n if (matchingApi?.id) {\n this.logger.info(`✅ Discovered API Gateway ID from API Gateway service: ${matchingApi.id}`)\n return matchingApi.id\n }\n } catch (apiError: any) {\n this.logger.warning(`API Gateway service lookup failed: ${apiError.message}`)\n }\n\n // Graceful failure: log warning and return undefined\n this.logger.warning('Could not discover API Gateway ID. Module will be registered without apiGatewayId.')\n return undefined\n\n } catch (error: any) {\n this.logger.warning(`Error discovering API Gateway ID: ${error.message}`)\n return undefined\n }\n }\n\n /**\n * Generate service package\n */\n async generateServicePackage(): Promise<void> {\n try {\n const packageCommand = new PackageGenerateCommand(this.config, this.logger)\n await packageCommand.execute({ force: true })\n } catch (error) {\n this.logger.error('❌ Error generating service file', error)\n // Don't fail deployment for service generation errors\n }\n }\n}","/**\n * Command definitions for the Module Registry plugin\n */\n\nexport const PLUGIN_COMMANDS = {\n registryGenerate: {\n usage: 'Generate registry files for a specific module using AI',\n lifecycleEvents: ['generate'],\n options: {\n module: {\n usage: 'Specify the module name to generate registry for',\n shortcut: 'm',\n required: true,\n type: 'string'\n },\n force: {\n usage: 'Force overwrite existing registry files',\n shortcut: 'f',\n required: false,\n type: 'boolean'\n }\n }\n },\n registryGeneratePackage: {\n usage: 'Generate virtual service package for module registry imports',\n lifecycleEvents: ['create'],\n options: {\n force: {\n usage: 'Force regeneration even if package exists',\n shortcut: 'f',\n required: false,\n type: 'boolean'\n },\n verbose: {\n usage: 'Show detailed generation information',\n shortcut: 'v',\n required: false,\n type: 'boolean'\n }\n }\n }\n} as const\n\nexport const PLUGIN_HOOKS = {\n 'before:package:initialize': 'processModuleRegistryWithVariableResolution',\n 'after:aws:deploy:deploy:updateStack': 'ensureTableAndUpdateData',\n 'registryGenerate:generate': 'registryGenerateHandler',\n 'registryGeneratePackage:create': 'generateServicePackageCommand'\n} as const","/*\n * Module Registry Service Functions\n * -------------------------------------------\n * Service layer for querying module registry data from DynamoDB.\n * These functions can be imported directly from the plugin package.\n * \n * Usage:\n * import { listAllModules, getModuleFeatures } from 'serverless-plugin-module-registry'\n */\n\nimport { DynamoDBClient, QueryCommand, ScanCommand, PutItemCommand, UpdateItemCommand, DeleteItemCommand } from '@aws-sdk/client-dynamodb'\nimport { unmarshall, marshall } from '@aws-sdk/util-dynamodb'\nimport { nanoid } from 'nanoid'\n\n// Self-contained feature ID utilities (to avoid import issues in generated service)\nconst generateFeatureId = () => nanoid(8)\nconst validateFeatureId = (featureId) => {\n if (featureId.length !== 8) return false\n if (!/^[A-Za-z0-9_-]+$/.test(featureId)) return false\n return true\n}\n\nconst TABLE_NAME = 'placeholder-table-name'\nconst GSI1_NAME = 'GSI1'\n\n// Validate table name at runtime\nif (TABLE_NAME === 'placeholder-table-name') {\n console.warn('[module-registry] Warning: TABLE_NAME not properly injected during build. Service functions may fail at runtime.')\n}\n\n// Client configuration with retry logic and connection pooling\nconst client = new DynamoDBClient({\n maxAttempts: 3,\n retryMode: 'adaptive',\n // Enable connection pooling for better performance\n maxSockets: 50,\n requestHandler: {\n requestTimeout: 30000, // 30 seconds\n httpsAgent: {\n keepAlive: true,\n maxSockets: 50\n }\n }\n})\n\n// Simple in-memory cache for frequently accessed data\nconst cache = new Map()\nconst CACHE_TTL = 5 * 60 * 1000 // 5 minutes\n\n/**\n * Simple caching utility\n */\nconst withCache = (key, ttl = CACHE_TTL) => {\n return {\n get: () => {\n const cached = cache.get(key)\n if (cached && Date.now() < cached.expires) {\n return cached.value\n }\n cache.delete(key)\n return null\n },\n set: (value) => {\n cache.set(key, {\n value,\n expires: Date.now() + ttl\n })\n }\n }\n}\n\n/**\n * Creates a structured logger for module registry operations\n * @param {string} context - The operation context (e.g., 'listAllModules')\n * @returns {Object} Logger with info, warn, error methods\n */\nexport const createModuleRegistryLogger = (context) => {\n const prefix = `[module-registry:${context}]`\n\n return {\n info: (message, data) => console.log(prefix, message, data ? JSON.stringify(data) : ''),\n warn: (message, data) => console.warn(prefix, message, data ? JSON.stringify(data) : ''),\n error: (message, error) => console.error(prefix, message, error?.message || error)\n }\n}\n\n/**\n * Sanitize feature name to only allow lowercase letters, numbers, and hyphens [a-z0-9-]\n * Converts underscores and spaces to hyphens\n * @param {string} featureName - The feature name to sanitize\n * @returns {string} - Sanitized feature name\n * @throws {Error} - If feature name is invalid after sanitization\n */\nconst sanitizeFeatureName = (featureName) => {\n if (!featureName || typeof featureName !== 'string') {\n throw new Error('Feature name is required and must be a string')\n }\n \n // Convert to lowercase, replace underscores and spaces with hyphens, then keep only [a-z0-9-] characters\n const sanitized = featureName.toLowerCase()\n .replace(/[_\\s]+/g, '-') // Replace underscores and spaces with hyphens\n .replace(/[^a-z0-9-]/g, '') // Keep only letters, numbers, and hyphens\n .replace(/--+/g, '-') // Replace multiple consecutive hyphens with single hyphen\n .replace(/^-+|-+$/g, '') // Remove leading and trailing hyphens\n \n if (sanitized.length === 0) {\n throw new Error('Feature name must contain at least one alphanumeric character [a-z0-9-]')\n }\n \n return sanitized\n}\n\n/**\n * Validates and sanitizes input parameters\n * @param {Object} params - Parameters to validate\n * @param {Array} required - Required parameter names\n * @throws {Error} If validation fails\n */\nconst validateParams = (params, required) => {\n for (const param of required) {\n if (!params[param] || (typeof params[param] === 'string' && params[param].trim() === '')) {\n throw new Error(`${param} is required and cannot be empty`)\n }\n }\n\n // Sanitize string parameters to prevent injection\n const sanitized = {}\n for (const [key, value] of Object.entries(params)) {\n if (typeof value === 'string') {\n // Basic sanitization - remove potentially dangerous characters\n let sanitizedValue = value.replace(/[<>\\\"']/g, '').trim()\n \n // Apply feature name sanitization for featureName parameters\n if (key === 'featureName') {\n sanitizedValue = sanitizeFeatureName(sanitizedValue)\n }\n \n sanitized[key] = sanitizedValue\n } else {\n sanitized[key] = value\n }\n }\n\n return sanitized\n}\n\n/**\n * Wraps DynamoDB operations with error handling and retry logic\n * @param {Function} operation - The DynamoDB operation to execute\n * @param {string} operationName - Name for logging\n * @returns {Promise} Operation result\n */\nconst withErrorHandling = async (operation, operationName) => {\n const logger = createModuleRegistryLogger(operationName)\n\n try {\n const startTime = Date.now()\n const result = await operation()\n const duration = Date.now() - startTime\n return result\n\n } catch (error) {\n logger.error(`Operation failed`, error)\n\n // Provide user-friendly error messages\n if (error.name === 'ResourceNotFoundException') {\n throw new Error(`Module registry table not found. Please ensure the plugin is properly deployed.`)\n } else if (error.name === 'ValidationException') {\n throw new Error(`Invalid request parameters: ${error.message}`)\n } else if (error.name === 'ProvisionedThroughputExceededException') {\n throw new Error(`Registry service is temporarily overloaded. Please try again in a moment.`)\n } else if (error.name === 'ThrottlingException') {\n throw new Error(`Request rate exceeded. Please try again in a moment.`)\n } else {\n throw new Error(`Registry service error: ${error.message}`)\n }\n }\n}\n/**\n * Lists all deployed modules with basic metadata\n * @returns {Promise<Array>} Array of module objects with clean data (no DDB internals)\n */\nexport const listAllModules = async () => {\n return withErrorHandling(async () => {\n // Check cache first\n const cacheKey = 'listAllModules'\n const cached = withCache(cacheKey)\n const cachedResult = cached.get()\n if (cachedResult) {\n return cachedResult\n }\n\n const command = new QueryCommand({\n TableName: TABLE_NAME,\n IndexName: GSI1_NAME,\n KeyConditionExpression: 'gsi1pk = :pk',\n FilterExpression: 'itemType = :itemType',\n ExpressionAttributeValues: {\n ':pk': { S: 'MODULES' },\n ':itemType': { S: 'module' }\n }\n })\n\n const response = await client.send(command)\n\n if (!response.Items) {\n return []\n }\n\n // Transform DDB items to clean module objects\n const modules = response.Items.map(item => {\n const data = unmarshall(item)\n\n return {\n moduleName: data.moduleName,\n description: data.description,\n version: data.version,\n maintainer: data.maintainer,\n tags: data.tags || [],\n apiGatewayId: data.apiGatewayId,\n lastUpdated: data.lastUpdated\n }\n })\n\n // Cache the result\n cached.set(modules)\n\n return modules\n }, 'listAllModules')\n}\n\n/**\n * Gets all features for a specific module\n * @param {string} moduleName - The module name (e.g., 'sign')\n * @returns {Promise<Array>} Array of feature objects with endpoints\n */\nexport const getModuleFeatures = async (moduleName) => {\n const params = validateParams({ moduleName }, ['moduleName'])\n\n return withErrorHandling(async () => {\n const command = new QueryCommand({\n TableName: TABLE_NAME,\n KeyConditionExpression: 'pk = :pk AND begins_with(sk, :skPrefix)',\n FilterExpression: 'itemType = :itemType',\n ExpressionAttributeValues: {\n ':pk': { S: `MODULE#${params.moduleName}` },\n ':skPrefix': { S: 'FEATURE#' },\n ':itemType': { S: 'feature' }\n }\n })\n\n const response = await client.send(command)\n\n if (!response.Items) {\n return []\n }\n\n // Transform DDB items to clean feature objects\n const features = response.Items.map(item => {\n const data = unmarshall(item)\n\n return {\n featureName: data.featureName,\n featureId: data.featureId, // Include featureId for ABAC tag generation\n description: data.description,\n version: data.version,\n endpoints: data.endpoints || [],\n customPolicies: data.customPolicies || [],\n endpointCount: (data.endpoints || []).length,\n lastUpdated: data.lastUpdated\n }\n })\n\n return features\n }, 'getModuleFeatures')\n}\n\n/**\n * Gets detailed information about a specific feature\n * @param {string} moduleName - The module name \n * @param {string} featureId - The feature ID (featureId is KING!)\n * @returns {Promise<Object|null>} Feature details with endpoints and policies\n */\nexport const getFeatureDetails = async (moduleName, featureId) => {\n const params = validateParams({ moduleName, featureId }, ['moduleName', 'featureId'])\n\n return withErrorHandling(async () => {\n const command = new QueryCommand({\n TableName: TABLE_NAME,\n KeyConditionExpression: 'pk = :pk AND sk = :sk',\n ExpressionAttributeValues: {\n ':pk': { S: `MODULE#${params.moduleName}` },\n ':sk': { S: `FEATURE#${params.featureId}` }\n }\n })\n\n const response = await client.send(command)\n\n if (!response.Items || response.Items.length === 0) {\n return null\n }\n\n // Transform DDB item to clean feature object\n return unmarshall(response.Items[0])\n }, 'getFeatureDetails')\n}\n\n\n/**\n * Gets module metadata (basic info without features)\n * @param {string} moduleName - The module name\n * @returns {Promise<Object|null>} Module metadata or null if not found\n */\nexport const getModuleMetadata = async (moduleName) => {\n const params = validateParams({ moduleName }, ['moduleName'])\n\n return withErrorHandling(async () => {\n const command = new QueryCommand({\n TableName: TABLE_NAME,\n KeyConditionExpression: 'pk = :pk AND sk = :sk',\n ExpressionAttributeValues: {\n ':pk': { S: `MODULE#${params.moduleName}` },\n ':sk': { S: 'MODULE' }\n }\n })\n\n const response = await client.send(command)\n\n if (!response.Items || response.Items.length === 0) {\n return null\n }\n\n // Transform DDB item to clean module object\n const data = unmarshall(response.Items[0])\n\n const moduleMetadata = {\n moduleName: data.moduleName,\n description: data.description,\n version: data.version,\n maintainer: data.maintainer,\n tags: data.tags || [],\n apiGatewayId: data.apiGatewayId,\n lastUpdated: data.lastUpdated\n }\n\n return moduleMetadata\n }, 'getModuleMetadata')\n}\n\n/**\n * Gets feature details by featureId only (without knowing module name)\n * Uses GSI2 to lookup feature directly by featureId\n * @param {string} featureId - The unique feature identifier\n * @returns {Promise<Object|null>} Feature details or null if not found\n */\nexport const getFeatureById = async (featureId) => {\n const params = validateParams({ featureId }, ['featureId'])\n\n return withErrorHandling(async () => {\n const command = new QueryCommand({\n TableName: TABLE_NAME,\n IndexName: 'GSI2',\n KeyConditionExpression: 'gsi2pk = :gsi2pk AND gsi2sk = :gsi2sk',\n ExpressionAttributeValues: {\n ':gsi2pk': { S: 'FEATURES' },\n ':gsi2sk': { S: `FEATURE#${params.featureId}` }\n }\n })\n\n const response = await client.send(command)\n\n if (!response.Items || response.Items.length === 0) {\n return null\n }\n\n // Transform DDB item to clean feature object\n const data = unmarshall(response.Items[0])\n\n return {\n moduleName: data.moduleName,\n featureName: data.featureName,\n featureId: data.featureId,\n description: data.description,\n version: data.version,\n endpoints: data.endpoints || [],\n customPolicies: data.customPolicies || [],\n customFeature: data.customFeature || false,\n endpointCount: (data.endpoints || []).length,\n policyCount: (data.customPolicies || []).length,\n lastUpdated: data.lastUpdated\n }\n }, 'getFeatureById')\n}\n\n/**\n * Gets endpoint summary across all modules for discovery\n * @returns {Promise<Array>} Array of endpoints with module/feature context\n */\nexport const getAllEndpoints = async () => {\n return withErrorHandling(async () => {\n const command = new ScanCommand({\n TableName: TABLE_NAME,\n FilterExpression: 'itemType = :itemType',\n ExpressionAttributeValues: {\n ':itemType': { S: 'feature' }\n }\n })\n\n const response = await client.send(command)\n\n if (!response.Items) {\n return []\n }\n\n // Transform to flat endpoint list with context\n const endpoints = []\n\n for (const item of response.Items) {\n const data = unmarshall(item)\n\n if (data.endpoints && Array.isArray(data.endpoints)) {\n for (const endpoint of data.endpoints) {\n endpoints.push({\n endpoint,\n moduleName: data.moduleName,\n featureName: data.featureName,\n featureDescription: data.description,\n version: data.version,\n customFeature: data.customFeature || false\n })\n }\n }\n }\n\n return endpoints\n }, 'getAllEndpoints')\n}\n\n/**\n * Gets endpoint summary from non-custom features only\n * @returns {Promise<Array>} Array of endpoints from regular (non-custom) features\n */\nexport const getNonCustomEndpoints = async () => {\n return withErrorHandling(async () => {\n const command = new ScanCommand({\n TableName: TABLE_NAME,\n FilterExpression: 'itemType = :itemType AND (attribute_not_exists(customFeature) OR customFeature = :customFeature)',\n ExpressionAttributeValues: {\n ':itemType': { S: 'feature' },\n ':customFeature': { BOOL: false }\n }\n })\n\n const response = await client.send(command)\n\n if (!response.Items) {\n return []\n }\n\n // Transform to flat endpoint list with context\n const endpoints = []\n\n for (const item of response.Items) {\n const data = unmarshall(item)\n\n if (data.endpoints && Array.isArray(data.endpoints)) {\n for (const endpoint of data.endpoints) {\n endpoints.push({\n endpoint,\n moduleName: data.moduleName,\n featureName: data.featureName,\n featureDescription: data.description,\n version: data.version,\n customFeature: false\n })\n }\n }\n }\n\n return endpoints\n }, 'getNonCustomEndpoints')\n}\n\n/**\n * Creates a new custom feature in the module registry\n * @param {Object} featureData - The feature data\n * @param {string} featureData.moduleName - The module name\n * @param {string} featureData.featureName - The feature name\n * @param {string} featureData.description - Feature description\n * @param {string} featureData.version - Feature version (defaults to '1.0.0')\n * @param {Array} featureData.endpoints - Array of endpoints\n * @param {Array} featureData.customPolicies - Array of custom policies\n * @param {string} featureData.tenantId - Required tenant ID for tenant-specific custom features\n * @param {string} featureData.policyArn - Optional policy ARN for the feature\n * @param {string} featureData.policyName - Optional policy name for the feature\n * @returns {Promise<Object>} Created feature object\n */\nexport const createCustomFeature = async (featureData) => {\n const params = validateParams(featureData, ['moduleName', 'featureName', 'description', 'tenantId'])\n \n return withErrorHandling(async () => {\n const logger = createModuleRegistryLogger('createCustomFeature')\n \n // Use tenant-isolated partition key for custom features\n const tenantModulePk = `TENANT#${params.tenantId}#MODULE#${params.moduleName}`\n \n // Generate feature ID if not provided\n const featureId = params.featureId || generateFeatureId()\n \n // Validate feature ID\n if (!validateFeatureId(featureId)) {\n throw new Error(`Invalid feature ID: ${featureId} for feature: ${params.featureName}`)\n }\n\n // Prepare the feature item with tenant-isolated keys\n const featureItem = {\n pk: tenantModulePk,\n sk: `FEATURE#${featureId}`,\n gsi1pk: `TENANT#${params.tenantId}`, // For querying all tenant's custom features\n gsi1sk: `MODULE#${params.moduleName}#FEATURE#${featureId}`,\n itemType: 'feature',\n moduleName: params.moduleName,\n featureName: params.featureName,\n featureId: featureId,\n description: params.description,\n version: params.version || '1.0.0',\n endpoints: params.endpoints || [],\n customPolicies: params.customPolicies || [],\n customFeature: true, // Mark as custom feature\n tenantId: params.tenantId, // Always required for custom features\n lastUpdated: new Date().toISOString()\n }\n \n if (params.policyArn) {\n featureItem.policyArn = params.policyArn\n }\n \n if (params.policyName) {\n featureItem.policyName = params.policyName\n }\n\n logger.info(`Creating tenant-isolated custom feature ${params.featureName} in module ${params.moduleName} for tenant ${params.tenantId}`)\n\n // Use PutItem to create the feature with tenant-isolated key\n const command = new PutItemCommand({\n TableName: TABLE_NAME,\n Item: marshall(featureItem, { removeUndefinedValues: true }),\n ConditionExpression: 'attribute_not_exists(pk)', // Prevent overwriting existing features\n })\n\n await client.send(command)\n\n logger.info(`Tenant-isolated custom feature ${params.featureName} created successfully for tenant ${params.tenantId}`)\n\n // Return clean feature object (without DDB internals)\n return {\n moduleName: featureItem.moduleName,\n featureName: featureItem.featureName,\n featureId: featureItem.featureId,\n description: featureItem.description,\n version: featureItem.version,\n endpoints: featureItem.endpoints,\n customPolicies: featureItem.customPolicies,\n customFeature: featureItem.customFeature,\n tenantId: featureItem.tenantId,\n endpointCount: featureItem.endpoints.length,\n lastUpdated: featureItem.lastUpdated,\n ...(featureItem.policyArn && { policyArn: featureItem.policyArn }),\n ...(featureItem.policyName && { policyName: featureItem.policyName })\n }\n }, 'createCustomFeature')\n}\n\n/**\n * Creates a new regular (non-custom) feature in the module registry\n * @param {Object} featureData - The feature data\n * @param {string} featureData.moduleName - The module name\n * @param {string} featureData.featureName - The feature name\n * @param {string} featureData.description - Feature description\n * @param {string} featureData.version - Feature version (defaults to '1.0.0')\n * @param {Array} featureData.endpoints - Array of endpoints\n * @param {Array} featureData.customPolicies - Array of custom policies\n * @param {string} featureData.policyArn - Optional policy ARN for the feature\n * @param {string} featureData.policyName - Optional policy name for the feature\n * @returns {Promise<Object>} Created feature object\n */\nexport const createFeature = async (featureData) => {\n const params = validateParams(featureData, ['moduleName', 'featureName', 'description'])\n \n return withErrorHandling(async () => {\n const logger = createModuleRegistryLogger('createFeature')\n \n // Generate feature ID if not provided\n const featureId = params.featureId || generateFeatureId()\n \n // Validate feature ID\n if (!validateFeatureId(featureId)) {\n throw new Error(`Invalid feature ID: ${featureId} for feature: ${params.featureName}`)\n }\n\n // Prepare the feature item\n const featureItem = {\n pk: `MODULE#${params.moduleName}`,\n sk: `FEATURE#${featureId}`,\n gsi1pk: `MODULE#${params.moduleName}`,\n gsi1sk: `FEATURE#${featureId}`,\n itemType: 'feature',\n moduleName: params.moduleName,\n featureName: params.featureName,\n featureId: featureId,\n description: params.description,\n version: params.version || '1.0.0',\n endpoints: params.endpoints || [],\n customPolicies: params.customPolicies || [],\n customFeature: false, // Mark as regular feature\n lastUpdated: new Date().toISOString()\n }\n\n // Add optional fields if provided\n if (params.policyArn) {\n featureItem.policyArn = params.policyArn\n }\n \n if (params.policyName) {\n featureItem.policyName = params.policyName\n }\n\n logger.info(`Creating feature ${params.featureName} in module ${params.moduleName}`)\n\n // Use PutItem to create the feature\n const command = new PutItemCommand({\n TableName: TABLE_NAME,\n Item: marshall(featureItem, { removeUndefinedValues: true }),\n ConditionExpression: 'attribute_not_exists(pk)', // Prevent overwriting existing features\n })\n\n await client.send(command)\n\n logger.info(`Feature ${params.featureName} created successfully`)\n\n // Return clean feature object (without DDB internals)\n return {\n moduleName: featureItem.moduleName,\n featureName: featureItem.featureName,\n featureId: featureItem.featureId,\n description: featureItem.description,\n version: featureItem.version,\n endpoints: featureItem.endpoints,\n customPolicies: featureItem.customPolicies,\n customFeature: featureItem.customFeature,\n endpointCount: featureItem.endpoints.length,\n lastUpdated: featureItem.lastUpdated,\n ...(featureItem.policyArn && { policyArn: featureItem.policyArn }),\n ...(featureItem.policyName && { policyName: featureItem.policyName })\n }\n }, 'createFeature')\n}\n\n/**\n * Updates a custom feature in the module registry\n * @param {string} moduleName - The module name\n * @param {string} featureId - The feature ID (featureId is KING!)\n * @param {Object} updates - Fields to update\n * @param {string} updates.description - Updated description\n * @param {Array} updates.endpoints - Updated endpoints array\n * @param {Array} updates.customPolicies - Updated custom policies\n * @param {string} tenantId - Required tenant ID for tenant-isolated custom features\n * @returns {Promise<Object>} Updated feature object\n */\nexport const updateCustomFeature = async (moduleName, featureId, updates, tenantId) => {\n const params = validateParams({ moduleName, featureId, tenantId }, ['moduleName', 'featureId', 'tenantId'])\n \n return withErrorHandling(async () => {\n const logger = createModuleRegistryLogger('updateCustomFeature')\n \n const allowedUpdates = ['description', 'endpoints', 'customPolicies']\n const filteredUpdates = {}\n \n for (const [key, value] of Object.entries(updates)) {\n if (allowedUpdates.includes(key)) {\n filteredUpdates[key] = value\n }\n }\n \n if (Object.keys(filteredUpdates).length === 0) {\n throw new Error(`No valid fields to update. Allowed fields: ${allowedUpdates.join(', ')}`)\n }\n \n // Add lastUpdated timestamp\n filteredUpdates.lastUpdated = new Date().toISOString()\n \n logger.info(`Updating tenant-isolated custom feature ${params.featureId} in module ${params.moduleName} for tenant ${params.tenantId}`)\n \n const expressionParts = []\n const expressionAttributeNames = {}\n const expressionAttributeValues = {}\n \n for (const [key, value] of Object.entries(filteredUpdates)) {\n expressionParts.push(`#${key} = :${key}`)\n expressionAttributeNames[`#${key}`] = key\n expressionAttributeValues[`:${key}`] = value\n }\n \n // Condition to ensure it exists, is a custom feature, and belongs to the tenant\n const conditionExpression = 'attribute_exists(pk) AND customFeature = :customFeature AND tenantId = :tenantId'\n expressionAttributeValues[':customFeature'] = true\n expressionAttributeValues[':tenantId'] = params.tenantId\n \n // Use tenant-isolated partition key\n const tenantModulePk = `TENANT#${params.tenantId}#MODULE#${params.moduleName}`\n \n const command = new UpdateItemCommand({\n TableName: TABLE_NAME,\n Key: marshall({\n pk: tenantModulePk,\n sk: `FEATURE#${params.featureId}`\n }),\n UpdateExpression: `SET ${expressionParts.join(', ')}`,\n ExpressionAttributeNames: expressionAttributeNames,\n ExpressionAttributeValues: marshall(expressionAttributeValues),\n ConditionExpression: conditionExpression,\n ReturnValues: 'ALL_NEW'\n })\n \n const { Attributes } = await client.send(command)\n \n logger.info(`Tenant-isolated custom feature ${params.featureId} updated successfully for tenant ${params.tenantId}`)\n \n // Return clean feature object (without DDB internals)\n const updatedFeature = unmarshall(Attributes)\n return {\n moduleName: updatedFeature.moduleName,\n featureName: updatedFeature.featureName,\n description: updatedFeature.description,\n version: updatedFeature.version,\n endpoints: updatedFeature.endpoints,\n customPolicies: updatedFeature.customPolicies,\n customFeature: updatedFeature.customFeature,\n tenantId: updatedFeature.tenantId,\n endpointCount: updatedFeature.endpoints.length,\n lastUpdated: updatedFeature.lastUpdated,\n ...(updatedFeature.policyArn && { policyArn: updatedFeature.policyArn }),\n ...(updatedFeature.policyName && { policyName: updatedFeature.policyName })\n }\n }, 'updateCustomFeature')\n}\n\n/**\n * Deletes a custom feature from the module registry\n * @param {string} moduleName - The module name\n * @param {string} featureId - The feature ID (featureId is KING!)\n * @param {string} tenantId - Required tenant ID for tenant-isolated custom features\n * @returns {Promise<boolean>} True if deleted successfully\n */\nexport const deleteCustomFeature = async (moduleName, featureId, tenantId) => {\n const params = validateParams({ moduleName, featureId, tenantId }, ['moduleName', 'featureId', 'tenantId'])\n \n return withErrorHandling(async () => {\n const logger = createModuleRegistryLogger('deleteCustomFeature')\n \n logger.info(`Deleting tenant-isolated custom feature ${params.featureId} from module ${params.moduleName} for tenant ${params.tenantId}`)\n \n // Condition to ensure it exists, is a custom feature, and belongs to the tenant\n const conditionExpression = 'attribute_exists(pk) AND customFeature = :customFeature AND tenantId = :tenantId'\n const expressionAttributeValues = {\n ':customFeature': true,\n ':tenantId': params.tenantId\n }\n \n // Use tenant-isolated partition key\n const tenantModulePk = `TENANT#${params.tenantId}#MODULE#${params.moduleName}`\n \n const command = new DeleteItemCommand({\n TableName: TABLE_NAME,\n Key: marshall({\n pk: tenantModulePk,\n sk: `FEATURE#${params.featureId}`\n }),\n ConditionExpression: conditionExpression,\n ExpressionAttributeValues: marshall(expressionAttributeValues)\n })\n \n await client.send(command)\n \n logger.info(`Tenant-isolated custom feature ${params.featureId} deleted successfully for tenant ${params.tenantId}`)\n return true\n }, 'deleteCustomFeature')\n}\n\n/**\n * Lists custom features for a specific tenant using efficient GSI query\n * @param {string} tenantId - The tenant ID\n * @returns {Promise<Array>} Array of custom features for the tenant\n */\nexport const listCustomFeaturesByTenant = async (tenantId) => {\n const params = validateParams({ tenantId }, ['tenantId'])\n \n return withErrorHandling(async () => {\n const logger = createModuleRegistryLogger('listCustomFeaturesByTenant')\n \n logger.info(`Listing tenant-isolated custom features for tenant ${params.tenantId}`)\n \n // Use GSI1 for efficient tenant-specific query\n const command = new QueryCommand({\n TableName: TABLE_NAME,\n IndexName: GSI1_NAME,\n KeyConditionExpression: 'gsi1pk = :tenantPk',\n FilterExpression: 'itemType = :itemType AND customFeature = :customFeature',\n ExpressionAttributeValues: {\n ':tenantPk': { S: `TENANT#${params.tenantId}` },\n ':itemType': { S: 'feature' },\n ':customFeature': { BOOL: true }\n }\n })\n \n const response = await client.send(command)\n \n if (!response.Items) {\n return []\n }\n \n // Transform to clean feature objects\n const customFeatures = response.Items.map(item => {\n const data = unmarshall(item)\n \n return {\n moduleName: data.moduleName,\n featureName: data.featureName,\n description: data.description,\n version: data.version,\n endpoints: data.endpoints || [],\n customPolicies: data.customPolicies || [],\n customFeature: data.customFeature,\n tenantId: data.tenantId,\n endpointCount: (data.endpoints || []).length,\n lastUpdated: data.lastUpdated,\n ...(data.policyArn && { policyArn: data.policyArn }),\n ...(data.policyName && { policyName: data.policyName })\n }\n })\n \n logger.info(`Found ${customFeatures.length} tenant-isolated custom features for tenant ${params.tenantId}`)\n return customFeatures\n }, 'listCustomFeaturesByTenant')\n}\n\n/**\n * Gets detailed information about a specific tenant custom feature\n * @param {string} moduleName - The module name\n * @param {string} featureId - The feature ID (featureId is KING!)\n * @param {string} tenantId - The tenant ID\n * @returns {Promise<Object|null>} Tenant custom feature details or null if not found\n */\nexport const getTenantCustomFeature = async (moduleName, featureId, tenantId) => {\n const params = validateParams({ moduleName, featureId, tenantId }, ['moduleName', 'featureId', 'tenantId'])\n\n return withErrorHandling(async () => {\n const logger = createModuleRegistryLogger('getTenantCustomFeature')\n \n logger.info(`Getting tenant-isolated custom feature ${params.featureId} from module ${params.moduleName} for tenant ${params.tenantId}`)\n \n // Use tenant-isolated partition key\n const tenantModulePk = `TENANT#${params.tenantId}#MODULE#${params.moduleName}`\n \n const command = new QueryCommand({\n TableName: TABLE_NAME,\n KeyConditionExpression: 'pk = :pk AND sk = :sk',\n ExpressionAttributeValues: {\n ':pk': { S: tenantModulePk },\n ':sk': { S: `FEATURE#${params.featureId}` }\n }\n })\n\n const response = await client.send(command)\n\n if (!response.Items || response.Items.length === 0) {\n logger.info(`Tenant custom feature ${params.featureId} not found for tenant ${params.tenantId}`)\n return null\n }\n\n // Transform DDB item to clean feature object\n const data = unmarshall(response.Items[0])\n \n return {\n moduleName: data.moduleName,\n featureName: data.featureName,\n description: data.description,\n version: data.version,\n endpoints: data.endpoints || [],\n customPolicies: data.customPolicies || [],\n customFeature: data.customFeature,\n tenantId: data.tenantId,\n endpointCount: (data.endpoints || []).length,\n lastUpdated: data.lastUpdated,\n ...(data.policyArn && { policyArn: data.policyArn }),\n ...(data.policyName && { policyName: data.policyName })\n }\n }, 'getTenantCustomFeature')\n}\n\n// =============================================================================\n// ABAC (Attribute-Based Access Control) Functions\n// =============================================================================\n\n/**\n * Converts an endpoint string to API Gateway ARN format\n * @param {string} endpoint - Endpoint in format \"METHOD - /path/pattern\"\n * @param {string} region - AWS region (defaults to *)\n * @param {string} accountId - AWS account ID (defaults to *)\n * @param {string} apiId - API Gateway ID (defaults to *)\n * @returns {string} API Gateway ARN\n */\nconst endpointToArn = (endpoint, region = '*', accountId = '*', apiId = '*') => {\n // Parse endpoint format: \"POST - /admin/tenants\" or \"POST - admin/tenants\"\n const parts = endpoint.split(' - ')\n if (parts.length !== 2) {\n throw new Error(`Invalid endpoint format: ${endpoint}. Expected \"METHOD - path\"`)\n }\n \n const method = parts[0].trim().toUpperCase()\n let path = parts[1].trim()\n \n // Normalize path to ensure it starts with /\n if (!path.startsWith('/')) {\n path = `/${path}`\n }\n \n // Convert path parameters from {param} to * for ARN matching\n const arnPath = path.replace(/\\{[^}]+\\}/g, '*')\n \n return `arn:aws:execute-api:${region}:${accountId}:${apiId}/*/${method}${arnPath}`\n}\n\n/**\n * Creates an ABAC condition for feature access\n * @param {string} moduleName - Module name (e.g., 'tenants')\n * @param {string} featureName - Feature name (e.g., 'criar-tenant')\n * @returns {Object} IAM condition object\n */\nconst createAbacCondition = (moduleName, featureName) => {\n return {\n StringLike: {\n [`aws:PrincipalTag/${moduleName}Features`]: `*${featureName}*`\n }\n }\n}\n\n/**\n * Generates a single module-level managed policy with ABAC conditions\n * @param {string} moduleName - The module name (e.g., 'tenants')\n * @param {Object} options - Generation options\n * @param {string} options.region - AWS region for ARN generation (defaults to *)\n * @param {string} options.accountId - AWS account ID for ARN generation (defaults to *)\n * @param {string} options.apiId - API Gateway ID for ARN generation (defaults to *)\n * @returns {Promise<Object>} Complete IAM policy document with ABAC conditions\n */\nexport const generateModuleAbacPolicy = async (moduleName, options = {}) => {\n const params = validateParams({ moduleName }, ['moduleName'])\n const { region = '*', accountId = '*', apiId = '*', selectedFeatures } = options\n \n return withErrorHandling(async () => {\n const logger = createModuleRegistryLogger('generateModuleAbacPolicy')\n \n logger.info(`Generating ABAC policy for module ${params.moduleName}`)\n \n // Use provided features if available, otherwise query the database\n let features\n if (selectedFeatures && Array.isArray(selectedFeatures)) {\n // Load full feature details for each selected feature to get endpoints\n logger.info(`Loading full details for ${selectedFeatures.length} selected features`)\n \n features = await Promise.all(\n selectedFeatures.map(async (sf) => {\n const featureDetails = await getFeatureDetails(sf.moduleName, sf.featureId)\n if (!featureDetails) {\n throw new Error(`Feature '${sf.featureId}' not found in module '${sf.moduleName}'. Make sure the feature exists in the registry.`)\n }\n \n return {\n featureName: featureDetails.featureName,\n moduleName: featureDetails.moduleName || sf.moduleName,\n endpoints: featureDetails.endpoints || [],\n customPolicies: featureDetails.customPolicies || []\n }\n })\n )\n \n logger.info(`Loaded feature details for ${features.length} features`)\n } else {\n // Get all features for this module (both regular and custom) from database\n features = await getModuleFeatures(params.moduleName)\n logger.info(`Retrieved ${features?.length || 0} features from database`)\n }\n \n if (!features || features.length === 0) {\n logger.warn(`No features found for module ${params.moduleName}`)\n return {\n Version: '2012-10-17',\n Statement: []\n }\n }\n \n // Generate policy statements for each feature\n const statements = features.map(feature => {\n // Sanitize both module name and feature name for AWS IAM SID requirements (alphanumeric only)\n const moduleNameSafeId = params.moduleName.replace(/[^a-zA-Z0-9]/g, '')\n const featureNameSafeId = feature.featureName.replace(/[^a-zA-Z0-9]/g, '')\n \n // Collect all actions (API Gateway + custom policies)\n const actions = ['execute-api:Invoke']\n const resources = []\n \n // Add API Gateway resources from endpoints\n if (feature.endpoints && feature.endpoints.length > 0) {\n feature.endpoints.forEach(endpoint => {\n resources.push(endpointToArn(endpoint, region, accountId, apiId))\n })\n }\n \n // Add custom policy actions and resources\n if (feature.customPolicies && feature.customPolicies.length > 0) {\n feature.customPolicies.forEach(policy => {\n if (policy.Action) {\n const policyActions = Array.isArray(policy.Action) ? policy.Action : [policy.Action]\n actions.push(...policyActions)\n }\n if (policy.Resource) {\n const policyResources = Array.isArray(policy.Resource) ? policy.Resource : [policy.Resource]\n resources.push(...policyResources)\n }\n })\n }\n \n // Remove duplicates\n const uniqueActions = [...new Set(actions)]\n const uniqueResources = [...new Set(resources)]\n \n // If no resources defined, default to all\n if (uniqueResources.length === 0) {\n uniqueResources.push('*')\n }\n \n return {\n Sid: `${moduleNameSafeId}${featureNameSafeId}Access`,\n Effect: 'Allow',\n Action: uniqueActions,\n Resource: uniqueResources,\n Condition: createAbacCondition(params.moduleName, feature.featureName)\n }\n })\n \n const policyDocument = {\n Version: '2012-10-17',\n Statement: statements\n }\n \n logger.info(`Generated ABAC policy for module ${params.moduleName} with ${statements.length} feature statements`)\n \n return policyDocument\n }, 'generateModuleAbacPolicy')\n}\n\n/**\n * Generates ABAC policy documents for all modules\n * @param {Object} options - Generation options\n * @param {string} options.region - AWS region for ARN generation\n * @param {string} options.accountId - AWS account ID for ARN generation \n * @param {string} options.apiId - API Gateway ID for ARN generation\n * @returns {Promise<Object>} Object with module names as keys and policy documents as values\n */\nexport const generateAllModuleAbacPolicies = async (options = {}) => {\n return withErrorHandling(async () => {\n const logger = createModuleRegistryLogger('generateAllModuleAbacPolicies')\n \n logger.info('Generating ABAC policies for all modules')\n \n // Get all modules\n const modules = await listAllModules()\n \n if (!modules || modules.length === 0) {\n logger.warn('No modules found')\n return {}\n }\n \n // Generate policy for each module\n const moduleAbacPolicies = {}\n \n for (const module of modules) {\n logger.info(`Processing module: ${module.moduleName}`)\n moduleAbacPolicies[module.moduleName] = await generateModuleAbacPolicy(module.moduleName, options)\n }\n \n logger.info(`Generated ABAC policies for ${Object.keys(moduleAbacPolicies).length} modules`)\n \n return moduleAbacPolicies\n }, 'generateAllModuleAbacPolicies')\n}\n\n/**\n * Creates resource tags for a role based on selected features\n * @param {Array} selectedFeatures - Array of feature objects with moduleName and featureId\n * @returns {Promise<Array>} Array of AWS resource tags for IAM role\n */\nexport const generateAbacTags = async (selectedFeatures) => {\n const params = validateParams({ selectedFeatures }, ['selectedFeatures'])\n \n return withErrorHandling(async () => {\n const logger = createModuleRegistryLogger('generateAbacTags')\n \n if (!Array.isArray(params.selectedFeatures)) {\n throw new Error('selectedFeatures must be an array')\n }\n \n logger.info(`Generating ABAC tags for ${params.selectedFeatures.length} selected features`)\n \n // Group features by module\n const featuresByModule = {}\n \n params.selectedFeatures.forEach(feature => {\n if (!feature.moduleName || !feature.featureId) {\n logger.warn('Skipping invalid feature - missing moduleName or featureId', feature)\n return\n }\n \n if (!featuresByModule[feature.moduleName]) {\n featuresByModule[feature.moduleName] = []\n }\n \n featuresByModule[feature.moduleName].push(feature.featureId)\n })\n \n // Convert to AWS tags format - using short featureIds for IAM compliance\n const tags = Object.entries(featuresByModule).map(([moduleName, featureIds]) => ({\n Key: `${moduleName}Features`,\n Value: featureIds.join(':')\n }))\n logger.info(`ABAC tags: ${JSON.stringify(tags)}`)\n logger.info(`Generated ${tags.length} ABAC tags for modules: ${Object.keys(featuresByModule).join(', ')}`)\n \n return tags\n }, 'generateAbacTags')\n}\n\n/**\n * Gets required managed policy ARNs for a set of selected features\n * @param {Array} selectedFeatures - Array of feature objects with moduleName and featureName\n * @param {string} accountId - AWS account ID for ARN generation\n * @param {string} service - Service name (e.g., 'api')\n * @param {string} stage - Stage name (e.g., 'live')\n * @returns {Promise<Array>} Array of managed policy ARNs needed for the selected features\n */\nexport const getRequiredModulePolicyArns = async (selectedFeatures, accountId, service, stage) => {\n const params = validateParams({ selectedFeatures, accountId, service, stage }, ['selectedFeatures', 'accountId', 'service', 'stage'])\n \n return withErrorHandling(async () => {\n const logger = createModuleRegistryLogger('getRequiredModulePolicyArns')\n \n if (!Array.isArray(params.selectedFeatures)) {\n throw new Error('selectedFeatures must be an array')\n }\n \n logger.info(`Getting required module policy ARNs for ${params.selectedFeatures.length} selected features`)\n \n // Get unique module names from selected features\n const moduleNames = [...new Set(params.selectedFeatures.map(f => f.moduleName).filter(Boolean))]\n \n // Generate module-specific ABAC policy ARNs\n const policyArns = moduleNames.map(moduleName => \n `arn:aws:iam::${params.accountId}:policy/${params.service}-${moduleName}-${params.stage}-abac`\n )\n \n logger.info(`Required module-specific ABAC policy ARNs for modules [${moduleNames.join(', ')}]: ${JSON.stringify(policyArns)}`)\n \n return policyArns\n }, 'getRequiredModulePolicyArns')\n}\n\n"],"mappings":"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","names":["exports","module","exports","module","expand","import_fs","import_path","import_client_api_gateway","path","import_serverless_utils","_a","_b","import_serverless_utils","_a","path","import_fs","import_path","glob","_a","_b","glob","hasMagic","start","final","i","acc","ext","_hasMagic","ext","qmark","star","expand","regExpEscape","import_node_url","_a","_b","v","bf","Stream","_a","_b","p","ret","res","rps","readdirCB","_size","path","_a","p","_children","_fs","sep","fs","filter","process","sync","defaultPlatform","_a","path","rest","path","_a","abs","target","_b","defaultPlatform","import_serverless_utils","import_fs","import_serverless_utils","path","fs","path","fs","import_fs","import_path","import_serverless_utils","import_fs","import_path","import_serverless_utils","OpenAI","path","fs","yaml","_a","_b","path","fs","import_fs","import_path","path","fs","module","_a","_b","import_client_dynamodb","import_util_dynamodb","import_nanoid","generateFeatureId","validateFeatureId","path","module","_a","_b","path","fs","module","policyResources","CloudFormationClient","DescribeStackResourcesCommand"]}