@cxbuilder/flow-config 2.2.2 → 2.2.3

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/.jsii CHANGED
@@ -9103,6 +9103,6 @@
9103
9103
  "symbolId": "infrastructure/FlowConfigStack:LambdaVpcConfig"
9104
9104
  }
9105
9105
  },
9106
- "version": "2.2.2",
9107
- "fingerprint": "HpNVMeRnWssTBe9zdfxT6CPcpkaU2v+sjc/rVnoeD5Q="
9106
+ "version": "2.2.3",
9107
+ "fingerprint": "M3UYVJLKiRmAA1Gd5XkHYVDZXrICXtoyUrx+xL8SAxo="
9108
9108
  }
package/CHANGELOG.md CHANGED
@@ -5,6 +5,11 @@ All notable changes to this project will be documented in this file.
5
5
  The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/),
6
6
  and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
7
7
 
8
+ ## 2.2.3 (2026-04-02)
9
+
10
+ - jeff.modzel
11
+ - Bugfix: CloudWatch LogGroup creation for Lambda Functions
12
+
8
13
  ## 2.2.2 (2026-01-05)
9
14
 
10
15
  - sfred
@@ -235,5 +235,5 @@ class FlowConfigStack extends cdk.Stack {
235
235
  }
236
236
  exports.FlowConfigStack = FlowConfigStack;
237
237
  _a = JSII_RTTI_SYMBOL_1;
