Skip to main content
Glama
mcp-prompts-stack.js78.1 kB
#!/usr/bin/env node "use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.McpPromptsStack = void 0; const cdk = require("aws-cdk-lib"); const lambda = require("aws-cdk-lib/aws-lambda"); const apigateway = require("aws-cdk-lib/aws-apigateway"); const dynamodb = require("aws-cdk-lib/aws-dynamodb"); const s3 = require("aws-cdk-lib/aws-s3"); const sqs = require("aws-cdk-lib/aws-sqs"); const iam = require("aws-cdk-lib/aws-iam"); const cloudfront = require("aws-cdk-lib/aws-cloudfront"); const origins = require("aws-cdk-lib/aws-cloudfront-origins"); const logs = require("aws-cdk-lib/aws-logs"); const cognito = require("aws-cdk-lib/aws-cognito"); const aws_lambda_event_sources_1 = require("aws-cdk-lib/aws-lambda-event-sources"); class McpPromptsStack extends cdk.Stack { constructor(scope, id, props) { super(scope, id, props); // S3 Bucket for catalog and assets const promptsBucket = new s3.Bucket(this, 'PromptsBucket', { bucketName: `mcp-prompts-catalog-${this.account}-${this.region}`, versioned: true, publicReadAccess: true, blockPublicAccess: { blockPublicAcls: false, blockPublicPolicy: false, ignorePublicAcls: false, restrictPublicBuckets: false }, cors: [{ allowedOrigins: ['*'], allowedMethods: [s3.HttpMethods.GET, s3.HttpMethods.PUT, s3.HttpMethods.POST], allowedHeaders: ['*'], exposedHeaders: ['ETag'] }], lifecycleRules: [{ id: 'DeleteOldVersions', noncurrentVersionExpiration: cdk.Duration.days(30) }] }); // S3 Bucket for user-uploaded prompts const userPromptsBucket = new s3.Bucket(this, 'UserPromptsBucket', { bucketName: `mcp-user-prompts-${this.account}-${this.region}`, versioned: true, cors: [{ allowedOrigins: ['*'], allowedMethods: [s3.HttpMethods.GET, s3.HttpMethods.PUT, s3.HttpMethods.POST, s3.HttpMethods.DELETE], allowedHeaders: ['*'], exposedHeaders: ['ETag'] }], lifecycleRules: [{ id: 'DeleteOldVersions', noncurrentVersionExpiration: cdk.Duration.days(30) }] }); // Cognito User Pool for authentication const userPool = new cognito.UserPool(this, 'McpPromptsUserPool', { userPoolName: 'mcp-prompts-users', selfSignUpEnabled: true, signInAliases: { email: true, username: false }, passwordPolicy: { minLength: 8, requireLowercase: true, requireUppercase: true, requireDigits: true, requireSymbols: false } }); // User Pool Client const userPoolClient = new cognito.UserPoolClient(this, 'McpPromptsUserPoolClient', { userPool, authFlows: { userPassword: true, userSrp: true, adminUserPassword: true }, generateSecret: false }); // DynamoDB Table for prompts metadata const promptsTable = new dynamodb.Table(this, 'PromptsTable', { tableName: 'mcp-prompts', partitionKey: { name: 'id', type: dynamodb.AttributeType.STRING }, sortKey: { name: 'version', type: dynamodb.AttributeType.STRING }, billingMode: dynamodb.BillingMode.PAY_PER_REQUEST, pointInTimeRecovery: true, stream: dynamodb.StreamViewType.NEW_AND_OLD_IMAGES }); // Add Global Secondary Indexes promptsTable.addGlobalSecondaryIndex({ indexName: 'category-index', partitionKey: { name: 'category', type: dynamodb.AttributeType.STRING }, sortKey: { name: 'created_at', type: dynamodb.AttributeType.STRING } }); promptsTable.addGlobalSecondaryIndex({ indexName: 'latest-version-index', partitionKey: { name: 'id', type: dynamodb.AttributeType.STRING }, sortKey: { name: 'is_latest', type: dynamodb.AttributeType.STRING } }); promptsTable.addGlobalSecondaryIndex({ indexName: 'access-level-index', partitionKey: { name: 'access_level', type: dynamodb.AttributeType.STRING }, sortKey: { name: 'created_at', type: dynamodb.AttributeType.STRING } }); promptsTable.addGlobalSecondaryIndex({ indexName: 'author-index', partitionKey: { name: 'author_id', type: dynamodb.AttributeType.STRING }, sortKey: { name: 'created_at', type: dynamodb.AttributeType.STRING } }); // DynamoDB Table for sessions and caching const sessionsTable = new dynamodb.Table(this, 'SessionsTable', { tableName: 'mcp-sessions', partitionKey: { name: 'session_id', type: dynamodb.AttributeType.STRING }, billingMode: dynamodb.BillingMode.PAY_PER_REQUEST, timeToLiveAttribute: 'expires_at' }); // DynamoDB Table for users and subscriptions const usersTable = new dynamodb.Table(this, 'UsersTable', { tableName: 'mcp-users', partitionKey: { name: 'user_id', type: dynamodb.AttributeType.STRING }, billingMode: dynamodb.BillingMode.PAY_PER_REQUEST, pointInTimeRecovery: true }); usersTable.addGlobalSecondaryIndex({ indexName: 'email-index', partitionKey: { name: 'email', type: dynamodb.AttributeType.STRING } }); usersTable.addGlobalSecondaryIndex({ indexName: 'subscription-index', partitionKey: { name: 'subscription_tier', type: dynamodb.AttributeType.STRING }, sortKey: { name: 'subscription_expires_at', type: dynamodb.AttributeType.STRING } }); // SQS Queue for async processing const processingDLQ = new sqs.Queue(this, 'ProcessingDLQ', { queueName: 'mcp-prompts-processing-dlq', retentionPeriod: cdk.Duration.days(14) }); const processingQueue = new sqs.Queue(this, 'ProcessingQueue', { queueName: 'mcp-prompts-processing', visibilityTimeout: cdk.Duration.minutes(5), deadLetterQueue: { queue: processingDLQ, maxReceiveCount: 3 } }); // SQS Queue for catalog sync const catalogSyncQueue = new sqs.Queue(this, 'CatalogSyncQueue', { queueName: 'mcp-catalog-sync', visibilityTimeout: cdk.Duration.minutes(10) }); // Lambda Layer for common dependencies const commonLayer = new lambda.LayerVersion(this, 'CommonLayer', { code: lambda.Code.fromAsset('../layers/common'), compatibleRuntimes: [lambda.Runtime.NODEJS_18_X], description: 'Common dependencies for MCP Prompts' }); // Main MCP Server Lambda const mcpServerLambda = new lambda.Function(this, 'McpServerFunction', { runtime: lambda.Runtime.NODEJS_18_X, code: lambda.Code.fromAsset('../dist'), handler: 'lambda/mcp-server.handler', layers: [commonLayer], environment: { PROMPTS_TABLE: promptsTable.tableName, SESSIONS_TABLE: sessionsTable.tableName, PROMPTS_BUCKET: promptsBucket.bucketName, PROCESSING_QUEUE: processingQueue.queueUrl, CATALOG_SYNC_QUEUE: catalogSyncQueue.queueUrl, NODE_ENV: 'production', LOG_LEVEL: 'info' }, timeout: cdk.Duration.seconds(30), memorySize: 512, logRetention: logs.RetentionDays.ONE_WEEK }); // Processing Lambda for async tasks const processingLambda = new lambda.Function(this, 'ProcessingFunction', { runtime: lambda.Runtime.NODEJS_18_X, code: lambda.Code.fromAsset('../dist'), handler: 'lambda/processor.handler', layers: [commonLayer], environment: { PROMPTS_TABLE: promptsTable.tableName, SESSIONS_TABLE: sessionsTable.tableName, PROMPTS_BUCKET: promptsBucket.bucketName }, timeout: cdk.Duration.minutes(5), memorySize: 1024, logRetention: logs.RetentionDays.ONE_WEEK }); // Catalog Sync Lambda const catalogSyncLambda = new lambda.Function(this, 'CatalogSyncFunction', { runtime: lambda.Runtime.NODEJS_18_X, code: lambda.Code.fromAsset('../dist'), handler: 'lambda/catalog-sync.handler', layers: [commonLayer], environment: { PROMPTS_TABLE: promptsTable.tableName, PROMPTS_BUCKET: promptsBucket.bucketName, GITHUB_REPO_URL: 'https://github.com/sparesparrow/mcp-prompts-catalog' }, timeout: cdk.Duration.minutes(10), memorySize: 1024, logRetention: logs.RetentionDays.ONE_WEEK }); // Auth Lambda for user authentication const authLambda = new lambda.Function(this, 'AuthFunction', { runtime: lambda.Runtime.NODEJS_18_X, code: lambda.Code.fromAsset('../dist'), handler: 'lambda/auth.handler', layers: [commonLayer], environment: { USER_POOL_ID: userPool.userPoolId, USER_POOL_CLIENT_ID: userPoolClient.userPoolClientId, USERS_TABLE: usersTable.tableName }, timeout: cdk.Duration.seconds(30), memorySize: 512, logRetention: logs.RetentionDays.ONE_WEEK }); // User Management Lambda const userLambda = new lambda.Function(this, 'UserFunction', { runtime: lambda.Runtime.NODEJS_18_X, code: lambda.Code.fromAsset('../dist'), handler: 'lambda/user.handler', layers: [commonLayer], environment: { USER_POOL_ID: userPool.userPoolId, USER_POOL_CLIENT_ID: userPoolClient.userPoolClientId, USERS_TABLE: usersTable.tableName, USER_PROMPTS_BUCKET: userPromptsBucket.bucketName }, timeout: cdk.Duration.seconds(30), memorySize: 512, logRetention: logs.RetentionDays.ONE_WEEK }); // Stripe Webhook Lambda const stripeWebhookLambda = new lambda.Function(this, 'StripeWebhookFunction', { runtime: lambda.Runtime.NODEJS_18_X, code: lambda.Code.fromAsset('../dist'), handler: 'lambda/stripe-webhook.handler', layers: [commonLayer], environment: { USERS_TABLE: usersTable.tableName, STRIPE_WEBHOOK_SECRET: process.env.STRIPE_WEBHOOK_SECRET || '' }, timeout: cdk.Duration.seconds(30), memorySize: 512, logRetention: logs.RetentionDays.ONE_WEEK }); // Cognito Authorizer const cognitoAuthorizer = new apigateway.CognitoUserPoolsAuthorizer(this, 'CognitoAuthorizer', { cognitoUserPools: [userPool] }); // API Gateway const api = new apigateway.RestApi(this, 'McpPromptsApi', { restApiName: 'MCP Prompts Service', description: 'API for MCP Prompts with AWS backend', defaultCorsPreflightOptions: { allowOrigins: apigateway.Cors.ALL_ORIGINS, allowMethods: apigateway.Cors.ALL_METHODS, allowHeaders: ['Content-Type', 'X-Amz-Date', 'Authorization', 'X-Api-Key', 'X-Amz-Security-Token'] }, endpointConfiguration: { types: [apigateway.EndpointType.REGIONAL] } }); // API Key for rate limiting const apiKey = new apigateway.ApiKey(this, 'McpPromptsApiKey', { apiKeyName: 'mcp-prompts-api-key', description: 'API Key for MCP Prompts' }); // Usage plan for rate limiting const usagePlan = new apigateway.UsagePlan(this, 'McpPromptsUsagePlan', { name: 'MCP Prompts Usage Plan', description: 'Usage plan for MCP Prompts API', apiStages: [{ api, stage: api.deploymentStage }], throttle: { rateLimit: 100, // requests per second burstLimit: 200 }, quota: { limit: 10000, // requests per month for free tier period: apigateway.Period.MONTH } }); usagePlan.addApiKey(apiKey); // API Gateway integration const mcpIntegration = new apigateway.LambdaIntegration(mcpServerLambda, { requestTemplates: { 'application/json': '{ "statusCode": "200" }' } }); // API routes const v1 = api.root.addResource('v1'); const prompts = v1.addResource('prompts'); prompts.addMethod('GET', mcpIntegration); // List prompts prompts.addMethod('POST', mcpIntegration); // Create prompt const promptById = prompts.addResource('{id}'); promptById.addMethod('GET', mcpIntegration); // Get prompt promptById.addMethod('PUT', mcpIntegration); // Update prompt promptById.addMethod('DELETE', mcpIntegration); // Delete prompt const promptApply = promptById.addResource('apply'); promptApply.addMethod('POST', mcpIntegration); // Apply template variables // Health endpoint const health = api.root.addResource('health'); health.addMethod('GET', mcpIntegration); // Auth endpoints const auth = api.root.addResource('auth'); const authIntegration = new apigateway.LambdaIntegration(authLambda); auth.addMethod('POST', authIntegration); // Login/Register const authRefresh = auth.addResource('refresh'); authRefresh.addMethod('POST', authIntegration); // Refresh token // User endpoints (protected) const users = api.root.addResource('users'); const userIntegration = new apigateway.LambdaIntegration(userLambda); const userById = users.addResource('{userId}'); userById.addMethod('GET', userIntegration, { authorizer: cognitoAuthorizer, authorizationType: apigateway.AuthorizationType.COGNITO }); // Get user profile userById.addMethod('PUT', userIntegration, { authorizer: cognitoAuthorizer, authorizationType: apigateway.AuthorizationType.COGNITO }); // Update user profile // User prompts endpoints (protected) const userPrompts = userById.addResource('prompts'); userPrompts.addMethod('GET', userIntegration, { authorizer: cognitoAuthorizer, authorizationType: apigateway.AuthorizationType.COGNITO }); // List user's prompts userPrompts.addMethod('POST', userIntegration, { authorizer: cognitoAuthorizer, authorizationType: apigateway.AuthorizationType.COGNITO }); // Upload prompt const userPromptById = userPrompts.addResource('{promptId}'); userPromptById.addMethod('GET', userIntegration, { authorizer: cognitoAuthorizer, authorizationType: apigateway.AuthorizationType.COGNITO }); // Get user's prompt userPromptById.addMethod('PUT', userIntegration, { authorizer: cognitoAuthorizer, authorizationType: apigateway.AuthorizationType.COGNITO }); // Update user's prompt userPromptById.addMethod('DELETE', userIntegration, { authorizer: cognitoAuthorizer, authorizationType: apigateway.AuthorizationType.COGNITO }); // Delete user's prompt // Stripe webhook endpoint (no auth required) const webhook = api.root.addResource('webhook'); const stripeWebhook = webhook.addResource('stripe'); const webhookIntegration = new apigateway.LambdaIntegration(stripeWebhookLambda); stripeWebhook.addMethod('POST', webhookIntegration); // Stripe webhook // MCP capabilities endpoint const mcp = api.root.addResource('mcp'); mcp.addMethod('GET', mcpIntegration); // MCP capabilities const tools = mcp.addResource('tools'); tools.addMethod('GET', mcpIntegration); // List tools tools.addMethod('POST', mcpIntegration); // Execute tool // S3 Bucket for web assets const webBucket = new s3.Bucket(this, 'WebBucket', { bucketName: `mcp-prompts-web-${this.account}-${this.region}`, websiteIndexDocument: 'index.html', publicReadAccess: true, blockPublicAccess: { blockPublicAcls: false, blockPublicPolicy: false, ignorePublicAcls: false, restrictPublicBuckets: false }, cors: [{ allowedOrigins: ['*'], allowedMethods: [s3.HttpMethods.GET], allowedHeaders: ['*'] }] }); // CloudFront Distribution const distribution = new cloudfront.Distribution(this, 'McpPromptsDistribution', { defaultBehavior: { origin: new origins.S3Origin(webBucket), viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS, cachePolicy: cloudfront.CachePolicy.CACHING_OPTIMIZED }, additionalBehaviors: { '/v1/*': { origin: new origins.RestApiOrigin(api), viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS, cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED, allowedMethods: cloudfront.AllowedMethods.ALLOW_ALL }, '/auth': { origin: new origins.RestApiOrigin(api), viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS, cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED, allowedMethods: cloudfront.AllowedMethods.ALLOW_ALL }, '/users/*': { origin: new origins.RestApiOrigin(api), viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS, cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED, allowedMethods: cloudfront.AllowedMethods.ALLOW_ALL }, '/mcp/*': { origin: new origins.RestApiOrigin(api), viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS, cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED, allowedMethods: cloudfront.AllowedMethods.ALLOW_ALL }, '/health': { origin: new origins.RestApiOrigin(api), viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS, cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED, allowedMethods: cloudfront.AllowedMethods.ALLOW_GET_HEAD }, '/webhook/*': { origin: new origins.RestApiOrigin(api), viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS, cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED, allowedMethods: cloudfront.AllowedMethods.ALLOW_ALL }, '/static/*': { origin: new origins.S3Origin(promptsBucket), viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS, cachePolicy: cloudfront.CachePolicy.CACHING_OPTIMIZED } } }); // Event source mappings processingLambda.addEventSource(new aws_lambda_event_sources_1.SqsEventSource(processingQueue, { batchSize: 10 })); catalogSyncLambda.addEventSource(new aws_lambda_event_sources_1.SqsEventSource(catalogSyncQueue, { batchSize: 1 })); // Permissions promptsTable.grantReadWriteData(mcpServerLambda); promptsTable.grantReadWriteData(processingLambda); promptsTable.grantReadWriteData(catalogSyncLambda); promptsTable.grantReadWriteData(authLambda); promptsTable.grantReadWriteData(userLambda); sessionsTable.grantReadWriteData(mcpServerLambda); usersTable.grantReadWriteData(authLambda); usersTable.grantReadWriteData(userLambda); usersTable.grantReadWriteData(stripeWebhookLambda); promptsBucket.grantReadWrite(mcpServerLambda); promptsBucket.grantReadWrite(processingLambda); promptsBucket.grantReadWrite(catalogSyncLambda); userPromptsBucket.grantReadWrite(authLambda); userPromptsBucket.grantReadWrite(userLambda); processingQueue.grantSendMessages(mcpServerLambda); catalogSyncQueue.grantSendMessages(mcpServerLambda); processingQueue.grantConsumeMessages(processingLambda); catalogSyncQueue.grantConsumeMessages(catalogSyncLambda); // CloudWatch permissions const cloudwatchPolicy = new iam.PolicyStatement({ actions: [ 'cloudwatch:PutMetricData', 'logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:PutLogEvents' ], resources: ['*'] }); mcpServerLambda.addToRolePolicy(cloudwatchPolicy); processingLambda.addToRolePolicy(cloudwatchPolicy); catalogSyncLambda.addToRolePolicy(cloudwatchPolicy); authLambda.addToRolePolicy(cloudwatchPolicy); userLambda.addToRolePolicy(cloudwatchPolicy); // Cognito permissions for auth lambda const cognitoPolicy = new iam.PolicyStatement({ actions: [ 'cognito-idp:AdminCreateUser', 'cognito-idp:AdminSetUserPassword', 'cognito-idp:AdminInitiateAuth', 'cognito-idp:AdminRespondToAuthChallenge', 'cognito-idp:AdminGetUser', 'cognito-idp:ListUsers', 'cognito-idp:AdminUpdateUserAttributes' ], resources: [userPool.userPoolArn] }); authLambda.addToRolePolicy(cognitoPolicy); userLambda.addToRolePolicy(cognitoPolicy); // Outputs new cdk.CfnOutput(this, 'ApiGatewayUrl', { value: api.url, description: 'API Gateway URL' }); new cdk.CfnOutput(this, 'CloudFrontUrl', { value: `https://${distribution.distributionDomainName}`, description: 'CloudFront Distribution URL (Web App)' }); new cdk.CfnOutput(this, 'WebBucketName', { value: webBucket.bucketName, description: 'S3 Bucket for web assets' }); new cdk.CfnOutput(this, 'S3BucketName', { value: promptsBucket.bucketName, description: 'S3 Bucket for prompts catalog' }); new cdk.CfnOutput(this, 'UserPromptsBucketName', { value: userPromptsBucket.bucketName, description: 'S3 Bucket for user-uploaded prompts' }); new cdk.CfnOutput(this, 'DynamoDBTableName', { value: promptsTable.tableName, description: 'DynamoDB table for prompts' }); new cdk.CfnOutput(this, 'UsersTableName', { value: usersTable.tableName, description: 'DynamoDB table for users' }); new cdk.CfnOutput(this, 'CognitoUserPoolId', { value: userPool.userPoolId, description: 'Cognito User Pool ID' }); new cdk.CfnOutput(this, 'CognitoUserPoolClientId', { value: userPoolClient.userPoolClientId, description: 'Cognito User Pool Client ID' }); new cdk.CfnOutput(this, 'ApiKey', { value: apiKey.keyId, description: 'API Gateway Key ID' }); } } exports.McpPromptsStack = McpPromptsStack; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mcp-prompts-stack.js","sourceRoot":"","sources":["mcp-prompts-stack.ts"],"names":[],"mappings":";;;;AAEA,mCAAmC;AACnC,iDAAiD;AACjD,yDAAyD;AAEzD,qDAAqD;AACrD,yCAAyC;AACzC,2CAA2C;AAC3C,2CAA2C;AAC3C,yDAAyD;AACzD,8DAA8D;AAC9D,6CAA6C;AAC7C,mDAAmD;AACnD,mFAAsE;AAGtE,MAAa,eAAgB,SAAQ,GAAG,CAAC,KAAK;IAC5C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAsB;QAC9D,KAAK,CAAC,KAAK,EAAE,EAAE,EAAE,KAAK,CAAC,CAAC;QAExB,mCAAmC;QACnC,MAAM,aAAa,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,eAAe,EAAE;YACzD,UAAU,EAAE,uBAAuB,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,MAAM,EAAE;YAChE,SAAS,EAAE,IAAI;YACf,gBAAgB,EAAE,IAAI;YACtB,iBAAiB,EAAE;gBACjB,eAAe,EAAE,KAAK;gBACtB,iBAAiB,EAAE,KAAK;gBACxB,gBAAgB,EAAE,KAAK;gBACvB,qBAAqB,EAAE,KAAK;aAC7B;YACD,IAAI,EAAE,CAAC;oBACL,cAAc,EAAE,CAAC,GAAG,CAAC;oBACrB,cAAc,EAAE,CAAC,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC;oBAC7E,cAAc,EAAE,CAAC,GAAG,CAAC;oBACrB,cAAc,EAAE,CAAC,MAAM,CAAC;iBACzB,CAAC;YACF,cAAc,EAAE,CAAC;oBACf,EAAE,EAAE,mBAAmB;oBACvB,2BAA2B,EAAE,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;iBACnD,CAAC;SACH,CAAC,CAAC;QAEH,sCAAsC;QACtC,MAAM,iBAAiB,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,mBAAmB,EAAE;YACjE,UAAU,EAAE,oBAAoB,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,MAAM,EAAE;YAC7D,SAAS,EAAE,IAAI;YACf,IAAI,EAAE,CAAC;oBACL,cAAc,EAAE,CAAC,GAAG,CAAC;oBACrB,cAAc,EAAE,CAAC,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,CAAC,WAAW,CAAC,IAAI,EAAE,EAAE,CAAC,WAAW,CAAC,MAAM,CAAC;oBACpG,cAAc,EAAE,CAAC,GAAG,CAAC;oBACrB,cAAc,EAAE,CAAC,MAAM,CAAC;iBACzB,CAAC;YACF,cAAc,EAAE,CAAC;oBACf,EAAE,EAAE,mBAAmB;oBACvB,2BAA2B,EAAE,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;iBACnD,CAAC;SACH,CAAC,CAAC;QAEH,uCAAuC;QACvC,MAAM,QAAQ,GAAG,IAAI,OAAO,CAAC,QAAQ,CAAC,IAAI,EAAE,oBAAoB,EAAE;YAChE,YAAY,EAAE,mBAAmB;YACjC,iBAAiB,EAAE,IAAI;YACvB,aAAa,EAAE;gBACb,KAAK,EAAE,IAAI;gBACX,QAAQ,EAAE,KAAK;aAChB;YACD,cAAc,EAAE;gBACd,SAAS,EAAE,CAAC;gBACZ,gBAAgB,EAAE,IAAI;gBACtB,gBAAgB,EAAE,IAAI;gBACtB,aAAa,EAAE,IAAI;gBACnB,cAAc,EAAE,KAAK;aACtB;SACF,CAAC,CAAC;QAEH,mBAAmB;QACnB,MAAM,cAAc,GAAG,IAAI,OAAO,CAAC,cAAc,CAAC,IAAI,EAAE,0BAA0B,EAAE;YAClF,QAAQ;YACR,SAAS,EAAE;gBACT,YAAY,EAAE,IAAI;gBAClB,OAAO,EAAE,IAAI;gBACb,iBAAiB,EAAE,IAAI;aACxB;YACD,cAAc,EAAE,KAAK;SACtB,CAAC,CAAC;QAEH,sCAAsC;QACtC,MAAM,YAAY,GAAG,IAAI,QAAQ,CAAC,KAAK,CAAC,IAAI,EAAE,cAAc,EAAE;YAC5D,SAAS,EAAE,aAAa;YACxB,YAAY,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;YACjE,OAAO,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;YACjE,WAAW,EAAE,QAAQ,CAAC,WAAW,CAAC,eAAe;YACjD,mBAAmB,EAAE,IAAI;YACzB,MAAM,EAAE,QAAQ,CAAC,cAAc,CAAC,kBAAkB;SACnD,CAAC,CAAC;QAEH,+BAA+B;QAC/B,YAAY,CAAC,uBAAuB,CAAC;YACnC,SAAS,EAAE,gBAAgB;YAC3B,YAAY,EAAE,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;YACvE,OAAO,EAAE,EAAE,IAAI,EAAE,YAAY,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;SACrE,CAAC,CAAC;QAEH,YAAY,CAAC,uBAAuB,CAAC;YACnC,SAAS,EAAE,sBAAsB;YACjC,YAAY,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;YACjE,OAAO,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;SACpE,CAAC,CAAC;QAEH,YAAY,CAAC,uBAAuB,CAAC;YACnC,SAAS,EAAE,oBAAoB;YAC/B,YAAY,EAAE,EAAE,IAAI,EAAE,cAAc,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;YAC3E,OAAO,EAAE,EAAE,IAAI,EAAE,YAAY,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;SACrE,CAAC,CAAC;QAEH,YAAY,CAAC,uBAAuB,CAAC;YACnC,SAAS,EAAE,cAAc;YACzB,YAAY,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;YACxE,OAAO,EAAE,EAAE,IAAI,EAAE,YAAY,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;SACrE,CAAC,CAAC;QAEH,0CAA0C;QAC1C,MAAM,aAAa,GAAG,IAAI,QAAQ,CAAC,KAAK,CAAC,IAAI,EAAE,eAAe,EAAE;YAC9D,SAAS,EAAE,cAAc;YACzB,YAAY,EAAE,EAAE,IAAI,EAAE,YAAY,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;YACzE,WAAW,EAAE,QAAQ,CAAC,WAAW,CAAC,eAAe;YACjD,mBAAmB,EAAE,YAAY;SAClC,CAAC,CAAC;QAEH,6CAA6C;QAC7C,MAAM,UAAU,GAAG,IAAI,QAAQ,CAAC,KAAK,CAAC,IAAI,EAAE,YAAY,EAAE;YACxD,SAAS,EAAE,WAAW;YACtB,YAAY,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;YACtE,WAAW,EAAE,QAAQ,CAAC,WAAW,CAAC,eAAe;YACjD,mBAAmB,EAAE,IAAI;SAC1B,CAAC,CAAC;QAEH,UAAU,CAAC,uBAAuB,CAAC;YACjC,SAAS,EAAE,aAAa;YACxB,YAAY,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;SACrE,CAAC,CAAC;QAEH,UAAU,CAAC,uBAAuB,CAAC;YACjC,SAAS,EAAE,oBAAoB;YAC/B,YAAY,EAAE,EAAE,IAAI,EAAE,mBAAmB,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;YAChF,OAAO,EAAE,EAAE,IAAI,EAAE,yBAAyB,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;SAClF,CAAC,CAAC;QAEH,iCAAiC;QACjC,MAAM,aAAa,GAAG,IAAI,GAAG,CAAC,KAAK,CAAC,IAAI,EAAE,eAAe,EAAE;YACzD,SAAS,EAAE,4BAA4B;YACvC,eAAe,EAAE,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;SACvC,CAAC,CAAC;QAEH,MAAM,eAAe,GAAG,IAAI,GAAG,CAAC,KAAK,CAAC,IAAI,EAAE,iBAAiB,EAAE;YAC7D,SAAS,EAAE,wBAAwB;YACnC,iBAAiB,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAC1C,eAAe,EAAE;gBACf,KAAK,EAAE,aAAa;gBACpB,eAAe,EAAE,CAAC;aACnB;SACF,CAAC,CAAC;QAEH,6BAA6B;QAC7B,MAAM,gBAAgB,GAAG,IAAI,GAAG,CAAC,KAAK,CAAC,IAAI,EAAE,kBAAkB,EAAE;YAC/D,SAAS,EAAE,kBAAkB;YAC7B,iBAAiB,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;SAC5C,CAAC,CAAC;QAEH,uCAAuC;QACvC,MAAM,WAAW,GAAG,IAAI,MAAM,CAAC,YAAY,CAAC,IAAI,EAAE,aAAa,EAAE;YAC/D,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,kBAAkB,CAAC;YAC/C,kBAAkB,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC;YAChD,WAAW,EAAE,qCAAqC;SACnD,CAAC,CAAC;QAEH,yBAAyB;QACzB,MAAM,eAAe,GAAG,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,mBAAmB,EAAE;YACrE,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC;YACtC,OAAO,EAAE,2BAA2B;YACpC,MAAM,EAAE,CAAC,WAAW,CAAC;YACrB,WAAW,EAAE;gBACX,aAAa,EAAE,YAAY,CAAC,SAAS;gBACrC,cAAc,EAAE,aAAa,CAAC,SAAS;gBACvC,cAAc,EAAE,aAAa,CAAC,UAAU;gBACxC,gBAAgB,EAAE,eAAe,CAAC,QAAQ;gBAC1C,kBAAkB,EAAE,gBAAgB,CAAC,QAAQ;gBAC7C,QAAQ,EAAE,YAAY;gBACtB,SAAS,EAAE,MAAM;aAClB;YACD,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACjC,UAAU,EAAE,GAAG;YACf,YAAY,EAAE,IAAI,CAAC,aAAa,CAAC,QAAQ;SAC1C,CAAC,CAAC;QAEH,oCAAoC;QACpC,MAAM,gBAAgB,GAAG,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,oBAAoB,EAAE;YACvE,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC;YACtC,OAAO,EAAE,0BAA0B;YACnC,MAAM,EAAE,CAAC,WAAW,CAAC;YACrB,WAAW,EAAE;gBACX,aAAa,EAAE,YAAY,CAAC,SAAS;gBACrC,cAAc,EAAE,aAAa,CAAC,SAAS;gBACvC,cAAc,EAAE,aAAa,CAAC,UAAU;aACzC;YACD,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAChC,UAAU,EAAE,IAAI;YAChB,YAAY,EAAE,IAAI,CAAC,aAAa,CAAC,QAAQ;SAC1C,CAAC,CAAC;QAEH,sBAAsB;QACtB,MAAM,iBAAiB,GAAG,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,qBAAqB,EAAE;YACzE,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC;YACtC,OAAO,EAAE,6BAA6B;YACtC,MAAM,EAAE,CAAC,WAAW,CAAC;YACrB,WAAW,EAAE;gBACX,aAAa,EAAE,YAAY,CAAC,SAAS;gBACrC,cAAc,EAAE,aAAa,CAAC,UAAU;gBACxC,eAAe,EAAE,qDAAqD;aACvE;YACD,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACjC,UAAU,EAAE,IAAI;YAChB,YAAY,EAAE,IAAI,CAAC,aAAa,CAAC,QAAQ;SAC1C,CAAC,CAAC;QAEH,sCAAsC;QACtC,MAAM,UAAU,GAAG,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,cAAc,EAAE;YAC3D,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC;YACtC,OAAO,EAAE,qBAAqB;YAC9B,MAAM,EAAE,CAAC,WAAW,CAAC;YACrB,WAAW,EAAE;gBACX,YAAY,EAAE,QAAQ,CAAC,UAAU;gBACjC,mBAAmB,EAAE,cAAc,CAAC,gBAAgB;gBACpD,WAAW,EAAE,UAAU,CAAC,SAAS;aAClC;YACD,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACjC,UAAU,EAAE,GAAG;YACf,YAAY,EAAE,IAAI,CAAC,aAAa,CAAC,QAAQ;SAC1C,CAAC,CAAC;QAEH,yBAAyB;QACzB,MAAM,UAAU,GAAG,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,cAAc,EAAE;YAC3D,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC;YACtC,OAAO,EAAE,qBAAqB;YAC9B,MAAM,EAAE,CAAC,WAAW,CAAC;YACrB,WAAW,EAAE;gBACX,YAAY,EAAE,QAAQ,CAAC,UAAU;gBACjC,mBAAmB,EAAE,cAAc,CAAC,gBAAgB;gBACpD,WAAW,EAAE,UAAU,CAAC,SAAS;gBACjC,mBAAmB,EAAE,iBAAiB,CAAC,UAAU;aAClD;YACD,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACjC,UAAU,EAAE,GAAG;YACf,YAAY,EAAE,IAAI,CAAC,aAAa,CAAC,QAAQ;SAC1C,CAAC,CAAC;QAEH,wBAAwB;QACxB,MAAM,mBAAmB,GAAG,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,uBAAuB,EAAE;YAC7E,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC;YACtC,OAAO,EAAE,+BAA+B;YACxC,MAAM,EAAE,CAAC,WAAW,CAAC;YACrB,WAAW,EAAE;gBACX,WAAW,EAAE,UAAU,CAAC,SAAS;gBACjC,qBAAqB,EAAE,OAAO,CAAC,GAAG,CAAC,qBAAqB,IAAI,EAAE;aAC/D;YACD,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACjC,UAAU,EAAE,GAAG;YACf,YAAY,EAAE,IAAI,CAAC,aAAa,CAAC,QAAQ;SAC1C,CAAC,CAAC;QAEH,qBAAqB;QACrB,MAAM,iBAAiB,GAAG,IAAI,UAAU,CAAC,0BAA0B,CAAC,IAAI,EAAE,mBAAmB,EAAE;YAC7F,gBAAgB,EAAE,CAAC,QAAQ,CAAC;SAC7B,CAAC,CAAC;QAEH,cAAc;QACd,MAAM,GAAG,GAAG,IAAI,UAAU,CAAC,OAAO,CAAC,IAAI,EAAE,eAAe,EAAE;YACxD,WAAW,EAAE,qBAAqB;YAClC,WAAW,EAAE,sCAAsC;YACnD,2BAA2B,EAAE;gBAC3B,YAAY,EAAE,UAAU,CAAC,IAAI,CAAC,WAAW;gBACzC,YAAY,EAAE,UAAU,CAAC,IAAI,CAAC,WAAW;gBACzC,YAAY,EAAE,CAAC,cAAc,EAAE,YAAY,EAAE,eAAe,EAAE,WAAW,EAAE,sBAAsB,CAAC;aACnG;YACD,qBAAqB,EAAE;gBACrB,KAAK,EAAE,CAAC,UAAU,CAAC,YAAY,CAAC,QAAQ,CAAC;aAC1C;SACF,CAAC,CAAC;QAEH,4BAA4B;QAC5B,MAAM,MAAM,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC,IAAI,EAAE,kBAAkB,EAAE;YAC7D,UAAU,EAAE,qBAAqB;YACjC,WAAW,EAAE,yBAAyB;SACvC,CAAC,CAAC;QAEH,+BAA+B;QAC/B,MAAM,SAAS,GAAG,IAAI,UAAU,CAAC,SAAS,CAAC,IAAI,EAAE,qBAAqB,EAAE;YACtE,IAAI,EAAE,wBAAwB;YAC9B,WAAW,EAAE,gCAAgC;YAC7C,SAAS,EAAE,CAAC;oBACV,GAAG;oBACH,KAAK,EAAE,GAAG,CAAC,eAAe;iBAC3B,CAAC;YACF,QAAQ,EAAE;gBACR,SAAS,EAAE,GAAG,EAAE,sBAAsB;gBACtC,UAAU,EAAE,GAAG;aAChB;YACD,KAAK,EAAE;gBACL,KAAK,EAAE,KAAK,EAAE,mCAAmC;gBACjD,MAAM,EAAE,UAAU,CAAC,MAAM,CAAC,KAAK;aAChC;SACF,CAAC,CAAC;QAEH,SAAS,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;QAE5B,0BAA0B;QAC1B,MAAM,cAAc,GAAG,IAAI,UAAU,CAAC,iBAAiB,CAAC,eAAe,EAAE;YACvE,gBAAgB,EAAE,EAAE,kBAAkB,EAAE,yBAAyB,EAAE;SACpE,CAAC,CAAC;QAEH,aAAa;QACb,MAAM,EAAE,GAAG,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QACtC,MAAM,OAAO,GAAG,EAAE,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC;QAE1C,OAAO,CAAC,SAAS,CAAC,KAAK,EAAE,cAAc,CAAC,CAAC,CAAC,eAAe;QACzD,OAAO,CAAC,SAAS,CAAC,MAAM,EAAE,cAAc,CAAC,CAAC,CAAC,gBAAgB;QAE3D,MAAM,UAAU,GAAG,OAAO,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;QAC/C,UAAU,CAAC,SAAS,CAAC,KAAK,EAAE,cAAc,CAAC,CAAC,CAAC,aAAa;QAC1D,UAAU,CAAC,SAAS,CAAC,KAAK,EAAE,cAAc,CAAC,CAAC,CAAC,gBAAgB;QAC7D,UAAU,CAAC,SAAS,CAAC,QAAQ,EAAE,cAAc,CAAC,CAAC,CAAC,gBAAgB;QAEhE,MAAM,WAAW,GAAG,UAAU,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;QACpD,WAAW,CAAC,SAAS,CAAC,MAAM,EAAE,cAAc,CAAC,CAAC,CAAC,2BAA2B;QAE1E,kBAAkB;QAClB,MAAM,MAAM,GAAG,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;QAC9C,MAAM,CAAC,SAAS,CAAC,KAAK,EAAE,cAAc,CAAC,CAAC;QAExC,iBAAiB;QACjB,MAAM,IAAI,GAAG,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;QAC1C,MAAM,eAAe,GAAG,IAAI,UAAU,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;QAErE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,eAAe,CAAC,CAAC,CAAC,iBAAiB;QAE1D,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC;QAChD,WAAW,CAAC,SAAS,CAAC,MAAM,EAAE,eAAe,CAAC,CAAC,CAAC,gBAAgB;QAEhE,6BAA6B;QAC7B,MAAM,KAAK,GAAG,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;QAC5C,MAAM,eAAe,GAAG,IAAI,UAAU,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;QAErE,MAAM,QAAQ,GAAG,KAAK,CAAC,WAAW,CAAC,UAAU,CAAC,CAAC;QAC/C,QAAQ,CAAC,SAAS,CAAC,KAAK,EAAE,eAAe,EAAE;YACzC,UAAU,EAAE,iBAAiB;YAC7B,iBAAiB,EAAE,UAAU,CAAC,iBAAiB,CAAC,OAAO;SACxD,CAAC,CAAC,CAAC,mBAAmB;QAEvB,QAAQ,CAAC,SAAS,CAAC,KAAK,EAAE,eAAe,EAAE;YACzC,UAAU,EAAE,iBAAiB;YAC7B,iBAAiB,EAAE,UAAU,CAAC,iBAAiB,CAAC,OAAO;SACxD,CAAC,CAAC,CAAC,sBAAsB;QAE1B,qCAAqC;QACrC,MAAM,WAAW,GAAG,QAAQ,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC;QACpD,WAAW,CAAC,SAAS,CAAC,KAAK,EAAE,eAAe,EAAE;YAC5C,UAAU,EAAE,iBAAiB;YAC7B,iBAAiB,EAAE,UAAU,CAAC,iBAAiB,CAAC,OAAO;SACxD,CAAC,CAAC,CAAC,sBAAsB;QAE1B,WAAW,CAAC,SAAS,CAAC,MAAM,EAAE,eAAe,EAAE;YAC7C,UAAU,EAAE,iBAAiB;YAC7B,iBAAiB,EAAE,UAAU,CAAC,iBAAiB,CAAC,OAAO;SACxD,CAAC,CAAC,CAAC,gBAAgB;QAEpB,MAAM,cAAc,GAAG,WAAW,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC;QAC7D,cAAc,CAAC,SAAS,CAAC,KAAK,EAAE,eAAe,EAAE;YAC/C,UAAU,EAAE,iBAAiB;YAC7B,iBAAiB,EAAE,UAAU,CAAC,iBAAiB,CAAC,OAAO;SACxD,CAAC,CAAC,CAAC,oBAAoB;QAExB,cAAc,CAAC,SAAS,CAAC,KAAK,EAAE,eAAe,EAAE;YAC/C,UAAU,EAAE,iBAAiB;YAC7B,iBAAiB,EAAE,UAAU,CAAC,iBAAiB,CAAC,OAAO;SACxD,CAAC,CAAC,CAAC,uBAAuB;QAE3B,cAAc,CAAC,SAAS,CAAC,QAAQ,EAAE,eAAe,EAAE;YAClD,UAAU,EAAE,iBAAiB;YAC7B,iBAAiB,EAAE,UAAU,CAAC,iBAAiB,CAAC,OAAO;SACxD,CAAC,CAAC,CAAC,uBAAuB;QAE3B,6CAA6C;QAC7C,MAAM,OAAO,GAAG,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC;QAChD,MAAM,aAAa,GAAG,OAAO,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;QACpD,MAAM,kBAAkB,GAAG,IAAI,UAAU,CAAC,iBAAiB,CAAC,mBAAmB,CAAC,CAAC;QAEjF,aAAa,CAAC,SAAS,CAAC,MAAM,EAAE,kBAAkB,CAAC,CAAC,CAAC,iBAAiB;QAEtE,4BAA4B;QAC5B,MAAM,GAAG,GAAG,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;QACxC,GAAG,CAAC,SAAS,CAAC,KAAK,EAAE,cAAc,CAAC,CAAC,CAAC,mBAAmB;QACzD,MAAM,KAAK,GAAG,GAAG,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;QACvC,KAAK,CAAC,SAAS,CAAC,KAAK,EAAE,cAAc,CAAC,CAAC,CAAC,aAAa;QACrD,KAAK,CAAC,SAAS,CAAC,MAAM,EAAE,cAAc,CAAC,CAAC,CAAC,eAAe;QAExD,2BAA2B;QAC3B,MAAM,SAAS,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,WAAW,EAAE;YACjD,UAAU,EAAE,mBAAmB,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,MAAM,EAAE;YAC5D,oBAAoB,EAAE,YAAY;YAClC,gBAAgB,EAAE,IAAI;YACtB,iBAAiB,EAAE;gBACjB,eAAe,EAAE,KAAK;gBACtB,iBAAiB,EAAE,KAAK;gBACxB,gBAAgB,EAAE,KAAK;gBACvB,qBAAqB,EAAE,KAAK;aAC7B;YACD,IAAI,EAAE,CAAC;oBACL,cAAc,EAAE,CAAC,GAAG,CAAC;oBACrB,cAAc,EAAE,CAAC,EAAE,CAAC,WAAW,CAAC,GAAG,CAAC;oBACpC,cAAc,EAAE,CAAC,GAAG,CAAC;iBACtB,CAAC;SACH,CAAC,CAAC;QAEH,0BAA0B;QAC1B,MAAM,YAAY,GAAG,IAAI,UAAU,CAAC,YAAY,CAAC,IAAI,EAAE,wBAAwB,EAAE;YAC/E,eAAe,EAAE;gBACf,MAAM,EAAE,IAAI,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC;gBACvC,oBAAoB,EAAE,UAAU,CAAC,oBAAoB,CAAC,iBAAiB;gBACvE,WAAW,EAAE,UAAU,CAAC,WAAW,CAAC,iBAAiB;aACtD;YACD,mBAAmB,EAAE;gBACnB,OAAO,EAAE;oBACP,MAAM,EAAE,IAAI,OAAO,CAAC,aAAa,CAAC,GAAG,CAAC;oBACtC,oBAAoB,EAAE,UAAU,CAAC,oBAAoB,CAAC,iBAAiB;oBACvE,WAAW,EAAE,UAAU,CAAC,WAAW,CAAC,gBAAgB;oBACpD,cAAc,EAAE,UAAU,CAAC,cAAc,CAAC,SAAS;iBACpD;gBACD,OAAO,EAAE;oBACP,MAAM,EAAE,IAAI,OAAO,CAAC,aAAa,CAAC,GAAG,CAAC;oBACtC,oBAAoB,EAAE,UAAU,CAAC,oBAAoB,CAAC,iBAAiB;oBACvE,WAAW,EAAE,UAAU,CAAC,WAAW,CAAC,gBAAgB;oBACpD,cAAc,EAAE,UAAU,CAAC,cAAc,CAAC,SAAS;iBACpD;gBACD,UAAU,EAAE;oBACV,MAAM,EAAE,IAAI,OAAO,CAAC,aAAa,CAAC,GAAG,CAAC;oBACtC,oBAAoB,EAAE,UAAU,CAAC,oBAAoB,CAAC,iBAAiB;oBACvE,WAAW,EAAE,UAAU,CAAC,WAAW,CAAC,gBAAgB;oBACpD,cAAc,EAAE,UAAU,CAAC,cAAc,CAAC,SAAS;iBACpD;gBACD,QAAQ,EAAE;oBACR,MAAM,EAAE,IAAI,OAAO,CAAC,aAAa,CAAC,GAAG,CAAC;oBACtC,oBAAoB,EAAE,UAAU,CAAC,oBAAoB,CAAC,iBAAiB;oBACvE,WAAW,EAAE,UAAU,CAAC,WAAW,CAAC,gBAAgB;oBACpD,cAAc,EAAE,UAAU,CAAC,cAAc,CAAC,SAAS;iBACpD;gBACD,SAAS,EAAE;oBACT,MAAM,EAAE,IAAI,OAAO,CAAC,aAAa,CAAC,GAAG,CAAC;oBACtC,oBAAoB,EAAE,UAAU,CAAC,oBAAoB,CAAC,iBAAiB;oBACvE,WAAW,EAAE,UAAU,CAAC,WAAW,CAAC,gBAAgB;oBACpD,cAAc,EAAE,UAAU,CAAC,cAAc,CAAC,cAAc;iBACzD;gBACD,YAAY,EAAE;oBACZ,MAAM,EAAE,IAAI,OAAO,CAAC,aAAa,CAAC,GAAG,CAAC;oBACtC,oBAAoB,EAAE,UAAU,CAAC,oBAAoB,CAAC,iBAAiB;oBACvE,WAAW,EAAE,UAAU,CAAC,WAAW,CAAC,gBAAgB;oBACpD,cAAc,EAAE,UAAU,CAAC,cAAc,CAAC,SAAS;iBACpD;gBACD,WAAW,EAAE;oBACX,MAAM,EAAE,IAAI,OAAO,CAAC,QAAQ,CAAC,aAAa,CAAC;oBAC3C,oBAAoB,EAAE,UAAU,CAAC,oBAAoB,CAAC,iBAAiB;oBACvE,WAAW,EAAE,UAAU,CAAC,WAAW,CAAC,iBAAiB;iBACtD;aACF;SACF,CAAC,CAAC;QAEH,wBAAwB;QACxB,gBAAgB,CAAC,cAAc,CAAC,IAAI,yCAAc,CAAC,eAAe,EAAE;YAClE,SAAS,EAAE,EAAE;SACd,CAAC,CAAC,CAAC;QAEJ,iBAAiB,CAAC,cAAc,CAAC,IAAI,yCAAc,CAAC,gBAAgB,EAAE;YACpE,SAAS,EAAE,CAAC;SACb,CAAC,CAAC,CAAC;QAEJ,cAAc;QACd,YAAY,CAAC,kBAAkB,CAAC,eAAe,CAAC,CAAC;QACjD,YAAY,CAAC,kBAAkB,CAAC,gBAAgB,CAAC,CAAC;QAClD,YAAY,CAAC,kBAAkB,CAAC,iBAAiB,CAAC,CAAC;QACnD,YAAY,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;QAC5C,YAAY,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;QAE5C,aAAa,CAAC,kBAAkB,CAAC,eAAe,CAAC,CAAC;QAElD,UAAU,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;QAC1C,UAAU,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;QAC1C,UAAU,CAAC,kBAAkB,CAAC,mBAAmB,CAAC,CAAC;QAEnD,aAAa,CAAC,cAAc,CAAC,eAAe,CAAC,CAAC;QAC9C,aAAa,CAAC,cAAc,CAAC,gBAAgB,CAAC,CAAC;QAC/C,aAAa,CAAC,cAAc,CAAC,iBAAiB,CAAC,CAAC;QAEhD,iBAAiB,CAAC,cAAc,CAAC,UAAU,CAAC,CAAC;QAC7C,iBAAiB,CAAC,cAAc,CAAC,UAAU,CAAC,CAAC;QAE7C,eAAe,CAAC,iBAAiB,CAAC,eAAe,CAAC,CAAC;QACnD,gBAAgB,CAAC,iBAAiB,CAAC,eAAe,CAAC,CAAC;QAEpD,eAAe,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,CAAC;QACvD,gBAAgB,CAAC,oBAAoB,CAAC,iBAAiB,CAAC,CAAC;QAEzD,yBAAyB;QACzB,MAAM,gBAAgB,GAAG,IAAI,GAAG,CAAC,eAAe,CAAC;YAC/C,OAAO,EAAE;gBACP,0BAA0B;gBAC1B,qBAAqB;gBACrB,sBAAsB;gBACtB,mBAAmB;aACpB;YACD,SAAS,EAAE,CAAC,GAAG,CAAC;SACjB,CAAC,CAAC;QAEH,eAAe,CAAC,eAAe,CAAC,gBAAgB,CAAC,CAAC;QAClD,gBAAgB,CAAC,eAAe,CAAC,gBAAgB,CAAC,CAAC;QACnD,iBAAiB,CAAC,eAAe,CAAC,gBAAgB,CAAC,CAAC;QACpD,UAAU,CAAC,eAAe,CAAC,gBAAgB,CAAC,CAAC;QAC7C,UAAU,CAAC,eAAe,CAAC,gBAAgB,CAAC,CAAC;QAE7C,sCAAsC;QACtC,MAAM,aAAa,GAAG,IAAI,GAAG,CAAC,eAAe,CAAC;YAC5C,OAAO,EAAE;gBACP,6BAA6B;gBAC7B,kCAAkC;gBAClC,+BAA+B;gBAC/B,yCAAyC;gBACzC,0BAA0B;gBAC1B,uBAAuB;gBACvB,uCAAuC;aACxC;YACD,SAAS,EAAE,CAAC,QAAQ,CAAC,WAAW,CAAC;SAClC,CAAC,CAAC;QAEH,UAAU,CAAC,eAAe,CAAC,aAAa,CAAC,CAAC;QAC1C,UAAU,CAAC,eAAe,CAAC,aAAa,CAAC,CAAC;QAE1C,UAAU;QACV,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,eAAe,EAAE;YACvC,KAAK,EAAE,GAAG,CAAC,GAAG;YACd,WAAW,EAAE,iBAAiB;SAC/B,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,eAAe,EAAE;YACvC,KAAK,EAAE,WAAW,YAAY,CAAC,sBAAsB,EAAE;YACvD,WAAW,EAAE,uCAAuC;SACrD,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,eAAe,EAAE;YACvC,KAAK,EAAE,SAAS,CAAC,UAAU;YAC3B,WAAW,EAAE,0BAA0B;SACxC,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,cAAc,EAAE;YACtC,KAAK,EAAE,aAAa,CAAC,UAAU;YAC/B,WAAW,EAAE,+BAA+B;SAC7C,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,uBAAuB,EAAE;YAC/C,KAAK,EAAE,iBAAiB,CAAC,UAAU;YACnC,WAAW,EAAE,qCAAqC;SACnD,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,mBAAmB,EAAE;YAC3C,KAAK,EAAE,YAAY,CAAC,SAAS;YAC7B,WAAW,EAAE,4BAA4B;SAC1C,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,gBAAgB,EAAE;YACxC,KAAK,EAAE,UAAU,CAAC,SAAS;YAC3B,WAAW,EAAE,0BAA0B;SACxC,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,mBAAmB,EAAE;YAC3C,KAAK,EAAE,QAAQ,CAAC,UAAU;YAC1B,WAAW,EAAE,sBAAsB;SACpC,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,yBAAyB,EAAE;YACjD,KAAK,EAAE,cAAc,CAAC,gBAAgB;YACtC,WAAW,EAAE,6BAA6B;SAC3C,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,QAAQ,EAAE;YAChC,KAAK,EAAE,MAAM,CAAC,KAAK;YACnB,WAAW,EAAE,oBAAoB;SAClC,CAAC,CAAC;IACL,CAAC;CACF;AA1kBD,0CA0kBC","sourcesContent":["#!/usr/bin/env node\n\nimport * as cdk from 'aws-cdk-lib';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as apigateway from 'aws-cdk-lib/aws-apigateway';\nimport * as apigatewayv2 from 'aws-cdk-lib/aws-apigatewayv2';\nimport * as dynamodb from 'aws-cdk-lib/aws-dynamodb';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport * as sqs from 'aws-cdk-lib/aws-sqs';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as cloudfront from 'aws-cdk-lib/aws-cloudfront';\nimport * as origins from 'aws-cdk-lib/aws-cloudfront-origins';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport * as cognito from 'aws-cdk-lib/aws-cognito';\nimport { SqsEventSource } from 'aws-cdk-lib/aws-lambda-event-sources';\nimport { Construct } from 'constructs';\n\nexport class McpPromptsStack extends cdk.Stack {\n  constructor(scope: Construct, id: string, props?: cdk.StackProps) {\n    super(scope, id, props);\n\n    // S3 Bucket for catalog and assets\n    const promptsBucket = new s3.Bucket(this, 'PromptsBucket', {\n      bucketName: `mcp-prompts-catalog-${this.account}-${this.region}`,\n      versioned: true,\n      publicReadAccess: true,\n      blockPublicAccess: {\n        blockPublicAcls: false,\n        blockPublicPolicy: false,\n        ignorePublicAcls: false,\n        restrictPublicBuckets: false\n      },\n      cors: [{\n        allowedOrigins: ['*'],\n        allowedMethods: [s3.HttpMethods.GET, s3.HttpMethods.PUT, s3.HttpMethods.POST],\n        allowedHeaders: ['*'],\n        exposedHeaders: ['ETag']\n      }],\n      lifecycleRules: [{\n        id: 'DeleteOldVersions',\n        noncurrentVersionExpiration: cdk.Duration.days(30)\n      }]\n    });\n\n    // S3 Bucket for user-uploaded prompts\n    const userPromptsBucket = new s3.Bucket(this, 'UserPromptsBucket', {\n      bucketName: `mcp-user-prompts-${this.account}-${this.region}`,\n      versioned: true,\n      cors: [{\n        allowedOrigins: ['*'],\n        allowedMethods: [s3.HttpMethods.GET, s3.HttpMethods.PUT, s3.HttpMethods.POST, s3.HttpMethods.DELETE],\n        allowedHeaders: ['*'],\n        exposedHeaders: ['ETag']\n      }],\n      lifecycleRules: [{\n        id: 'DeleteOldVersions',\n        noncurrentVersionExpiration: cdk.Duration.days(30)\n      }]\n    });\n\n    // Cognito User Pool for authentication\n    const userPool = new cognito.UserPool(this, 'McpPromptsUserPool', {\n      userPoolName: 'mcp-prompts-users',\n      selfSignUpEnabled: true,\n      signInAliases: {\n        email: true,\n        username: false\n      },\n      passwordPolicy: {\n        minLength: 8,\n        requireLowercase: true,\n        requireUppercase: true,\n        requireDigits: true,\n        requireSymbols: false\n      }\n    });\n\n    // User Pool Client\n    const userPoolClient = new cognito.UserPoolClient(this, 'McpPromptsUserPoolClient', {\n      userPool,\n      authFlows: {\n        userPassword: true,\n        userSrp: true,\n        adminUserPassword: true\n      },\n      generateSecret: false\n    });\n\n    // DynamoDB Table for prompts metadata\n    const promptsTable = new dynamodb.Table(this, 'PromptsTable', {\n      tableName: 'mcp-prompts',\n      partitionKey: { name: 'id', type: dynamodb.AttributeType.STRING },\n      sortKey: { name: 'version', type: dynamodb.AttributeType.STRING },\n      billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,\n      pointInTimeRecovery: true,\n      stream: dynamodb.StreamViewType.NEW_AND_OLD_IMAGES\n    });\n\n    // Add Global Secondary Indexes\n    promptsTable.addGlobalSecondaryIndex({\n      indexName: 'category-index',\n      partitionKey: { name: 'category', type: dynamodb.AttributeType.STRING },\n      sortKey: { name: 'created_at', type: dynamodb.AttributeType.STRING }\n    });\n\n    promptsTable.addGlobalSecondaryIndex({\n      indexName: 'latest-version-index',\n      partitionKey: { name: 'id', type: dynamodb.AttributeType.STRING },\n      sortKey: { name: 'is_latest', type: dynamodb.AttributeType.STRING }\n    });\n\n    promptsTable.addGlobalSecondaryIndex({\n      indexName: 'access-level-index',\n      partitionKey: { name: 'access_level', type: dynamodb.AttributeType.STRING },\n      sortKey: { name: 'created_at', type: dynamodb.AttributeType.STRING }\n    });\n\n    promptsTable.addGlobalSecondaryIndex({\n      indexName: 'author-index',\n      partitionKey: { name: 'author_id', type: dynamodb.AttributeType.STRING },\n      sortKey: { name: 'created_at', type: dynamodb.AttributeType.STRING }\n    });\n\n    // DynamoDB Table for sessions and caching\n    const sessionsTable = new dynamodb.Table(this, 'SessionsTable', {\n      tableName: 'mcp-sessions',\n      partitionKey: { name: 'session_id', type: dynamodb.AttributeType.STRING },\n      billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,\n      timeToLiveAttribute: 'expires_at'\n    });\n\n    // DynamoDB Table for users and subscriptions\n    const usersTable = new dynamodb.Table(this, 'UsersTable', {\n      tableName: 'mcp-users',\n      partitionKey: { name: 'user_id', type: dynamodb.AttributeType.STRING },\n      billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,\n      pointInTimeRecovery: true\n    });\n\n    usersTable.addGlobalSecondaryIndex({\n      indexName: 'email-index',\n      partitionKey: { name: 'email', type: dynamodb.AttributeType.STRING }\n    });\n\n    usersTable.addGlobalSecondaryIndex({\n      indexName: 'subscription-index',\n      partitionKey: { name: 'subscription_tier', type: dynamodb.AttributeType.STRING },\n      sortKey: { name: 'subscription_expires_at', type: dynamodb.AttributeType.STRING }\n    });\n\n    // SQS Queue for async processing\n    const processingDLQ = new sqs.Queue(this, 'ProcessingDLQ', {\n      queueName: 'mcp-prompts-processing-dlq',\n      retentionPeriod: cdk.Duration.days(14)\n    });\n\n    const processingQueue = new sqs.Queue(this, 'ProcessingQueue', {\n      queueName: 'mcp-prompts-processing',\n      visibilityTimeout: cdk.Duration.minutes(5),\n      deadLetterQueue: {\n        queue: processingDLQ,\n        maxReceiveCount: 3\n      }\n    });\n\n    // SQS Queue for catalog sync\n    const catalogSyncQueue = new sqs.Queue(this, 'CatalogSyncQueue', {\n      queueName: 'mcp-catalog-sync',\n      visibilityTimeout: cdk.Duration.minutes(10)\n    });\n\n    // Lambda Layer for common dependencies\n    const commonLayer = new lambda.LayerVersion(this, 'CommonLayer', {\n      code: lambda.Code.fromAsset('../layers/common'),\n      compatibleRuntimes: [lambda.Runtime.NODEJS_18_X],\n      description: 'Common dependencies for MCP Prompts'\n    });\n\n    // Main MCP Server Lambda\n    const mcpServerLambda = new lambda.Function(this, 'McpServerFunction', {\n      runtime: lambda.Runtime.NODEJS_18_X,\n      code: lambda.Code.fromAsset('../dist'),\n      handler: 'lambda/mcp-server.handler',\n      layers: [commonLayer],\n      environment: {\n        PROMPTS_TABLE: promptsTable.tableName,\n        SESSIONS_TABLE: sessionsTable.tableName,\n        PROMPTS_BUCKET: promptsBucket.bucketName,\n        PROCESSING_QUEUE: processingQueue.queueUrl,\n        CATALOG_SYNC_QUEUE: catalogSyncQueue.queueUrl,\n        NODE_ENV: 'production',\n        LOG_LEVEL: 'info'\n      },\n      timeout: cdk.Duration.seconds(30),\n      memorySize: 512,\n      logRetention: logs.RetentionDays.ONE_WEEK\n    });\n\n    // Processing Lambda for async tasks\n    const processingLambda = new lambda.Function(this, 'ProcessingFunction', {\n      runtime: lambda.Runtime.NODEJS_18_X,\n      code: lambda.Code.fromAsset('../dist'),\n      handler: 'lambda/processor.handler',\n      layers: [commonLayer],\n      environment: {\n        PROMPTS_TABLE: promptsTable.tableName,\n        SESSIONS_TABLE: sessionsTable.tableName,\n        PROMPTS_BUCKET: promptsBucket.bucketName\n      },\n      timeout: cdk.Duration.minutes(5),\n      memorySize: 1024,\n      logRetention: logs.RetentionDays.ONE_WEEK\n    });\n\n    // Catalog Sync Lambda\n    const catalogSyncLambda = new lambda.Function(this, 'CatalogSyncFunction', {\n      runtime: lambda.Runtime.NODEJS_18_X,\n      code: lambda.Code.fromAsset('../dist'),\n      handler: 'lambda/catalog-sync.handler',\n      layers: [commonLayer],\n      environment: {\n        PROMPTS_TABLE: promptsTable.tableName,\n        PROMPTS_BUCKET: promptsBucket.bucketName,\n        GITHUB_REPO_URL: 'https://github.com/sparesparrow/mcp-prompts-catalog'\n      },\n      timeout: cdk.Duration.minutes(10),\n      memorySize: 1024,\n      logRetention: logs.RetentionDays.ONE_WEEK\n    });\n\n    // Auth Lambda for user authentication\n    const authLambda = new lambda.Function(this, 'AuthFunction', {\n      runtime: lambda.Runtime.NODEJS_18_X,\n      code: lambda.Code.fromAsset('../dist'),\n      handler: 'lambda/auth.handler',\n      layers: [commonLayer],\n      environment: {\n        USER_POOL_ID: userPool.userPoolId,\n        USER_POOL_CLIENT_ID: userPoolClient.userPoolClientId,\n        USERS_TABLE: usersTable.tableName\n      },\n      timeout: cdk.Duration.seconds(30),\n      memorySize: 512,\n      logRetention: logs.RetentionDays.ONE_WEEK\n    });\n\n    // User Management Lambda\n    const userLambda = new lambda.Function(this, 'UserFunction', {\n      runtime: lambda.Runtime.NODEJS_18_X,\n      code: lambda.Code.fromAsset('../dist'),\n      handler: 'lambda/user.handler',\n      layers: [commonLayer],\n      environment: {\n        USER_POOL_ID: userPool.userPoolId,\n        USER_POOL_CLIENT_ID: userPoolClient.userPoolClientId,\n        USERS_TABLE: usersTable.tableName,\n        USER_PROMPTS_BUCKET: userPromptsBucket.bucketName\n      },\n      timeout: cdk.Duration.seconds(30),\n      memorySize: 512,\n      logRetention: logs.RetentionDays.ONE_WEEK\n    });\n\n    // Stripe Webhook Lambda\n    const stripeWebhookLambda = new lambda.Function(this, 'StripeWebhookFunction', {\n      runtime: lambda.Runtime.NODEJS_18_X,\n      code: lambda.Code.fromAsset('../dist'),\n      handler: 'lambda/stripe-webhook.handler',\n      layers: [commonLayer],\n      environment: {\n        USERS_TABLE: usersTable.tableName,\n        STRIPE_WEBHOOK_SECRET: process.env.STRIPE_WEBHOOK_SECRET || ''\n      },\n      timeout: cdk.Duration.seconds(30),\n      memorySize: 512,\n      logRetention: logs.RetentionDays.ONE_WEEK\n    });\n\n    // Cognito Authorizer\n    const cognitoAuthorizer = new apigateway.CognitoUserPoolsAuthorizer(this, 'CognitoAuthorizer', {\n      cognitoUserPools: [userPool]\n    });\n\n    // API Gateway\n    const api = new apigateway.RestApi(this, 'McpPromptsApi', {\n      restApiName: 'MCP Prompts Service',\n      description: 'API for MCP Prompts with AWS backend',\n      defaultCorsPreflightOptions: {\n        allowOrigins: apigateway.Cors.ALL_ORIGINS,\n        allowMethods: apigateway.Cors.ALL_METHODS,\n        allowHeaders: ['Content-Type', 'X-Amz-Date', 'Authorization', 'X-Api-Key', 'X-Amz-Security-Token']\n      },\n      endpointConfiguration: {\n        types: [apigateway.EndpointType.REGIONAL]\n      }\n    });\n\n    // API Key for rate limiting\n    const apiKey = new apigateway.ApiKey(this, 'McpPromptsApiKey', {\n      apiKeyName: 'mcp-prompts-api-key',\n      description: 'API Key for MCP Prompts'\n    });\n\n    // Usage plan for rate limiting\n    const usagePlan = new apigateway.UsagePlan(this, 'McpPromptsUsagePlan', {\n      name: 'MCP Prompts Usage Plan',\n      description: 'Usage plan for MCP Prompts API',\n      apiStages: [{\n        api,\n        stage: api.deploymentStage\n      }],\n      throttle: {\n        rateLimit: 100, // requests per second\n        burstLimit: 200\n      },\n      quota: {\n        limit: 10000, // requests per month for free tier\n        period: apigateway.Period.MONTH\n      }\n    });\n\n    usagePlan.addApiKey(apiKey);\n\n    // API Gateway integration\n    const mcpIntegration = new apigateway.LambdaIntegration(mcpServerLambda, {\n      requestTemplates: { 'application/json': '{ \"statusCode\": \"200\" }' }\n    });\n\n    // API routes\n    const v1 = api.root.addResource('v1');\n    const prompts = v1.addResource('prompts');\n\n    prompts.addMethod('GET', mcpIntegration); // List prompts\n    prompts.addMethod('POST', mcpIntegration); // Create prompt\n\n    const promptById = prompts.addResource('{id}');\n    promptById.addMethod('GET', mcpIntegration); // Get prompt\n    promptById.addMethod('PUT', mcpIntegration); // Update prompt\n    promptById.addMethod('DELETE', mcpIntegration); // Delete prompt\n\n    const promptApply = promptById.addResource('apply');\n    promptApply.addMethod('POST', mcpIntegration); // Apply template variables\n\n    // Health endpoint\n    const health = api.root.addResource('health');\n    health.addMethod('GET', mcpIntegration);\n\n    // Auth endpoints\n    const auth = api.root.addResource('auth');\n    const authIntegration = new apigateway.LambdaIntegration(authLambda);\n\n    auth.addMethod('POST', authIntegration); // Login/Register\n\n    const authRefresh = auth.addResource('refresh');\n    authRefresh.addMethod('POST', authIntegration); // Refresh token\n\n    // User endpoints (protected)\n    const users = api.root.addResource('users');\n    const userIntegration = new apigateway.LambdaIntegration(userLambda);\n\n    const userById = users.addResource('{userId}');\n    userById.addMethod('GET', userIntegration, {\n      authorizer: cognitoAuthorizer,\n      authorizationType: apigateway.AuthorizationType.COGNITO\n    }); // Get user profile\n\n    userById.addMethod('PUT', userIntegration, {\n      authorizer: cognitoAuthorizer,\n      authorizationType: apigateway.AuthorizationType.COGNITO\n    }); // Update user profile\n\n    // User prompts endpoints (protected)\n    const userPrompts = userById.addResource('prompts');\n    userPrompts.addMethod('GET', userIntegration, {\n      authorizer: cognitoAuthorizer,\n      authorizationType: apigateway.AuthorizationType.COGNITO\n    }); // List user's prompts\n\n    userPrompts.addMethod('POST', userIntegration, {\n      authorizer: cognitoAuthorizer,\n      authorizationType: apigateway.AuthorizationType.COGNITO\n    }); // Upload prompt\n\n    const userPromptById = userPrompts.addResource('{promptId}');\n    userPromptById.addMethod('GET', userIntegration, {\n      authorizer: cognitoAuthorizer,\n      authorizationType: apigateway.AuthorizationType.COGNITO\n    }); // Get user's prompt\n\n    userPromptById.addMethod('PUT', userIntegration, {\n      authorizer: cognitoAuthorizer,\n      authorizationType: apigateway.AuthorizationType.COGNITO\n    }); // Update user's prompt\n\n    userPromptById.addMethod('DELETE', userIntegration, {\n      authorizer: cognitoAuthorizer,\n      authorizationType: apigateway.AuthorizationType.COGNITO\n    }); // Delete user's prompt\n\n    // Stripe webhook endpoint (no auth required)\n    const webhook = api.root.addResource('webhook');\n    const stripeWebhook = webhook.addResource('stripe');\n    const webhookIntegration = new apigateway.LambdaIntegration(stripeWebhookLambda);\n\n    stripeWebhook.addMethod('POST', webhookIntegration); // Stripe webhook\n\n    // MCP capabilities endpoint\n    const mcp = api.root.addResource('mcp');\n    mcp.addMethod('GET', mcpIntegration); // MCP capabilities\n    const tools = mcp.addResource('tools');\n    tools.addMethod('GET', mcpIntegration); // List tools\n    tools.addMethod('POST', mcpIntegration); // Execute tool\n\n    // S3 Bucket for web assets\n    const webBucket = new s3.Bucket(this, 'WebBucket', {\n      bucketName: `mcp-prompts-web-${this.account}-${this.region}`,\n      websiteIndexDocument: 'index.html',\n      publicReadAccess: true,\n      blockPublicAccess: {\n        blockPublicAcls: false,\n        blockPublicPolicy: false,\n        ignorePublicAcls: false,\n        restrictPublicBuckets: false\n      },\n      cors: [{\n        allowedOrigins: ['*'],\n        allowedMethods: [s3.HttpMethods.GET],\n        allowedHeaders: ['*']\n      }]\n    });\n\n    // CloudFront Distribution\n    const distribution = new cloudfront.Distribution(this, 'McpPromptsDistribution', {\n      defaultBehavior: {\n        origin: new origins.S3Origin(webBucket),\n        viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        cachePolicy: cloudfront.CachePolicy.CACHING_OPTIMIZED\n      },\n      additionalBehaviors: {\n        '/v1/*': {\n          origin: new origins.RestApiOrigin(api),\n          viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n          cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED,\n          allowedMethods: cloudfront.AllowedMethods.ALLOW_ALL\n        },\n        '/auth': {\n          origin: new origins.RestApiOrigin(api),\n          viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n          cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED,\n          allowedMethods: cloudfront.AllowedMethods.ALLOW_ALL\n        },\n        '/users/*': {\n          origin: new origins.RestApiOrigin(api),\n          viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n          cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED,\n          allowedMethods: cloudfront.AllowedMethods.ALLOW_ALL\n        },\n        '/mcp/*': {\n          origin: new origins.RestApiOrigin(api),\n          viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n          cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED,\n          allowedMethods: cloudfront.AllowedMethods.ALLOW_ALL\n        },\n        '/health': {\n          origin: new origins.RestApiOrigin(api),\n          viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n          cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED,\n          allowedMethods: cloudfront.AllowedMethods.ALLOW_GET_HEAD\n        },\n        '/webhook/*': {\n          origin: new origins.RestApiOrigin(api),\n          viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n          cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED,\n          allowedMethods: cloudfront.AllowedMethods.ALLOW_ALL\n        },\n        '/static/*': {\n          origin: new origins.S3Origin(promptsBucket),\n          viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n          cachePolicy: cloudfront.CachePolicy.CACHING_OPTIMIZED\n        }\n      }\n    });\n\n    // Event source mappings\n    processingLambda.addEventSource(new SqsEventSource(processingQueue, {\n      batchSize: 10\n    }));\n\n    catalogSyncLambda.addEventSource(new SqsEventSource(catalogSyncQueue, {\n      batchSize: 1\n    }));\n\n    // Permissions\n    promptsTable.grantReadWriteData(mcpServerLambda);\n    promptsTable.grantReadWriteData(processingLambda);\n    promptsTable.grantReadWriteData(catalogSyncLambda);\n    promptsTable.grantReadWriteData(authLambda);\n    promptsTable.grantReadWriteData(userLambda);\n\n    sessionsTable.grantReadWriteData(mcpServerLambda);\n\n    usersTable.grantReadWriteData(authLambda);\n    usersTable.grantReadWriteData(userLambda);\n    usersTable.grantReadWriteData(stripeWebhookLambda);\n\n    promptsBucket.grantReadWrite(mcpServerLambda);\n    promptsBucket.grantReadWrite(processingLambda);\n    promptsBucket.grantReadWrite(catalogSyncLambda);\n\n    userPromptsBucket.grantReadWrite(authLambda);\n    userPromptsBucket.grantReadWrite(userLambda);\n\n    processingQueue.grantSendMessages(mcpServerLambda);\n    catalogSyncQueue.grantSendMessages(mcpServerLambda);\n\n    processingQueue.grantConsumeMessages(processingLambda);\n    catalogSyncQueue.grantConsumeMessages(catalogSyncLambda);\n\n    // CloudWatch permissions\n    const cloudwatchPolicy = new iam.PolicyStatement({\n      actions: [\n        'cloudwatch:PutMetricData',\n        'logs:CreateLogGroup',\n        'logs:CreateLogStream',\n        'logs:PutLogEvents'\n      ],\n      resources: ['*']\n    });\n\n    mcpServerLambda.addToRolePolicy(cloudwatchPolicy);\n    processingLambda.addToRolePolicy(cloudwatchPolicy);\n    catalogSyncLambda.addToRolePolicy(cloudwatchPolicy);\n    authLambda.addToRolePolicy(cloudwatchPolicy);\n    userLambda.addToRolePolicy(cloudwatchPolicy);\n\n    // Cognito permissions for auth lambda\n    const cognitoPolicy = new iam.PolicyStatement({\n      actions: [\n        'cognito-idp:AdminCreateUser',\n        'cognito-idp:AdminSetUserPassword',\n        'cognito-idp:AdminInitiateAuth',\n        'cognito-idp:AdminRespondToAuthChallenge',\n        'cognito-idp:AdminGetUser',\n        'cognito-idp:ListUsers',\n        'cognito-idp:AdminUpdateUserAttributes'\n      ],\n      resources: [userPool.userPoolArn]\n    });\n\n    authLambda.addToRolePolicy(cognitoPolicy);\n    userLambda.addToRolePolicy(cognitoPolicy);\n\n    // Outputs\n    new cdk.CfnOutput(this, 'ApiGatewayUrl', {\n      value: api.url,\n      description: 'API Gateway URL'\n    });\n\n    new cdk.CfnOutput(this, 'CloudFrontUrl', {\n      value: `https://${distribution.distributionDomainName}`,\n      description: 'CloudFront Distribution URL (Web App)'\n    });\n\n    new cdk.CfnOutput(this, 'WebBucketName', {\n      value: webBucket.bucketName,\n      description: 'S3 Bucket for web assets'\n    });\n\n    new cdk.CfnOutput(this, 'S3BucketName', {\n      value: promptsBucket.bucketName,\n      description: 'S3 Bucket for prompts catalog'\n    });\n\n    new cdk.CfnOutput(this, 'UserPromptsBucketName', {\n      value: userPromptsBucket.bucketName,\n      description: 'S3 Bucket for user-uploaded prompts'\n    });\n\n    new cdk.CfnOutput(this, 'DynamoDBTableName', {\n      value: promptsTable.tableName,\n      description: 'DynamoDB table for prompts'\n    });\n\n    new cdk.CfnOutput(this, 'UsersTableName', {\n      value: usersTable.tableName,\n      description: 'DynamoDB table for users'\n    });\n\n    new cdk.CfnOutput(this, 'CognitoUserPoolId', {\n      value: userPool.userPoolId,\n      description: 'Cognito User Pool ID'\n    });\n\n    new cdk.CfnOutput(this, 'CognitoUserPoolClientId', {\n      value: userPoolClient.userPoolClientId,\n      description: 'Cognito User Pool Client ID'\n    });\n\n    new cdk.CfnOutput(this, 'ApiKey', {\n      value: apiKey.keyId,\n      description: 'API Gateway Key ID'\n    });\n  }\n}\n"]}

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/sparesparrow/mcp-prompts'

If you have feedback or need assistance with the MCP directory API, please join our Discord server