238
- FlowConfigStack[_a] = { fqn: "@cxbuilder/flow-config.FlowConfigStack", version: "2.2.2" };
238
+ FlowConfigStack[_a] = { fqn: "@cxbuilder/flow-config.FlowConfigStack", version: "2.2.3" };
239
239
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FlowConfigStack.js","sourceRoot":"","sources":["../../infrastructure/FlowConfigStack.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  IUserPool,\n  UserPool,\n  UserPoolClient,\n  UserPoolClientIdentityProvider,\n  CfnUserPoolGroup,\n} from 'aws-cdk-lib/aws-cognito';\nimport * as dynamodb from 'aws-cdk-lib/aws-dynamodb';\nimport { IVpc, ISecurityGroup, ISubnet } from 'aws-cdk-lib/aws-ec2';\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\nimport { Construct } from 'constructs';\nimport { Api } from './api/Api';\nimport { Topic } from 'aws-cdk-lib/aws-sns';\nimport { EmailSubscription } from 'aws-cdk-lib/aws-sns-subscriptions';\nimport { SecurityProfileProvider } from './SecurityProfileProvider';\nimport { CfnIntegrationAssociation } from 'aws-cdk-lib/aws-connect';\nimport { CfnApplication } from 'aws-cdk-lib/aws-appintegrations';\nimport { GetConfig } from './GetConfig';\n\n/**\n * Cognito configuration for FlowConfig stack\n */\nexport interface CognitoConfig {\n  readonly userPoolId: string;\n\n  /**\n   * Full domain name\n   */\n  readonly domain: string;\n\n  /**\n   * If provided, client will auth to SSO. Otherwise will auth to user pool\n   */\n  readonly ssoProviderName?: string;\n}\n\n/**\n * VPC configuration for API Gateway\n * If provided, the API will be deployed in a private VPC.\n */\nexport interface ApiVpcConfig {\n  /**\n   * The VPC ID to use for the API\n   */\n  readonly vpcId: string;\n  /**\n   * The VPC endpoint ID to use for the API\n   */\n  readonly vpcEndpointId: string;\n}\n\n/**\n * Lambda VPC configuration\n */\nexport interface LambdaVpcConfig {\n  /**\n   * The VPC ID to deploy resources into\n   */\n  readonly vpcId: string;\n\n  /**\n   * Security group IDs for Lambda functions\n   */\n  readonly securityGroupIds: string[];\n\n  /**\n   * Private subnet IDs for Lambda functions\n   */\n  readonly subnetIds: string[];\n}\n\n/**\n * Global table configuration for multi-region deployments\n */\nexport interface GlobalTableConfig {\n  /**\n   * Whether this is the primary region that creates the global table\n   */\n  readonly isPrimaryRegion: boolean;\n\n  /**\n   * List of all regions that should have replicas\n   * Only used by the primary region\n   */\n  readonly replicaRegions?: string[];\n}\n\n/**\n * Internal interface for resolved VPC resources\n */\nexport interface ResolvedVpcConfig {\n  readonly vpc: IVpc;\n  readonly lambdaSecurityGroups: ISecurityGroup[];\n  readonly privateSubnets: ISubnet[];\n}\n\nexport interface FlowConfigStackProps extends cdk.StackProps {\n  /**\n   * Used for resource naming. Will also be the name of the Connect Lambda\n   * @example `cxbuilder-flow-config`\n   */\n  readonly prefix: string;\n  readonly cognito: CognitoConfig;\n  readonly connectInstanceArn: string;\n\n  /**\n   * Who to notify for unhandled exceptions\n   */\n  readonly alertEmails: string[];\n  readonly prod?: boolean;\n\n  /**\n   * If provided, the API will be deployed in a VPC.\n   */\n  readonly apiVpcConfig?: ApiVpcConfig;\n\n  /**\n   * If provided, the Lambda functions will be deployed in a VPC.\n   * Note: VPC should contain endpoints to: CloudFormation, Lambda, DynamoDB, SNS, and Polly.\n   */\n  readonly lambdaVpcConfig?: LambdaVpcConfig;\n\n  /**\n   * Global table configuration for multi-region deployments.\n   * If provided, enables global table support.\n   * If undefined, creates a single-region table.\n   */\n  readonly globalTable?: GlobalTableConfig;\n\n  /**\n   * Set to false to remove CXBuilder branding from the web app.\n   * @default true\n   */\n  readonly branding?: boolean;\n\n  /**\n   * Whether to associate the app with the Connect Agent Workspace.\n   * Set to false to disable automatic association.\n   * @default true\n   */\n  readonly associate3pApp?: boolean;\n}\n\nexport class FlowConfigStack extends cdk.Stack {\n  userPool: IUserPool;\n  private api: Api;\n  userPoolClient: UserPoolClient;\n  alertTopic: Topic;\n  table: cdk.aws_dynamodb.ITable;\n\n  /**\n   * Resolved VPC configuration if private deployment is enabled\n   */\n  private readonly _resolvedVpcConfig?: ResolvedVpcConfig;\n\n  /**\n   * Get resolved VPC configuration for child constructs\n   * @internal\n   */\n  public _getResolvedVpcConfig(): ResolvedVpcConfig | undefined {\n    return this._resolvedVpcConfig;\n  }\n\n  get appUrl(): string {\n    return this.api.url;\n  }\n\n  constructor(\n    scope: Construct,\n    id: string,\n    public props: FlowConfigStackProps\n  ) {\n    const {\n      prefix,\n      cognito,\n      alertEmails,\n      prod = false,\n      lambdaVpcConfig,\n      globalTable,\n    } = props;\n    super(scope, id, {\n      ...props,\n      stackName: props.stackName ?? prefix,\n      description:\n        props.description ??\n        'Web-based interface for managing flow variables and prompts that are dynamically retrieved by Amazon Connect during customer interactions',\n      terminationProtection: prod,\n    });\n\n    // Resolve VPC configuration if provided\n    this._resolvedVpcConfig = lambdaVpcConfig\n      ? this.resolveVpcConfig(lambdaVpcConfig)\n      : undefined;\n\n    // DynamoDB table for storing flow configs\n    this.table = this.createTable(prefix, prod, globalTable);\n\n    this.alertTopic = new Topic(this, 'AlertTopic', {\n      topicName: `${prefix}-error-alerts`,\n    });\n    alertEmails.forEach((e) =>\n      this.alertTopic.addSubscription(new EmailSubscription(e))\n    );\n\n    this.userPool = UserPool.fromUserPoolId(\n      this,\n      'UserPool',\n      cognito.userPoolId\n    );\n\n    const getConfig = new GetConfig(this);\n    this.api = new Api(this);\n\n    this.userPoolClient = this.createUserPoolClient();\n    this.createUserPoolGroups();\n\n    this.associate3pApp();\n\n    new cdk.CfnOutput(this, 'AppUrl', {\n      value: this.appUrl,\n      description: 'Base URL for the Flow Config application',\n    });\n    new cdk.CfnOutput(this, 'ApiUrl', {\n      value: `${this.appUrl}/api`,\n      description: 'Base URL for the Flow Config API',\n    });\n    new cdk.CfnOutput(this, 'GetConfigLambdaName', {\n      value: getConfig.function.functionName,\n      description:\n        'Lambda function name for accessing flow configs from Connect',\n    });\n  }\n\n  createUserPoolClient(): UserPoolClient {\n    const { prefix, cognito } = this.props;\n\n    const domains = ['http://localhost:3000', this.appUrl];\n\n    const oAuth = {\n      callbackUrls: domains.map((x) => `${x}/popup.html`),\n      logoutUrls: domains,\n    };\n    const client = new UserPoolClient(this, 'UserPoolClient', {\n      userPoolClientName: prefix,\n      userPool: this.userPool,\n      supportedIdentityProviders: cognito.ssoProviderName\n        ? [UserPoolClientIdentityProvider.custom(cognito.ssoProviderName)]\n        : [UserPoolClientIdentityProvider.COGNITO],\n      generateSecret: false,\n      oAuth,\n    });\n\n    new cdk.CfnOutput(this, 'UserPoolClientId', {\n      value: client.userPoolClientId,\n    });\n    return client;\n  }\n\n  /**\n   * Create Cognito User Groups for role-based access control\n   */\n  createUserPoolGroups(): void {\n    // FlowConfigAdmin - Full CRUD access\n    new CfnUserPoolGroup(this, 'FlowConfigAdminGroup', {\n      userPoolId: this.userPool.userPoolId,\n      groupName: 'FlowConfigAdmin',\n      description:\n        'Full administrative access to all flow configs - can create, read, update, and delete flow configs and all their properties',\n      precedence: 1,\n    });\n\n    // FlowConfigEdit - Edit variable/prompt values only\n    new CfnUserPoolGroup(this, 'FlowConfigEditGroup', {\n      userPoolId: this.userPool.userPoolId,\n      groupName: 'FlowConfigEdit',\n      description:\n        'Edit access to flow configs - can read and modify variable values and prompt content but cannot add/remove fields or delete configs',\n      precedence: 2,\n    });\n\n    // FlowConfigRead - Read-only access\n    new CfnUserPoolGroup(this, 'FlowConfigReadGroup', {\n      userPoolId: this.userPool.userPoolId,\n      groupName: 'FlowConfigRead',\n      description:\n        'Read-only access to all flow configs for reporting and reference purposes',\n      precedence: 3,\n    });\n  }\n\n  /**\n   * Associate FlowConfig as Agent Workspace app\n   */\n  associate3pApp() {\n    const {\n      props: { prefix, connectInstanceArn, associate3pApp = true },\n      appUrl: url,\n    } = this;\n\n    if (!associate3pApp) {\n      return;\n    }\n\n    let namespace = `cxbuilder.${prefix}`;\n    if (namespace.length > 32 && !cdk.Token.isUnresolved(namespace)) {\n      namespace = namespace.substring(0, 32);\n    }\n\n    const app = new CfnApplication(this, '3pApp', {\n      name: this.props.prefix,\n      description:\n        'Agent Workspace app for configuring contact flow variables and prompts',\n      namespace,\n      applicationSourceConfig: {\n        externalUrlConfig: {\n          accessUrl: url,\n        },\n      },\n      // docs: https://docs.aws.amazon.com/connect/latest/adminguide/3p-apps-events-requests.html\n      permissions: [\n        'User.Details.View',\n        'User.Configuration.View',\n        'User.Status.View',\n        'Contact.Details.View',\n        'Contact.CustomerDetails.View',\n        'Contact.Variables.View',\n      ],\n    });\n\n    const association = new CfnIntegrationAssociation(\n      this,\n      '3pAppAssociation',\n      {\n        instanceId: connectInstanceArn,\n        integrationType: 'APPLICATION',\n        integrationArn: app.attrApplicationArn,\n      }\n    );\n\n    const provider = new SecurityProfileProvider(\n      this,\n      'SecurityProfileProvider',\n      {\n        connectInstanceArn,\n      }\n    );\n    provider.node.addDependency(association);\n\n    // Automatically associate with security profiles\n    const securityProfiles = ['Admin'];\n    for (const profile of securityProfiles) {\n      provider.allowApplication(`Allow${profile}`, profile, namespace);\n    }\n  }\n\n  /**\n   * Create DynamoDB table with optional global table support\n   */\n  private createTable(\n    prefix: string,\n    prod: boolean,\n    globalTable?: GlobalTableConfig\n  ): dynamodb.ITable {\n    const tableName = prefix;\n\n    if (!globalTable) {\n      // Single-region table\n      return new dynamodb.TableV2(this, 'Table', {\n        tableName,\n        partitionKey: {\n          name: 'id',\n          type: dynamodb.AttributeType.STRING,\n        },\n        billing: dynamodb.Billing.onDemand(),\n        pointInTimeRecoverySpecification: {\n          pointInTimeRecoveryEnabled: true,\n        },\n        deletionProtection: prod,\n        removalPolicy: prod\n          ? cdk.RemovalPolicy.RETAIN\n          : cdk.RemovalPolicy.DESTROY,\n      });\n    }\n\n    if (globalTable.isPrimaryRegion) {\n      // Primary region creates global table with replicas\n      const table = new dynamodb.TableV2(this, 'Table', {\n        tableName,\n        partitionKey: {\n          name: 'id',\n          type: dynamodb.AttributeType.STRING,\n        },\n        billing: dynamodb.Billing.onDemand(),\n        pointInTimeRecoverySpecification: {\n          pointInTimeRecoveryEnabled: true,\n        },\n        deletionProtection: prod,\n        removalPolicy: prod\n          ? cdk.RemovalPolicy.RETAIN\n          : cdk.RemovalPolicy.DESTROY,\n      });\n      if (globalTable.replicaRegions) {\n        for (const region of globalTable.replicaRegions) {\n          table.addReplica({\n            region,\n            deletionProtection: prod,\n            pointInTimeRecoverySpecification: {\n              pointInTimeRecoveryEnabled: true,\n            },\n          });\n        }\n      }\n      return table;\n    } else {\n      // Secondary region references existing global table\n      return dynamodb.Table.fromTableArn(\n        this,\n        'Table',\n        `arn:aws:dynamodb:${this.region}:${this.account}:table/${tableName}`\n      );\n    }\n  }\n\n  /**\n   * Resolve VPC configuration string IDs to CDK objects\n   */\n  private resolveVpcConfig(vpcConfig: LambdaVpcConfig): ResolvedVpcConfig {\n    const { vpcId, securityGroupIds, subnetIds } = vpcConfig;\n\n    // Resolve VPC\n    const vpc = ec2.Vpc.fromLookup(this, 'ResolvedVpc', { vpcId });\n\n    // Resolve Lambda security groups\n    const lambdaSecurityGroups = securityGroupIds.map((sgId, index) =>\n      ec2.SecurityGroup.fromSecurityGroupId(\n        this,\n        `LambdaSecurityGroup${index}`,\n        sgId\n      )\n    );\n\n    // Resolve private subnets\n    const privateSubnets = subnetIds.map((subnetId, index) =>\n      ec2.Subnet.fromSubnetId(this, `PrivateSubnet${index}`, subnetId)\n    );\n\n    return {\n      vpc,\n      lambdaSecurityGroups,\n      privateSubnets,\n    };\n  }\n}\n"]}
@@ -36,6 +36,14 @@ function createLambda(scope, id, props) {
36
36
  // Check if we're in a FlowConfigStack and get VPC configuration
37
37
  const stack = aws_cdk_lib_1.Stack.of(scope);
38
38
  const vpcConfig = stack._getResolvedVpcConfig();
39
+ // Create the log group BEFORE the function so CloudFormation establishes the dependency
40
+ // correctly. If the log group is created after using func.functionName, Lambda may
41
+ // auto-create the log group on first invocation before CloudFormation gets to it,
42
+ // causing a "already exists" failure during stack creation.
43
+ const logGroup = new aws_logs_1.LogGroup(scope, `${id}LogGroup`, {
44
+ retention: props.logRetention ?? aws_logs_1.RetentionDays.ONE_MONTH,
45
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
46
+ });
39
47
  const func = new aws_lambda_1.Function(scope, `${id}Function`, {
40
48
  // Apply defaults
41
49
  handler: 'index.handler',
@@ -57,6 +65,8 @@ function createLambda(scope, id, props) {
57
65
  ...props,
58
66
  // Ensure that the log retention custom resource is not created
59
67
  logRetention: undefined,
68
+ // Use our managed log group (must come after spread to override any logGroup in props)
69
+ logGroup,
60
70
  environment: {
61
71
  // see https://docs.aws.amazon.com/lambda/latest/dg/typescript-exceptions.html
62
72
  NODE_OPTIONS: '--enable-source-maps',
@@ -69,17 +79,9 @@ function createLambda(scope, id, props) {
69
79
  },
70
80
  });
71
81
  props.alertTopic?.grantPublish(func);
72
- // Note: we are using this policy instead of log.grantWrite(func) because the func.functionName a circular dependency.
73
- // To constrain LogGroup permissions further, you can use a static functionName with log.grantWrite
74
82
  func.role?.addManagedPolicy(aws_iam_1.ManagedPolicy.fromAwsManagedPolicyName(vpcConfig
75
83
  ? 'service-role/AWSLambdaVPCAccessExecutionRole'
76
84
  : 'service-role/AWSLambdaBasicExecutionRole'));
77
- // Create a self-managed log group
78
- new aws_logs_1.LogGroup(scope, `${id}LogGroup`, {
79
- retention: props.logRetention ?? aws_logs_1.RetentionDays.ONE_MONTH,
80
- logGroupName: `/aws/lambda/${func.functionName}`,
81
- removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
82
- });
83
85
  return func;
84
86
  }
85
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createLambda.js","sourceRoot":"","sources":["../../infrastructure/createLambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, RemovalPolicy, Stack } from 'aws-cdk-lib';\nimport { ManagedPolicy } from 'aws-cdk-lib/aws-iam';\nimport {\n  Architecture,\n  Runtime,\n  Tracing,\n  Function,\n  FunctionProps,\n  Code,\n  LayerVersion,\n} from 'aws-cdk-lib/aws-lambda';\nimport { LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { ITopic } from 'aws-cdk-lib/aws-sns';\nimport { Construct } from 'constructs';\nimport { existsSync } from 'fs';\nimport { resolve } from 'path';\nimport { FlowConfigStack } from './FlowConfigStack';\n\ninterface CreateLambdaProps<TEnv>\n  extends Omit<FunctionProps, 'handler' | 'environment' | 'code' | 'runtime'> {\n  environment?: TEnv;\n\n  /**\n   * Notify someone when your lambda fails!\n   */\n  alertTopic?: ITopic;\n}\n/**\n * Utility function for creating a log group and lambda with app defaults.\n * Enforces environment variables via generic parameter TEnv\n */\nexport function createLambda<TEnv>(\n  scope: Construct,\n  id: string,\n  props: CreateLambdaProps<TEnv>\n) {\n  // Determine the code location based on logical ID\n  // If id is \"Handler\", use the parent construct's ID instead\n  const backendId = id === 'Handler' ? scope.node.id : id;\n  let codeLocation = resolve(__dirname, '..', 'backend', backendId);\n  let expectedIndexFile = resolve(codeLocation, 'index.js');\n\n  // Verify the bundled Lambda code exists, try ../dist/backend if not found\n  if (!existsSync(expectedIndexFile)) {\n    // Try alternative location in ../dist/backend\n    codeLocation = resolve(__dirname, '..', 'dist', 'backend', backendId);\n    expectedIndexFile = resolve(codeLocation, 'index.js');\n\n    if (!existsSync(expectedIndexFile)) {\n      throw new Error(\n        `Lambda function ${backendId} bundled code not found at: ${expectedIndexFile} or ${resolve(\n          __dirname,\n          '..',\n          'backend',\n          backendId,\n          'index.js'\n        )}. ` +\n          'Please run \"npm run build:lambdas\" to bundle the Lambda functions.'\n      );\n    }\n  }\n\n  console.log(\n    `📦 Using Lambda code for ${backendId} (id=${id}) from: ${codeLocation}`\n  );\n\n  // Get the current region for the Lambda Powertools layer\n  const region = Stack.of(scope).region;\n\n  // AWS Lambda Powertools layer ARN\n  // See: https://docs.powertools.aws.dev/lambda/typescript/latest/#lambda-layer\n  const powertoolsLayerArn = `arn:aws:lambda:${region}:094274105915:layer:AWSLambdaPowertoolsTypeScriptV2:22`;\n\n  // Check if we're in a FlowConfigStack and get VPC configuration\n  const stack = Stack.of(scope) as FlowConfigStack;\n  const vpcConfig = stack._getResolvedVpcConfig();\n\n  const func = new Function(scope, `${id}Function`, {\n    // Apply defaults\n    handler: 'index.handler',\n    runtime: Runtime.NODEJS_22_X,\n    architecture: Architecture.ARM_64,\n    tracing: Tracing.ACTIVE,\n    timeout: Duration.seconds(30),\n    code: Code.fromAsset(codeLocation),\n    layers: [\n      LayerVersion.fromLayerVersionArn(\n        scope,\n        `${id}PowertoolsLayer`,\n        powertoolsLayerArn\n      ),\n    ],\n    // VPC configuration if provided\n    ...(vpcConfig && {\n      vpc: vpcConfig.vpc,\n      vpcSubnets: { subnets: vpcConfig.privateSubnets },\n      securityGroups: vpcConfig.lambdaSecurityGroups,\n    }),\n    // Apply overrides\n    ...props,\n    // Ensure that the log retention custom resource is not created\n    logRetention: undefined,\n    environment: {\n      // see https://docs.aws.amazon.com/lambda/latest/dg/typescript-exceptions.html\n      NODE_OPTIONS: '--enable-source-maps',\n      // Log level. Do not emit DEBUG in prod environment because it will bloat your CW costs\n      POWERTOOLS_LOG_LEVEL: 'DEBUG',\n      // Log incoming event\n      POWERTOOLS_LOGGER_LOG_EVENT: 'true',\n      ALERT_TOPIC_ARN: props.alertTopic?.topicArn ?? 'N/A',\n      ...props.environment,\n    },\n  });\n\n  props.alertTopic?.grantPublish(func);\n\n  // Note: we are using this policy instead of log.grantWrite(func) because the func.functionName a circular dependency.\n  // To constrain LogGroup permissions further, you can use a static functionName with log.grantWrite\n  func.role?.addManagedPolicy(\n    ManagedPolicy.fromAwsManagedPolicyName(\n      vpcConfig\n        ? 'service-role/AWSLambdaVPCAccessExecutionRole'\n        : 'service-role/AWSLambdaBasicExecutionRole'\n    )\n  );\n\n  // Create a self-managed log group\n  new LogGroup(scope, `${id}LogGroup`, {\n    retention: props.logRetention ?? RetentionDays.ONE_MONTH,\n    logGroupName: `/aws/lambda/${func.functionName}`,\n    removalPolicy: RemovalPolicy.DESTROY,\n  });\n  return func;\n}\n"]}
87
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createLambda.js","sourceRoot":"","sources":["../../infrastructure/createLambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, RemovalPolicy, Stack } from 'aws-cdk-lib';\nimport { ManagedPolicy } from 'aws-cdk-lib/aws-iam';\nimport {\n  Architecture,\n  Runtime,\n  Tracing,\n  Function,\n  FunctionProps,\n  Code,\n  LayerVersion,\n} from 'aws-cdk-lib/aws-lambda';\nimport { LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { ITopic } from 'aws-cdk-lib/aws-sns';\nimport { Construct } from 'constructs';\nimport { existsSync } from 'fs';\nimport { resolve } from 'path';\nimport { FlowConfigStack } from './FlowConfigStack';\n\ninterface CreateLambdaProps<TEnv>\n  extends Omit<FunctionProps, 'handler' | 'environment' | 'code' | 'runtime'> {\n  environment?: TEnv;\n\n  /**\n   * Notify someone when your lambda fails!\n   */\n  alertTopic?: ITopic;\n}\n/**\n * Utility function for creating a log group and lambda with app defaults.\n * Enforces environment variables via generic parameter TEnv\n */\nexport function createLambda<TEnv>(\n  scope: Construct,\n  id: string,\n  props: CreateLambdaProps<TEnv>\n) {\n  // Determine the code location based on logical ID\n  // If id is \"Handler\", use the parent construct's ID instead\n  const backendId = id === 'Handler' ? scope.node.id : id;\n  let codeLocation = resolve(__dirname, '..', 'backend', backendId);\n  let expectedIndexFile = resolve(codeLocation, 'index.js');\n\n  // Verify the bundled Lambda code exists, try ../dist/backend if not found\n  if (!existsSync(expectedIndexFile)) {\n    // Try alternative location in ../dist/backend\n    codeLocation = resolve(__dirname, '..', 'dist', 'backend', backendId);\n    expectedIndexFile = resolve(codeLocation, 'index.js');\n\n    if (!existsSync(expectedIndexFile)) {\n      throw new Error(\n        `Lambda function ${backendId} bundled code not found at: ${expectedIndexFile} or ${resolve(\n          __dirname,\n          '..',\n          'backend',\n          backendId,\n          'index.js'\n        )}. ` +\n          'Please run \"npm run build:lambdas\" to bundle the Lambda functions.'\n      );\n    }\n  }\n\n  console.log(\n    `📦 Using Lambda code for ${backendId} (id=${id}) from: ${codeLocation}`\n  );\n\n  // Get the current region for the Lambda Powertools layer\n  const region = Stack.of(scope).region;\n\n  // AWS Lambda Powertools layer ARN\n  // See: https://docs.powertools.aws.dev/lambda/typescript/latest/#lambda-layer\n  const powertoolsLayerArn = `arn:aws:lambda:${region}:094274105915:layer:AWSLambdaPowertoolsTypeScriptV2:22`;\n\n  // Check if we're in a FlowConfigStack and get VPC configuration\n  const stack = Stack.of(scope) as FlowConfigStack;\n  const vpcConfig = stack._getResolvedVpcConfig();\n\n  // Create the log group BEFORE the function so CloudFormation establishes the dependency\n  // correctly. If the log group is created after using func.functionName, Lambda may\n  // auto-create the log group on first invocation before CloudFormation gets to it,\n  // causing a \"already exists\" failure during stack creation.\n  const logGroup = new LogGroup(scope, `${id}LogGroup`, {\n    retention: props.logRetention ?? RetentionDays.ONE_MONTH,\n    removalPolicy: RemovalPolicy.DESTROY,\n  });\n\n  const func = new Function(scope, `${id}Function`, {\n    // Apply defaults\n    handler: 'index.handler',\n    runtime: Runtime.NODEJS_22_X,\n    architecture: Architecture.ARM_64,\n    tracing: Tracing.ACTIVE,\n    timeout: Duration.seconds(30),\n    code: Code.fromAsset(codeLocation),\n    layers: [\n      LayerVersion.fromLayerVersionArn(\n        scope,\n        `${id}PowertoolsLayer`,\n        powertoolsLayerArn\n      ),\n    ],\n    // VPC configuration if provided\n    ...(vpcConfig && {\n      vpc: vpcConfig.vpc,\n      vpcSubnets: { subnets: vpcConfig.privateSubnets },\n      securityGroups: vpcConfig.lambdaSecurityGroups,\n    }),\n    // Apply overrides\n    ...props,\n    // Ensure that the log retention custom resource is not created\n    logRetention: undefined,\n    // Use our managed log group (must come after spread to override any logGroup in props)\n    logGroup,\n    environment: {\n      // see https://docs.aws.amazon.com/lambda/latest/dg/typescript-exceptions.html\n      NODE_OPTIONS: '--enable-source-maps',\n      // Log level. Do not emit DEBUG in prod environment because it will bloat your CW costs\n      POWERTOOLS_LOG_LEVEL: 'DEBUG',\n      // Log incoming event\n      POWERTOOLS_LOGGER_LOG_EVENT: 'true',\n      ALERT_TOPIC_ARN: props.alertTopic?.topicArn ?? 'N/A',\n      ...props.environment,\n    },\n  });\n\n  props.alertTopic?.grantPublish(func);\n\n  func.role?.addManagedPolicy(\n    ManagedPolicy.fromAwsManagedPolicyName(\n      vpcConfig\n        ? 'service-role/AWSLambdaVPCAccessExecutionRole'\n        : 'service-role/AWSLambdaBasicExecutionRole'\n    )\n  );\n\n  return func;\n}\n"]}