@cxbuilder/flow-config 1.1.0 → 2.0.0

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.
Files changed (33) hide show
  1. package/.jsii +131 -65
  2. package/CHANGELOG.md +20 -0
  3. package/README.md +8 -5
  4. package/dist/backend/FlowConfig/index.js +2 -1
  5. package/dist/backend/FlowConfig/index.js.map +2 -2
  6. package/dist/backend/GetConfig/index.js +1 -1
  7. package/dist/backend/GetConfig/index.js.map +2 -2
  8. package/dist/backend/Init/index.js +2 -1
  9. package/dist/backend/Init/index.js.map +2 -2
  10. package/dist/backend/Settings/index.js +255 -0
  11. package/dist/backend/Settings/index.js.map +7 -0
  12. package/dist/backend/Static/static/assets/index-Cejunttu.js +61 -0
  13. package/dist/backend/Static/static/assets/{index-NRh8x3FI.css → index-SZuscj14.css} +1 -1
  14. package/dist/backend/Static/static/index.html +3 -3
  15. package/dist/infrastructure/FlowConfigStack.d.ts +32 -14
  16. package/dist/infrastructure/FlowConfigStack.js +27 -18
  17. package/dist/infrastructure/GetConfig/index.js +2 -2
  18. package/dist/infrastructure/api/Api.d.ts +5 -2
  19. package/dist/infrastructure/api/Api.js +21 -15
  20. package/dist/infrastructure/api/Init/Init.interface.d.ts +4 -0
  21. package/dist/infrastructure/api/Init/Init.interface.js +1 -1
  22. package/dist/infrastructure/api/Init/index.js +2 -1
  23. package/dist/infrastructure/api/Settings/Settings.interface.d.ts +3 -0
  24. package/dist/infrastructure/api/Settings/Settings.interface.js +3 -0
  25. package/dist/infrastructure/api/Settings/index.d.ts +7 -0
  26. package/dist/infrastructure/api/Settings/index.js +21 -0
  27. package/dist/infrastructure/api/spec.yaml +122 -0
  28. package/dist/infrastructure/createLambda.js +1 -1
  29. package/dist/infrastructure/index.d.ts +1 -1
  30. package/dist/infrastructure/index.js +1 -1
  31. package/dist/infrastructure/tsconfig.tsbuildinfo +1 -1
  32. package/package.json +1 -1
  33. package/dist/backend/Static/static/assets/index-FHwnAA8f.js +0 -61
@@ -3,9 +3,9 @@
3
3
  <head>
4
4
  <meta charset="UTF-8" />
5
5
  <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
- <title>CXBuilder Flow Config</title>
7
- <script type="module" crossorigin src="/prod/assets/index-FHwnAA8f.js"></script>
8
- <link rel="stylesheet" crossorigin href="/prod/assets/index-NRh8x3FI.css">
6
+ <title>Flow Config</title>
7
+ <script type="module" crossorigin src="/prod/assets/index-Cejunttu.js"></script>
8
+ <link rel="stylesheet" crossorigin href="/prod/assets/index-SZuscj14.css">
9
9
  </head>
10
10
  <body>
11
11
  <div id="root" style="width: 100%;"></div>
@@ -18,9 +18,23 @@ export interface CognitoConfig {
18
18
  readonly ssoProviderName?: string;
19
19
  }
20
20
  /**
21
- * VPC configuration for private deployment using string IDs
21
+ * VPC configuration for API Gateway
22
+ * If provided, the API will be deployed in a private VPC.
22
23
  */
23
- export interface VpcConfig {
24
+ export interface ApiVpcConfig {
25
+ /**
26
+ * The VPC ID to use for the API
27
+ */
28
+ readonly vpcId: string;
29
+ /**
30
+ * The VPC endpoint ID to use for the API
31
+ */
32
+ readonly vpcEndpointId: string;
33
+ }
34
+ /**
35
+ * Lambda VPC configuration
36
+ */
37
+ export interface LambdaVpcConfig {
24
38
  /**
25
39
  * The VPC ID to deploy resources into
26
40
  */
@@ -28,15 +42,11 @@ export interface VpcConfig {
28
42
  /**
29
43
  * Security group IDs for Lambda functions
30
44
  */
31
- readonly lambdaSecurityGroupIds: string[];
45
+ readonly securityGroupIds: string[];
32
46
  /**
33
47
  * Private subnet IDs for Lambda functions
34
48
  */
35
- readonly privateSubnetIds: string[];
36
- /**
37
- * Security group IDs for VPC endpoints
38
- */
39
- readonly vpcEndpointSecurityGroupIds: string[];
49
+ readonly subnetIds: string[];
40
50
  }
41
51
  /**
42
52
  * Global table configuration for multi-region deployments
@@ -59,11 +69,11 @@ export interface ResolvedVpcConfig {
59
69
  readonly vpc: IVpc;
60
70
  readonly lambdaSecurityGroups: ISecurityGroup[];
61
71
  readonly privateSubnets: ISubnet[];
62
- readonly vpcEndpointSecurityGroups: ISecurityGroup[];
63
72
  }
64
73
  export interface FlowConfigStackProps extends cdk.StackProps {
65
74
  /**
66
- * Used for resource naming
75
+ * Used for resource naming. Will also be the name of the Connect Lambda
76
+ * @example `cxbuilder-flow-config`
67
77
  */
68
78
  readonly prefix: string;
69
79
  readonly cognito: CognitoConfig;
@@ -74,17 +84,25 @@ export interface FlowConfigStackProps extends cdk.StackProps {
74
84
  readonly alertEmails: string[];
75
85
  readonly prod?: boolean;
76
86
  /**
77
- * VPC configuration for private deployment.
78
- * If provided, the application will be configured for VPC-only access.
79
- * If undefined, uses the current public configuration.
87
+ * If provided, the API will be deployed in a VPC.
80
88
  */
81
- readonly vpc?: VpcConfig;
89
+ readonly apiVpcConfig?: ApiVpcConfig;
90
+ /**
91
+ * If provided, the Lambda functions will be deployed in a VPC.
92
+ * Note: VPC should contain endpoints to: CloudFormation, Lambda, DynamoDB, SNS, and Polly.
93
+ */
94
+ readonly lambdaVpcConfig?: LambdaVpcConfig;
82
95
  /**
83
96
  * Global table configuration for multi-region deployments.
84
97
  * If provided, enables global table support.
85
98
  * If undefined, creates a single-region table.
86
99
  */
87
100
  readonly globalTable?: GlobalTableConfig;
101
+ /**
102
+ * Set to false to remove CXBuilder branding from the web app.
103
+ * @default true
104
+ */
105
+ readonly branding?: boolean;
88
106
  }
89
107
  export declare class FlowConfigStack extends cdk.Stack {
90
108
  props: FlowConfigStackProps;
@@ -23,11 +23,10 @@ class FlowConfigStack extends cdk.Stack {
23
23
  return this._resolvedVpcConfig;
24
24
  }
25
25
  get appUrl() {
26
- const { region } = cdk.Stack.of(this);
27
- return `https://${this.api.restApi.restApiId}.execute-api.${region}.amazonaws.com/prod`;
26
+ return this.api.url;
28
27
  }
29
28
  constructor(scope, id, props) {
30
- const { prefix, cognito, alertEmails, prod = false, vpc, globalTable, } = props;
29
+ const { prefix, cognito, alertEmails, prod = false, lambdaVpcConfig, globalTable, } = props;
31
30
  super(scope, id, {
32
31
  ...props,
33
32
  stackName: props.stackName ?? prefix,
@@ -37,7 +36,9 @@ class FlowConfigStack extends cdk.Stack {
37
36
  });
38
37
  this.props = props;
39
38
  // Resolve VPC configuration if provided
40
- this._resolvedVpcConfig = vpc ? this.resolveVpcConfig(vpc) : undefined;
39
+ this._resolvedVpcConfig = lambdaVpcConfig
40
+ ? this.resolveVpcConfig(lambdaVpcConfig)
41
+ : undefined;
41
42
  // DynamoDB table for storing flow configs
42
43
  this.table = this.createTable(prefix, prod, globalTable);
43
44
  this.alertTopic = new aws_sns_1.Topic(this, 'AlertTopic', {
@@ -160,13 +161,13 @@ class FlowConfigStack extends cdk.Stack {
160
161
  const tableName = prefix;
161
162
  if (!globalTable) {
162
163
  // Single-region table
163
- return new dynamodb.Table(this, 'FlowConfigsTable', {
164
+ return new dynamodb.TableV2(this, 'Table', {
164
165
  tableName,
165
166
  partitionKey: {
166
167
  name: 'id',
167
168
  type: dynamodb.AttributeType.STRING,
168
169
  },
169
- billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,
170
+ billing: dynamodb.Billing.onDemand(),
170
171
  pointInTimeRecoverySpecification: {
171
172
  pointInTimeRecoveryEnabled: true,
172
173
  },
@@ -178,50 +179,58 @@ class FlowConfigStack extends cdk.Stack {
178
179
  }
179
180
  if (globalTable.isPrimaryRegion) {
180
181
  // Primary region creates global table with replicas
181
- return new dynamodb.Table(this, 'FlowConfigsTable', {
182
+ const table = new dynamodb.TableV2(this, 'Table', {
182
183
  tableName,
183
184
  partitionKey: {
184
185
  name: 'id',
185
186
  type: dynamodb.AttributeType.STRING,
186
187
  },
187
- billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,
188
+ billing: dynamodb.Billing.onDemand(),
188
189
  pointInTimeRecoverySpecification: {
189
190
  pointInTimeRecoveryEnabled: true,
190
191
  },
191
- replicationRegions: globalTable.replicaRegions || [],
192
192
  deletionProtection: prod,
193
193
  removalPolicy: prod
194
194
  ? cdk.RemovalPolicy.RETAIN
195
195
  : cdk.RemovalPolicy.DESTROY,
196
196
  });
197
+ if (globalTable.replicaRegions) {
198
+ for (const region of globalTable.replicaRegions) {
199
+ table.addReplica({
200
+ region,
201
+ deletionProtection: prod,
202
+ pointInTimeRecoverySpecification: {
203
+ pointInTimeRecoveryEnabled: true,
204
+ },
205
+ });
206
+ }
207
+ }
208
+ return table;
197
209
  }
198
210
  else {
199
211
  // Secondary region references existing global table
200
- return dynamodb.Table.fromTableArn(this, 'FlowConfigsTable', `arn:aws:dynamodb:${this.region}:${this.account}:table/${tableName}`);
212
+ return dynamodb.Table.fromTableArn(this, 'Table', `arn:aws:dynamodb:${this.region}:${this.account}:table/${tableName}`);
201
213
  }
202
214
  }
203
215
  /**
204
216
  * Resolve VPC configuration string IDs to CDK objects
205
217
  */
206
218
  resolveVpcConfig(vpcConfig) {
207
- const { vpcId, lambdaSecurityGroupIds, privateSubnetIds, vpcEndpointSecurityGroupIds, } = vpcConfig;
219
+ const { vpcId, securityGroupIds, subnetIds } = vpcConfig;
208
220
  // Resolve VPC
209
221
  const vpc = ec2.Vpc.fromLookup(this, 'ResolvedVpc', { vpcId });
210
222
  // Resolve Lambda security groups
211
- const lambdaSecurityGroups = lambdaSecurityGroupIds.map((sgId, index) => ec2.SecurityGroup.fromSecurityGroupId(this, `LambdaSecurityGroup${index}`, sgId));
223
+ const lambdaSecurityGroups = securityGroupIds.map((sgId, index) => ec2.SecurityGroup.fromSecurityGroupId(this, `LambdaSecurityGroup${index}`, sgId));
212
224
  // Resolve private subnets
213
- const privateSubnets = privateSubnetIds.map((subnetId, index) => ec2.Subnet.fromSubnetId(this, `PrivateSubnet${index}`, subnetId));
214
- // Resolve VPC endpoint security groups
215
- const vpcEndpointSecurityGroups = vpcEndpointSecurityGroupIds.map((sgId, index) => ec2.SecurityGroup.fromSecurityGroupId(this, `VpcEndpointSecurityGroup${index}`, sgId));
225
+ const privateSubnets = subnetIds.map((subnetId, index) => ec2.Subnet.fromSubnetId(this, `PrivateSubnet${index}`, subnetId));
216
226
  return {
217
227
  vpc,
218
228
  lambdaSecurityGroups,
219
229
  privateSubnets,
220
- vpcEndpointSecurityGroups,
221
230
  };
222
231
  }
223
232
  }
224
233
  exports.FlowConfigStack = FlowConfigStack;
225
234
  _a = JSII_RTTI_SYMBOL_1;
226
- FlowConfigStack[_a] = { fqn: "@cxbuilder/flow-config.FlowConfigStack", version: "1.1.0" };
227
- //# 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 private deployment using string IDs\n */\nexport interface VpcConfig {\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 lambdaSecurityGroupIds: string[];\n\n  /**\n   * Private subnet IDs for Lambda functions\n   */\n  readonly privateSubnetIds: string[];\n\n  /**\n   * Security group IDs for VPC endpoints\n   */\n  readonly vpcEndpointSecurityGroupIds: 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  readonly vpcEndpointSecurityGroups: ISecurityGroup[];\n}\n\nexport interface FlowConfigStackProps extends cdk.StackProps {\n  /**\n   * Used for resource naming\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   * VPC configuration for private deployment.\n   * If provided, the application will be configured for VPC-only access.\n   * If undefined, uses the current public configuration.\n   */\n  readonly vpc?: VpcConfig;\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\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    const { region } = cdk.Stack.of(this);\n    return `https://${this.api.restApi.restApiId}.execute-api.${region}.amazonaws.com/prod`;\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      vpc,\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 = vpc ? this.resolveVpcConfig(vpc) : 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   * Associated Voicemail as Agent Workspace app\n   */\n  associate3pApp() {\n    const {\n      props: { prefix, connectInstanceArn },\n      appUrl: url,\n    } = this;\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.Table(this, 'FlowConfigsTable', {\n        tableName,\n        partitionKey: {\n          name: 'id',\n          type: dynamodb.AttributeType.STRING,\n        },\n        billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,\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      return new dynamodb.Table(this, 'FlowConfigsTable', {\n        tableName,\n        partitionKey: {\n          name: 'id',\n          type: dynamodb.AttributeType.STRING,\n        },\n        billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,\n        pointInTimeRecoverySpecification: {\n          pointInTimeRecoveryEnabled: true,\n        },\n        replicationRegions: globalTable.replicaRegions || [],\n        deletionProtection: prod,\n        removalPolicy: prod\n          ? cdk.RemovalPolicy.RETAIN\n          : cdk.RemovalPolicy.DESTROY,\n      });\n    } else {\n      // Secondary region references existing global table\n      return dynamodb.Table.fromTableArn(\n        this,\n        'FlowConfigsTable',\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: VpcConfig): ResolvedVpcConfig {\n    const {\n      vpcId,\n      lambdaSecurityGroupIds,\n      privateSubnetIds,\n      vpcEndpointSecurityGroupIds,\n    } = vpcConfig;\n\n    // Resolve VPC\n    const vpc = ec2.Vpc.fromLookup(this, 'ResolvedVpc', { vpcId });\n\n    // Resolve Lambda security groups\n    const lambdaSecurityGroups = lambdaSecurityGroupIds.map((sgId, index) =>\n      ec2.SecurityGroup.fromSecurityGroupId(\n        this,\n        `LambdaSecurityGroup${index}`,\n        sgId\n      )\n    );\n\n    // Resolve private subnets\n    const privateSubnets = privateSubnetIds.map((subnetId, index) =>\n      ec2.Subnet.fromSubnetId(this, `PrivateSubnet${index}`, subnetId)\n    );\n\n    // Resolve VPC endpoint security groups\n    const vpcEndpointSecurityGroups = vpcEndpointSecurityGroupIds.map(\n      (sgId, index) =>\n        ec2.SecurityGroup.fromSecurityGroupId(\n          this,\n          `VpcEndpointSecurityGroup${index}`,\n          sgId\n        )\n    );\n\n    return {\n      vpc,\n      lambdaSecurityGroups,\n      privateSubnets,\n      vpcEndpointSecurityGroups,\n    };\n  }\n}\n"]}
235
+ FlowConfigStack[_a] = { fqn: "@cxbuilder/flow-config.FlowConfigStack", version: "2.0.0" };
236
+ //# 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\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   * Associated Voicemail as Agent Workspace app\n   */\n  associate3pApp() {\n    const {\n      props: { prefix, connectInstanceArn },\n      appUrl: url,\n    } = this;\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"]}
@@ -10,7 +10,7 @@ class GetConfig extends constructs_1.Construct {
10
10
  super(stack, 'GetConfig');
11
11
  const { connectInstanceArn, prefix } = stack.props;
12
12
  this.function = (0, createLambda_1.createLambda)(this, 'Handler', {
13
- functionName: `${prefix}-get-config`,
13
+ functionName: prefix,
14
14
  environment: {
15
15
  FLOW_CONFIGS_TABLE_NAME: stack.table.tableName,
16
16
  },
@@ -31,4 +31,4 @@ class GetConfig extends constructs_1.Construct {
31
31
  }
32
32
  }
33
33
  exports.GetConfig = GetConfig;
34
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9pbmZyYXN0cnVjdHVyZS9HZXRDb25maWcvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsMkNBQXVDO0FBQ3ZDLGtEQUErQztBQUkvQyx5REFBb0U7QUFDcEUsbUNBQW1DO0FBRW5DLE1BQWEsU0FBVSxTQUFRLHNCQUFTO0lBRXRDLFlBQVksS0FBc0I7UUFDaEMsS0FBSyxDQUFDLEtBQUssRUFBRSxXQUFXLENBQUMsQ0FBQztRQUUxQixNQUFNLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxFQUFFLEdBQUcsS0FBSyxDQUFDLEtBQUssQ0FBQztRQUVuRCxJQUFJLENBQUMsUUFBUSxHQUFHLElBQUEsMkJBQVksRUFBZSxJQUFJLEVBQUUsU0FBUyxFQUFFO1lBQzFELFlBQVksRUFBRSxHQUFHLE1BQU0sYUFBYTtZQUNwQyxXQUFXLEVBQUU7Z0JBQ1gsdUJBQXVCLEVBQUUsS0FBSyxDQUFDLEtBQUssQ0FBQyxTQUFTO2FBQy9DO1lBQ0QsVUFBVSxFQUFFLEtBQUssQ0FBQyxVQUFVO1NBQzdCLENBQUMsQ0FBQztRQUVILEtBQUssQ0FBQyxLQUFLLENBQUMsa0JBQWtCLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBRTlDLDRFQUE0RTtRQUM1RSxJQUFJLHVDQUF5QixDQUFDLElBQUksRUFBRSwwQkFBMEIsRUFBRTtZQUM5RCxVQUFVLEVBQUUsa0JBQWtCO1lBQzlCLGVBQWUsRUFBRSxpQkFBaUI7WUFDbEMsY0FBYyxFQUFFLElBQUksQ0FBQyxRQUFRLENBQUMsV0FBVztTQUMxQyxDQUFDLENBQUM7UUFFSCxnREFBZ0Q7UUFDaEQsSUFBSSxDQUFDLFFBQVEsQ0FBQyxhQUFhLENBQUMseUJBQXlCLEVBQUU7WUFDckQsU0FBUyxFQUFFLElBQUksR0FBRyxDQUFDLE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQyx1QkFBdUIsQ0FBQztZQUNwRSxTQUFTLEVBQUUsa0JBQWtCO1NBQzlCLENBQUMsQ0FBQztJQUNMLENBQUM7Q0FDRjtBQTlCRCw4QkE4QkMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbmltcG9ydCB7IGNyZWF0ZUxhbWJkYSB9IGZyb20gJy4uL2NyZWF0ZUxhbWJkYSc7XG5pbXBvcnQgeyBGdW5jdGlvbiB9IGZyb20gJ2F3cy1jZGstbGliL2F3cy1sYW1iZGEnO1xuaW1wb3J0IHsgRmxvd0NvbmZpZ1N0YWNrIH0gZnJvbSAnLi4vRmxvd0NvbmZpZ1N0YWNrJztcbmltcG9ydCB7IEdldENvbmZpZ0VudiB9IGZyb20gJy4vR2V0Q29uZmlnLmludGVyZmFjZSc7XG5pbXBvcnQgeyBDZm5JbnRlZ3JhdGlvbkFzc29jaWF0aW9uIH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWNvbm5lY3QnO1xuaW1wb3J0ICogYXMgY2RrIGZyb20gJ2F3cy1jZGstbGliJztcblxuZXhwb3J0IGNsYXNzIEdldENvbmZpZyBleHRlbmRzIENvbnN0cnVjdCB7XG4gIHJlYWRvbmx5IGZ1bmN0aW9uOiBGdW5jdGlvbjtcbiAgY29uc3RydWN0b3Ioc3RhY2s6IEZsb3dDb25maWdTdGFjaykge1xuICAgIHN1cGVyKHN0YWNrLCAnR2V0Q29uZmlnJyk7XG5cbiAgICBjb25zdCB7IGNvbm5lY3RJbnN0YW5jZUFybiwgcHJlZml4IH0gPSBzdGFjay5wcm9wcztcblxuICAgIHRoaXMuZnVuY3Rpb24gPSBjcmVhdGVMYW1iZGE8R2V0Q29uZmlnRW52Pih0aGlzLCAnSGFuZGxlcicsIHtcbiAgICAgIGZ1bmN0aW9uTmFtZTogYCR7cHJlZml4fS1nZXQtY29uZmlnYCxcbiAgICAgIGVudmlyb25tZW50OiB7XG4gICAgICAgIEZMT1dfQ09ORklHU19UQUJMRV9OQU1FOiBzdGFjay50YWJsZS50YWJsZU5hbWUsXG4gICAgICB9LFxuICAgICAgYWxlcnRUb3BpYzogc3RhY2suYWxlcnRUb3BpYyxcbiAgICB9KTtcblxuICAgIHN0YWNrLnRhYmxlLmdyYW50UmVhZFdyaXRlRGF0YSh0aGlzLmZ1bmN0aW9uKTtcblxuICAgIC8vIEFzc29jaWF0ZSBMYW1iZGEgd2l0aCBDb25uZWN0IGluc3RhbmNlIHNvIGl0IGNhbiBiZSB1c2VkIGluIGNvbnRhY3QgZmxvd3NcbiAgICBuZXcgQ2ZuSW50ZWdyYXRpb25Bc3NvY2lhdGlvbih0aGlzLCAnQ29ubmVjdExhbWJkYUFzc29jaWF0aW9uJywge1xuICAgICAgaW5zdGFuY2VJZDogY29ubmVjdEluc3RhbmNlQXJuLFxuICAgICAgaW50ZWdyYXRpb25UeXBlOiAnTEFNQkRBX0ZVTkNUSU9OJyxcbiAgICAgIGludGVncmF0aW9uQXJuOiB0aGlzLmZ1bmN0aW9uLmZ1bmN0aW9uQXJuLFxuICAgIH0pO1xuXG4gICAgLy8gR3JhbnQgQ29ubmVjdCBwZXJtaXNzaW9uIHRvIGludm9rZSB0aGUgTGFtYmRhXG4gICAgdGhpcy5mdW5jdGlvbi5hZGRQZXJtaXNzaW9uKCdDb25uZWN0SW52b2tlUGVybWlzc2lvbicsIHtcbiAgICAgIHByaW5jaXBhbDogbmV3IGNkay5hd3NfaWFtLlNlcnZpY2VQcmluY2lwYWwoJ2Nvbm5lY3QuYW1hem9uYXdzLmNvbScpLFxuICAgICAgc291cmNlQXJuOiBjb25uZWN0SW5zdGFuY2VBcm4sXG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
34
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9pbmZyYXN0cnVjdHVyZS9HZXRDb25maWcvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsMkNBQXVDO0FBQ3ZDLGtEQUErQztBQUkvQyx5REFBb0U7QUFDcEUsbUNBQW1DO0FBRW5DLE1BQWEsU0FBVSxTQUFRLHNCQUFTO0lBRXRDLFlBQVksS0FBc0I7UUFDaEMsS0FBSyxDQUFDLEtBQUssRUFBRSxXQUFXLENBQUMsQ0FBQztRQUUxQixNQUFNLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxFQUFFLEdBQUcsS0FBSyxDQUFDLEtBQUssQ0FBQztRQUVuRCxJQUFJLENBQUMsUUFBUSxHQUFHLElBQUEsMkJBQVksRUFBZSxJQUFJLEVBQUUsU0FBUyxFQUFFO1lBQzFELFlBQVksRUFBRSxNQUFNO1lBQ3BCLFdBQVcsRUFBRTtnQkFDWCx1QkFBdUIsRUFBRSxLQUFLLENBQUMsS0FBSyxDQUFDLFNBQVM7YUFDL0M7WUFDRCxVQUFVLEVBQUUsS0FBSyxDQUFDLFVBQVU7U0FDN0IsQ0FBQyxDQUFDO1FBRUgsS0FBSyxDQUFDLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUM7UUFFOUMsNEVBQTRFO1FBQzVFLElBQUksdUNBQXlCLENBQUMsSUFBSSxFQUFFLDBCQUEwQixFQUFFO1lBQzlELFVBQVUsRUFBRSxrQkFBa0I7WUFDOUIsZUFBZSxFQUFFLGlCQUFpQjtZQUNsQyxjQUFjLEVBQUUsSUFBSSxDQUFDLFFBQVEsQ0FBQyxXQUFXO1NBQzFDLENBQUMsQ0FBQztRQUVILGdEQUFnRDtRQUNoRCxJQUFJLENBQUMsUUFBUSxDQUFDLGFBQWEsQ0FBQyx5QkFBeUIsRUFBRTtZQUNyRCxTQUFTLEVBQUUsSUFBSSxHQUFHLENBQUMsT0FBTyxDQUFDLGdCQUFnQixDQUFDLHVCQUF1QixDQUFDO1lBQ3BFLFNBQVMsRUFBRSxrQkFBa0I7U0FDOUIsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBOUJELDhCQThCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuaW1wb3J0IHsgY3JlYXRlTGFtYmRhIH0gZnJvbSAnLi4vY3JlYXRlTGFtYmRhJztcbmltcG9ydCB7IEZ1bmN0aW9uIH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWxhbWJkYSc7XG5pbXBvcnQgeyBGbG93Q29uZmlnU3RhY2sgfSBmcm9tICcuLi9GbG93Q29uZmlnU3RhY2snO1xuaW1wb3J0IHsgR2V0Q29uZmlnRW52IH0gZnJvbSAnLi9HZXRDb25maWcuaW50ZXJmYWNlJztcbmltcG9ydCB7IENmbkludGVncmF0aW9uQXNzb2NpYXRpb24gfSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtY29ubmVjdCc7XG5pbXBvcnQgKiBhcyBjZGsgZnJvbSAnYXdzLWNkay1saWInO1xuXG5leHBvcnQgY2xhc3MgR2V0Q29uZmlnIGV4dGVuZHMgQ29uc3RydWN0IHtcbiAgcmVhZG9ubHkgZnVuY3Rpb246IEZ1bmN0aW9uO1xuICBjb25zdHJ1Y3RvcihzdGFjazogRmxvd0NvbmZpZ1N0YWNrKSB7XG4gICAgc3VwZXIoc3RhY2ssICdHZXRDb25maWcnKTtcblxuICAgIGNvbnN0IHsgY29ubmVjdEluc3RhbmNlQXJuLCBwcmVmaXggfSA9IHN0YWNrLnByb3BzO1xuXG4gICAgdGhpcy5mdW5jdGlvbiA9IGNyZWF0ZUxhbWJkYTxHZXRDb25maWdFbnY+KHRoaXMsICdIYW5kbGVyJywge1xuICAgICAgZnVuY3Rpb25OYW1lOiBwcmVmaXgsXG4gICAgICBlbnZpcm9ubWVudDoge1xuICAgICAgICBGTE9XX0NPTkZJR1NfVEFCTEVfTkFNRTogc3RhY2sudGFibGUudGFibGVOYW1lLFxuICAgICAgfSxcbiAgICAgIGFsZXJ0VG9waWM6IHN0YWNrLmFsZXJ0VG9waWMsXG4gICAgfSk7XG5cbiAgICBzdGFjay50YWJsZS5ncmFudFJlYWRXcml0ZURhdGEodGhpcy5mdW5jdGlvbik7XG5cbiAgICAvLyBBc3NvY2lhdGUgTGFtYmRhIHdpdGggQ29ubmVjdCBpbnN0YW5jZSBzbyBpdCBjYW4gYmUgdXNlZCBpbiBjb250YWN0IGZsb3dzXG4gICAgbmV3IENmbkludGVncmF0aW9uQXNzb2NpYXRpb24odGhpcywgJ0Nvbm5lY3RMYW1iZGFBc3NvY2lhdGlvbicsIHtcbiAgICAgIGluc3RhbmNlSWQ6IGNvbm5lY3RJbnN0YW5jZUFybixcbiAgICAgIGludGVncmF0aW9uVHlwZTogJ0xBTUJEQV9GVU5DVElPTicsXG4gICAgICBpbnRlZ3JhdGlvbkFybjogdGhpcy5mdW5jdGlvbi5mdW5jdGlvbkFybixcbiAgICB9KTtcblxuICAgIC8vIEdyYW50IENvbm5lY3QgcGVybWlzc2lvbiB0byBpbnZva2UgdGhlIExhbWJkYVxuICAgIHRoaXMuZnVuY3Rpb24uYWRkUGVybWlzc2lvbignQ29ubmVjdEludm9rZVBlcm1pc3Npb24nLCB7XG4gICAgICBwcmluY2lwYWw6IG5ldyBjZGsuYXdzX2lhbS5TZXJ2aWNlUHJpbmNpcGFsKCdjb25uZWN0LmFtYXpvbmF3cy5jb20nKSxcbiAgICAgIHNvdXJjZUFybjogY29ubmVjdEluc3RhbmNlQXJuLFxuICAgIH0pO1xuICB9XG59XG4iXX0=
@@ -1,11 +1,14 @@
1
1
  import { Construct } from 'constructs';
2
2
  import { SpecRestApi } from 'aws-cdk-lib/aws-apigateway';
3
+ import { IInterfaceVpcEndpoint } from 'aws-cdk-lib/aws-ec2';
3
4
  import { FlowConfigStack } from '../FlowConfigStack';
4
5
  import { Static } from './Static';
5
6
  export declare class Api extends Construct {
6
7
  stack: FlowConfigStack;
7
- staticHosting: Static;
8
- restApi: SpecRestApi;
8
+ readonly staticHosting: Static;
9
+ readonly restApi: SpecRestApi;
10
+ readonly vpcEndpoint?: IInterfaceVpcEndpoint;
11
+ readonly url: string;
9
12
  constructor(stack: FlowConfigStack);
10
13
  /**
11
14
  * Create a resource policy for the VPC endpoint to restrict access to the specific VPC
@@ -9,6 +9,7 @@ const aws_ec2_1 = require("aws-cdk-lib/aws-ec2");
9
9
  const Init_1 = require("./Init");
10
10
  const FlowConfig_1 = require("./FlowConfig");
11
11
  const PreviewSpeech_1 = require("./PreviewSpeech");
12
+ const Settings_1 = require("./Settings");
12
13
  const yaml_1 = require("yaml");
13
14
  const fs_1 = require("fs");
14
15
  const Users_1 = require("./Users");
@@ -19,32 +20,31 @@ class Api extends constructs_1.Construct {
19
20
  this.stack = stack;
20
21
  // Create static hosting construct first
21
22
  this.staticHosting = new Static_1.Static(this);
22
- // Configure API Gateway based on VPC settings
23
- const vpcConfig = stack._getResolvedVpcConfig();
24
- const apiGatewayProps = vpcConfig
23
+ const { prefix, apiVpcConfig: apiVpcEndpoint } = stack.props;
24
+ this.vpcEndpoint = apiVpcEndpoint
25
+ ? aws_ec2_1.InterfaceVpcEndpoint.fromInterfaceVpcEndpointAttributes(this, 'ExistingVpcEndpoint', {
26
+ vpcEndpointId: apiVpcEndpoint.vpcEndpointId,
27
+ port: 443,
28
+ })
29
+ : undefined;
30
+ const apiGatewayProps = this.vpcEndpoint && apiVpcEndpoint?.vpcId
25
31
  ? {
26
- restApiName: stack.props.prefix,
32
+ restApiName: prefix,
27
33
  endpointConfiguration: {
28
34
  types: [aws_apigateway_1.EndpointType.PRIVATE],
29
- vpcEndpoints: [
30
- new aws_ec2_1.InterfaceVpcEndpoint(this, 'ApiVpcEndpoint', {
31
- vpc: vpcConfig.vpc,
32
- service: aws_ec2_1.InterfaceVpcEndpointAwsService.APIGATEWAY,
33
- subnets: { subnets: vpcConfig.privateSubnets },
34
- securityGroups: vpcConfig.vpcEndpointSecurityGroups,
35
- }),
36
- ],
35
+ vpcEndpoints: [this.vpcEndpoint],
37
36
  },
38
- policy: this.createVpcEndpointPolicy(vpcConfig.vpc.vpcId),
37
+ policy: this.createVpcEndpointPolicy(apiVpcEndpoint.vpcId),
39
38
  }
40
39
  : {
41
- restApiName: stack.props.prefix,
40
+ restApiName: prefix,
42
41
  };
43
42
  // Create Lambda functions
44
43
  const initLambda = new Init_1.Init(this).lambda;
45
44
  const usersLambda = new Users_1.Users(this).lambda;
46
45
  const flowConfigLambda = new FlowConfig_1.FlowConfig(this).lambda;
47
46
  const previewSpeechLambda = new PreviewSpeech_1.PreviewSpeech(this).lambda;
47
+ const settingsLambda = new Settings_1.Settings(this).lambda;
48
48
  const staticLambda = this.staticHosting.handler;
49
49
  // Update Authorizer Provider ARN
50
50
  const sourceFileName = (0, path_1.join)(__dirname, 'spec.yaml');
@@ -67,6 +67,8 @@ class Api extends constructs_1.Construct {
67
67
  setLambdaIntegration('/api/flow-config/{id}', 'delete', flowConfigLambda);
68
68
  setLambdaIntegration('/api/flow-config/preview', 'post', flowConfigLambda);
69
69
  setLambdaIntegration('/api/preview-speech', 'post', previewSpeechLambda);
70
+ setLambdaIntegration('/api/settings', 'get', settingsLambda);
71
+ setLambdaIntegration('/api/settings', 'post', settingsLambda);
70
72
  setLambdaIntegration('/{proxy+}', 'get', staticLambda);
71
73
  setLambdaIntegration('/', 'get', staticLambda);
72
74
  // Create API Gateway from updated OpenAPI spec
@@ -80,7 +82,11 @@ class Api extends constructs_1.Construct {
80
82
  usersLambda.grantInvoke(apiGatewayPrincipal);
81
83
  flowConfigLambda.grantInvoke(apiGatewayPrincipal);
82
84
  previewSpeechLambda.grantInvoke(apiGatewayPrincipal);
85
+ settingsLambda.grantInvoke(apiGatewayPrincipal);
83
86
  staticLambda.grantInvoke(apiGatewayPrincipal);
87
+ this.url = this.vpcEndpoint
88
+ ? `https://${this.restApi.restApiId}-${this.vpcEndpoint.vpcEndpointId}.execute-api.${stack.region}.amazonaws.com/prod`
89
+ : `https://${this.restApi.restApiId}.execute-api.${stack.region}.amazonaws.com/prod`;
84
90
  }
85
91
  /**
86
92
  * Create a resource policy for the VPC endpoint to restrict access to the specific VPC
@@ -104,4 +110,4 @@ class Api extends constructs_1.Construct {
104
110
  }
105
111
  }
106
112
  exports.Api = Api;
107
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Api.js","sourceRoot":"","sources":["../../../infrastructure/api/Api.ts"],"names":[],"mappings":";;;AAAA,2CAAuC;AACvC,+BAA4B;AAC5B,+DAKoC;AACpC,iDAM6B;AAC7B,iDAG6B;AAC7B,iCAA8B;AAC9B,6CAA0C;AAC1C,mDAAgD;AAChD,+BAA6B;AAC7B,2BAAkC;AAElC,mCAAgC;AAChC,qCAAkC;AAElC,MAAa,GAAI,SAAQ,sBAAS;IAIhC,YAAmB,KAAsB;QACvC,KAAK,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;QADH,UAAK,GAAL,KAAK,CAAiB;QAGvC,wCAAwC;QACxC,IAAI,CAAC,aAAa,GAAG,IAAI,eAAM,CAAC,IAAI,CAAC,CAAC;QAEtC,8CAA8C;QAC9C,MAAM,SAAS,GAAG,KAAK,CAAC,qBAAqB,EAAE,CAAC;QAChD,MAAM,eAAe,GAAqB,SAAS;YACjD,CAAC,CAAC;gBACE,WAAW,EAAE,KAAK,CAAC,KAAK,CAAC,MAAM;gBAC/B,qBAAqB,EAAE;oBACrB,KAAK,EAAE,CAAC,6BAAY,CAAC,OAAO,CAAC;oBAC7B,YAAY,EAAE;wBACZ,IAAI,8BAAoB,CAAC,IAAI,EAAE,gBAAgB,EAAE;4BAC/C,GAAG,EAAE,SAAS,CAAC,GAAG;4BAClB,OAAO,EAAE,wCAA8B,CAAC,UAAU;4BAClD,OAAO,EAAE,EAAE,OAAO,EAAE,SAAS,CAAC,cAAc,EAAE;4BAC9C,cAAc,EAAE,SAAS,CAAC,yBAAyB;yBACpD,CAAC;qBACH;iBACF;gBACD,MAAM,EAAE,IAAI,CAAC,uBAAuB,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC;aAC1D;YACH,CAAC,CAAC;gBACE,WAAW,EAAE,KAAK,CAAC,KAAK,CAAC,MAAM;aAChC,CAAC;QAEN,0BAA0B;QAC1B,MAAM,UAAU,GAAG,IAAI,WAAI,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC;QACzC,MAAM,WAAW,GAAG,IAAI,aAAK,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC;QAC3C,MAAM,gBAAgB,GAAG,IAAI,uBAAU,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC;QACrD,MAAM,mBAAmB,GAAG,IAAI,6BAAa,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC;QAC3D,MAAM,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC;QAEhD,iCAAiC;QACjC,MAAM,cAAc,GAAG,IAAA,WAAI,EAAC,SAAS,EAAE,WAAW,CAAC,CAAC;QACpD,MAAM,GAAG,GAAG,IAAA,iBAAY,EAAC,cAAc,CAAC,CAAC,QAAQ,EAAE,CAAC;QACpD,MAAM,IAAI,GAAG,IAAA,YAAK,EAAC,GAAG,CAAC,CAAC;QAExB,MAAM,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,eAAe,CAAC,iBAAiB,CAAC;QACrE,UAAU,CAAC,gCAAgC,CAAC,CAAC,YAAY,GAAG;YAC1D,KAAK,CAAC,QAAQ,CAAC,WAAW;SAC3B,CAAC;QAEF,gDAAgD;QAChD,MAAM,oBAAoB,GAAG,CAC3B,IAAY,EACZ,MAAc,EACd,MAAW,EACX,EAAE;YACF,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,CACtB,iCAAiC,CAClC,CAAC,GAAG,GAAG,sBAAsB,KAAK,CAAC,MAAM,qCAAqC,MAAM,CAAC,WAAW,cAAc,CAAC;QAClH,CAAC,CAAC;QAEF,kEAAkE;QAClE,oBAAoB,CAAC,WAAW,EAAE,KAAK,EAAE,UAAU,CAAC,CAAC;QACrD,oBAAoB,CAAC,qBAAqB,EAAE,KAAK,EAAE,WAAW,CAAC,CAAC;QAChE,oBAAoB,CAAC,kBAAkB,EAAE,KAAK,EAAE,gBAAgB,CAAC,CAAC;QAClE,oBAAoB,CAAC,uBAAuB,EAAE,KAAK,EAAE,gBAAgB,CAAC,CAAC;QACvE,oBAAoB,CAAC,uBAAuB,EAAE,MAAM,EAAE,gBAAgB,CAAC,CAAC;QACxE,oBAAoB,CAAC,uBAAuB,EAAE,QAAQ,EAAE,gBAAgB,CAAC,CAAC;QAC1E,oBAAoB,CAAC,0BAA0B,EAAE,MAAM,EAAE,gBAAgB,CAAC,CAAC;QAC3E,oBAAoB,CAAC,qBAAqB,EAAE,MAAM,EAAE,mBAAmB,CAAC,CAAC;QACzE,oBAAoB,CAAC,WAAW,EAAE,KAAK,EAAE,YAAY,CAAC,CAAC;QACvD,oBAAoB,CAAC,GAAG,EAAE,KAAK,EAAE,YAAY,CAAC,CAAC;QAE/C,+CAA+C;QAC/C,IAAI,CAAC,OAAO,GAAG,IAAI,4BAAW,CAAC,IAAI,EAAE,SAAS,EAAE;YAC9C,GAAG,eAAe;YAClB,aAAa,EAAE,8BAAa,CAAC,UAAU,CAAC,IAAI,CAAC;SAC9C,CAAC,CAAC;QAEH,0DAA0D;QAC1D,MAAM,mBAAmB,GAAG,IAAI,0BAAgB,CAC9C,0BAA0B,CAC3B,CAAC;QACF,UAAU,CAAC,WAAW,CAAC,mBAAmB,CAAC,CAAC;QAC5C,WAAW,CAAC,WAAW,CAAC,mBAAmB,CAAC,CAAC;QAC7C,gBAAgB,CAAC,WAAW,CAAC,mBAAmB,CAAC,CAAC;QAClD,mBAAmB,CAAC,WAAW,CAAC,mBAAmB,CAAC,CAAC;QACrD,YAAY,CAAC,WAAW,CAAC,mBAAmB,CAAC,CAAC;IAChD,CAAC;IAED;;OAEG;IACK,uBAAuB,CAAC,KAAa;QAC3C,OAAO,IAAI,wBAAc,CAAC;YACxB,UAAU,EAAE;gBACV,IAAI,yBAAe,CAAC;oBAClB,MAAM,EAAE,gBAAM,CAAC,KAAK;oBACpB,UAAU,EAAE,CAAC,IAAI,sBAAY,EAAE,CAAC;oBAChC,OAAO,EAAE,CAAC,eAAe,CAAC;oBAC1B,SAAS,EAAE,CAAC,GAAG,CAAC;oBAChB,UAAU,EAAE;wBACV,YAAY,EAAE;4BACZ,eAAe,EAAE,KAAK;yBACvB;qBACF;iBACF,CAAC;aACH;SACF,CAAC,CAAC;IACL,CAAC;CACF;AA7GD,kBA6GC","sourcesContent":["import { Construct } from 'constructs';\nimport { join } from 'path';\nimport {\n  EndpointType,\n  RestApiBaseProps,\n  SpecRestApi,\n  ApiDefinition,\n} from 'aws-cdk-lib/aws-apigateway';\nimport {\n  PolicyDocument,\n  PolicyStatement,\n  Effect,\n  AnyPrincipal,\n  ServicePrincipal,\n} from 'aws-cdk-lib/aws-iam';\nimport {\n  InterfaceVpcEndpoint,\n  InterfaceVpcEndpointAwsService,\n} from 'aws-cdk-lib/aws-ec2';\nimport { Init } from './Init';\nimport { FlowConfig } from './FlowConfig';\nimport { PreviewSpeech } from './PreviewSpeech';\nimport { parse } from 'yaml';\nimport { readFileSync } from 'fs';\nimport { FlowConfigStack } from '../FlowConfigStack';\nimport { Users } from './Users';\nimport { Static } from './Static';\n\nexport class Api extends Construct {\n  staticHosting: Static;\n  restApi: SpecRestApi;\n\n  constructor(public stack: FlowConfigStack) {\n    super(stack, 'Api');\n\n    // Create static hosting construct first\n    this.staticHosting = new Static(this);\n\n    // Configure API Gateway based on VPC settings\n    const vpcConfig = stack._getResolvedVpcConfig();\n    const apiGatewayProps: RestApiBaseProps = vpcConfig\n      ? {\n          restApiName: stack.props.prefix,\n          endpointConfiguration: {\n            types: [EndpointType.PRIVATE],\n            vpcEndpoints: [\n              new InterfaceVpcEndpoint(this, 'ApiVpcEndpoint', {\n                vpc: vpcConfig.vpc,\n                service: InterfaceVpcEndpointAwsService.APIGATEWAY,\n                subnets: { subnets: vpcConfig.privateSubnets },\n                securityGroups: vpcConfig.vpcEndpointSecurityGroups,\n              }),\n            ],\n          },\n          policy: this.createVpcEndpointPolicy(vpcConfig.vpc.vpcId),\n        }\n      : {\n          restApiName: stack.props.prefix,\n        };\n\n    // Create Lambda functions\n    const initLambda = new Init(this).lambda;\n    const usersLambda = new Users(this).lambda;\n    const flowConfigLambda = new FlowConfig(this).lambda;\n    const previewSpeechLambda = new PreviewSpeech(this).lambda;\n    const staticLambda = this.staticHosting.handler;\n\n    // Update Authorizer Provider ARN\n    const sourceFileName = join(__dirname, 'spec.yaml');\n    const yml = readFileSync(sourceFileName).toString();\n    const spec = parse(yml);\n\n    const authorizer = spec.components.securitySchemes.CognitoAuthorizer;\n    authorizer['x-amazon-apigateway-authorizer'].providerARNs = [\n      stack.userPool.userPoolArn,\n    ];\n\n    // Helper function to set Lambda integration URI\n    const setLambdaIntegration = (\n      path: string,\n      method: string,\n      lambda: any\n    ) => {\n      spec.paths[path][method][\n        'x-amazon-apigateway-integration'\n      ].uri = `arn:aws:apigateway:${stack.region}:lambda:path/2015-03-31/functions/${lambda.functionArn}/invocations`;\n    };\n\n    // Update the spec to replace handler URIs with actual Lambda ARNs\n    setLambdaIntegration('/api/init', 'get', initLambda);\n    setLambdaIntegration('/api/users/{userId}', 'get', usersLambda);\n    setLambdaIntegration('/api/flow-config', 'get', flowConfigLambda);\n    setLambdaIntegration('/api/flow-config/{id}', 'get', flowConfigLambda);\n    setLambdaIntegration('/api/flow-config/{id}', 'post', flowConfigLambda);\n    setLambdaIntegration('/api/flow-config/{id}', 'delete', flowConfigLambda);\n    setLambdaIntegration('/api/flow-config/preview', 'post', flowConfigLambda);\n    setLambdaIntegration('/api/preview-speech', 'post', previewSpeechLambda);\n    setLambdaIntegration('/{proxy+}', 'get', staticLambda);\n    setLambdaIntegration('/', 'get', staticLambda);\n\n    // Create API Gateway from updated OpenAPI spec\n    this.restApi = new SpecRestApi(this, 'RestApi', {\n      ...apiGatewayProps,\n      apiDefinition: ApiDefinition.fromInline(spec),\n    });\n\n    // Grant API Gateway permission to invoke Lambda functions\n    const apiGatewayPrincipal = new ServicePrincipal(\n      'apigateway.amazonaws.com'\n    );\n    initLambda.grantInvoke(apiGatewayPrincipal);\n    usersLambda.grantInvoke(apiGatewayPrincipal);\n    flowConfigLambda.grantInvoke(apiGatewayPrincipal);\n    previewSpeechLambda.grantInvoke(apiGatewayPrincipal);\n    staticLambda.grantInvoke(apiGatewayPrincipal);\n  }\n\n  /**\n   * Create a resource policy for the VPC endpoint to restrict access to the specific VPC\n   */\n  private createVpcEndpointPolicy(vpcId: string): PolicyDocument {\n    return new PolicyDocument({\n      statements: [\n        new PolicyStatement({\n          effect: Effect.ALLOW,\n          principals: [new AnyPrincipal()],\n          actions: ['execute-api:*'],\n          resources: ['*'],\n          conditions: {\n            StringEquals: {\n              'aws:SourceVpc': vpcId,\n            },\n          },\n        }),\n      ],\n    });\n  }\n}\n"]}
113
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Api.js","sourceRoot":"","sources":["../../../infrastructure/api/Api.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from 'constructs';\nimport { join } from 'path';\nimport {\n  EndpointType,\n  RestApiBaseProps,\n  SpecRestApi,\n  ApiDefinition,\n} from 'aws-cdk-lib/aws-apigateway';\nimport {\n  PolicyDocument,\n  PolicyStatement,\n  Effect,\n  AnyPrincipal,\n  ServicePrincipal,\n} from 'aws-cdk-lib/aws-iam';\nimport {\n  IInterfaceVpcEndpoint,\n  InterfaceVpcEndpoint,\n} from 'aws-cdk-lib/aws-ec2';\nimport { Init } from './Init';\nimport { FlowConfig } from './FlowConfig';\nimport { PreviewSpeech } from './PreviewSpeech';\nimport { Settings } from './Settings';\nimport { parse } from 'yaml';\nimport { readFileSync } from 'fs';\nimport { FlowConfigStack } from '../FlowConfigStack';\nimport { Users } from './Users';\nimport { Static } from './Static';\n\nexport class Api extends Construct {\n  readonly staticHosting: Static;\n  readonly restApi: SpecRestApi;\n  readonly vpcEndpoint?: IInterfaceVpcEndpoint;\n  readonly url: string;\n\n  constructor(public stack: FlowConfigStack) {\n    super(stack, 'Api');\n\n    // Create static hosting construct first\n    this.staticHosting = new Static(this);\n\n    const { prefix, apiVpcConfig: apiVpcEndpoint } = stack.props;\n\n    this.vpcEndpoint = apiVpcEndpoint\n      ? InterfaceVpcEndpoint.fromInterfaceVpcEndpointAttributes(\n          this,\n          'ExistingVpcEndpoint',\n          {\n            vpcEndpointId: apiVpcEndpoint.vpcEndpointId,\n            port: 443,\n          }\n        )\n      : undefined;\n\n    const apiGatewayProps: RestApiBaseProps =\n      this.vpcEndpoint && apiVpcEndpoint?.vpcId\n        ? {\n            restApiName: prefix,\n            endpointConfiguration: {\n              types: [EndpointType.PRIVATE],\n              vpcEndpoints: [this.vpcEndpoint],\n            },\n            policy: this.createVpcEndpointPolicy(apiVpcEndpoint.vpcId),\n          }\n        : {\n            restApiName: prefix,\n          };\n\n    // Create Lambda functions\n    const initLambda = new Init(this).lambda;\n    const usersLambda = new Users(this).lambda;\n    const flowConfigLambda = new FlowConfig(this).lambda;\n    const previewSpeechLambda = new PreviewSpeech(this).lambda;\n    const settingsLambda = new Settings(this).lambda;\n    const staticLambda = this.staticHosting.handler;\n\n    // Update Authorizer Provider ARN\n    const sourceFileName = join(__dirname, 'spec.yaml');\n    const yml = readFileSync(sourceFileName).toString();\n    const spec = parse(yml);\n\n    const authorizer = spec.components.securitySchemes.CognitoAuthorizer;\n    authorizer['x-amazon-apigateway-authorizer'].providerARNs = [\n      stack.userPool.userPoolArn,\n    ];\n\n    // Helper function to set Lambda integration URI\n    const setLambdaIntegration = (\n      path: string,\n      method: string,\n      lambda: any\n    ) => {\n      spec.paths[path][method][\n        'x-amazon-apigateway-integration'\n      ].uri = `arn:aws:apigateway:${stack.region}:lambda:path/2015-03-31/functions/${lambda.functionArn}/invocations`;\n    };\n\n    // Update the spec to replace handler URIs with actual Lambda ARNs\n    setLambdaIntegration('/api/init', 'get', initLambda);\n    setLambdaIntegration('/api/users/{userId}', 'get', usersLambda);\n    setLambdaIntegration('/api/flow-config', 'get', flowConfigLambda);\n    setLambdaIntegration('/api/flow-config/{id}', 'get', flowConfigLambda);\n    setLambdaIntegration('/api/flow-config/{id}', 'post', flowConfigLambda);\n    setLambdaIntegration('/api/flow-config/{id}', 'delete', flowConfigLambda);\n    setLambdaIntegration('/api/flow-config/preview', 'post', flowConfigLambda);\n    setLambdaIntegration('/api/preview-speech', 'post', previewSpeechLambda);\n    setLambdaIntegration('/api/settings', 'get', settingsLambda);\n    setLambdaIntegration('/api/settings', 'post', settingsLambda);\n    setLambdaIntegration('/{proxy+}', 'get', staticLambda);\n    setLambdaIntegration('/', 'get', staticLambda);\n\n    // Create API Gateway from updated OpenAPI spec\n    this.restApi = new SpecRestApi(this, 'RestApi', {\n      ...apiGatewayProps,\n      apiDefinition: ApiDefinition.fromInline(spec),\n    });\n\n    // Grant API Gateway permission to invoke Lambda functions\n    const apiGatewayPrincipal = new ServicePrincipal(\n      'apigateway.amazonaws.com'\n    );\n    initLambda.grantInvoke(apiGatewayPrincipal);\n    usersLambda.grantInvoke(apiGatewayPrincipal);\n    flowConfigLambda.grantInvoke(apiGatewayPrincipal);\n    previewSpeechLambda.grantInvoke(apiGatewayPrincipal);\n    settingsLambda.grantInvoke(apiGatewayPrincipal);\n    staticLambda.grantInvoke(apiGatewayPrincipal);\n\n    this.url = this.vpcEndpoint\n      ? `https://${this.restApi.restApiId}-${this.vpcEndpoint.vpcEndpointId}.execute-api.${stack.region}.amazonaws.com/prod`\n      : `https://${this.restApi.restApiId}.execute-api.${stack.region}.amazonaws.com/prod`;\n  }\n\n  /**\n   * Create a resource policy for the VPC endpoint to restrict access to the specific VPC\n   */\n  private createVpcEndpointPolicy(vpcId: string): PolicyDocument {\n    return new PolicyDocument({\n      statements: [\n        new PolicyStatement({\n          effect: Effect.ALLOW,\n          principals: [new AnyPrincipal()],\n          actions: ['execute-api:*'],\n          resources: ['*'],\n          conditions: {\n            StringEquals: {\n              'aws:SourceVpc': vpcId,\n            },\n          },\n        }),\n      ],\n    });\n  }\n}\n"]}
@@ -2,4 +2,8 @@ export interface InitEnv {
2
2
  stackName: string;
3
3
  userPoolId: string;
4
4
  AWS_REGION?: string;
5
+ /**
6
+ * true/false flag to enable/disable CXBuilder branding in the web app.
7
+ */
8
+ branding: string;
5
9
  }
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiSW5pdC5pbnRlcmZhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9pbmZyYXN0cnVjdHVyZS9hcGkvSW5pdC9Jbml0LmludGVyZmFjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBJbml0RW52IHtcbiAgc3RhY2tOYW1lOiBzdHJpbmc7XG4gIHVzZXJQb29sSWQ6IHN0cmluZztcbiAgQVdTX1JFR0lPTj86IHN0cmluZztcbn1cbiJdfQ==
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiSW5pdC5pbnRlcmZhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9pbmZyYXN0cnVjdHVyZS9hcGkvSW5pdC9Jbml0LmludGVyZmFjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBJbml0RW52IHtcbiAgc3RhY2tOYW1lOiBzdHJpbmc7XG4gIHVzZXJQb29sSWQ6IHN0cmluZztcbiAgQVdTX1JFR0lPTj86IHN0cmluZztcbiAgLyoqXG4gICAqIHRydWUvZmFsc2UgZmxhZyB0byBlbmFibGUvZGlzYWJsZSBDWEJ1aWxkZXIgYnJhbmRpbmcgaW4gdGhlIHdlYiBhcHAuXG4gICAqL1xuICBicmFuZGluZzogc3RyaW5nO1xufVxuIl19
@@ -16,6 +16,7 @@ class Init extends constructs_1.Construct {
16
16
  environment: {
17
17
  stackName: stackName,
18
18
  userPoolId: api.stack.props.cognito.userPoolId,
19
+ branding: (api.stack.props.branding ?? true).toString(),
19
20
  },
20
21
  timeout: aws_cdk_lib_1.Duration.seconds(15),
21
22
  initialPolicy: [
@@ -32,4 +33,4 @@ class Init extends constructs_1.Construct {
32
33
  }
33
34
  }
34
35
  exports.Init = Init;
35
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9pbmZyYXN0cnVjdHVyZS9hcGkvSW5pdC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSwyQ0FBdUM7QUFJdkMsNkNBQThDO0FBQzlDLGlEQUE4RDtBQUM5RCxxREFBa0Q7QUFFbEQ7O0dBRUc7QUFDSCxNQUFhLElBQUssU0FBUSxzQkFBUztJQUVqQyxZQUFZLEdBQVE7UUFDbEIsS0FBSyxDQUFDLEdBQUcsRUFBRSxNQUFNLENBQUMsQ0FBQztRQUVuQixNQUFNLEVBQUUsU0FBUyxFQUFFLE1BQU0sRUFBRSxPQUFPLEVBQUUsR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUV0RCxJQUFJLENBQUMsTUFBTSxHQUFHLElBQUEsMkJBQVksRUFBVSxJQUFJLEVBQUUsU0FBUyxFQUFFO1lBQ25ELFdBQVcsRUFBRTtnQkFDWCxTQUFTLEVBQUUsU0FBUztnQkFDcEIsVUFBVSxFQUFFLEdBQUcsQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxVQUFVO2FBQy9DO1lBQ0QsT0FBTyxFQUFFLHNCQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUM3QixhQUFhLEVBQUU7Z0JBQ2IsSUFBSSx5QkFBZSxDQUFDO29CQUNsQixNQUFNLEVBQUUsZ0JBQU0sQ0FBQyxLQUFLO29CQUNwQixPQUFPLEVBQUUsQ0FBQywrQkFBK0IsQ0FBQztvQkFDMUMsU0FBUyxFQUFFO3dCQUNULDBCQUEwQixNQUFNLElBQUksT0FBTyxVQUFVLFNBQVMsSUFBSTtxQkFDbkU7aUJBQ0YsQ0FBQzthQUNIO1lBQ0QsVUFBVSxFQUFFLEdBQUcsQ0FBQyxLQUFLLENBQUMsVUFBVTtTQUNqQyxDQUFDLENBQUM7SUFDTCxDQUFDO0NBQ0Y7QUF6QkQsb0JBeUJDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5pbXBvcnQgeyBGdW5jdGlvbiB9IGZyb20gJ2F3cy1jZGstbGliL2F3cy1sYW1iZGEnO1xuaW1wb3J0IHsgSW5pdEVudiB9IGZyb20gJy4vSW5pdC5pbnRlcmZhY2UnO1xuaW1wb3J0IHsgQXBpIH0gZnJvbSAnLi4vQXBpJztcbmltcG9ydCB7IER1cmF0aW9uLCBTdGFjayB9IGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCB7IEVmZmVjdCwgUG9saWN5U3RhdGVtZW50IH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWlhbSc7XG5pbXBvcnQgeyBjcmVhdGVMYW1iZGEgfSBmcm9tICcuLi8uLi9jcmVhdGVMYW1iZGEnO1xuXG4vKipcbiAqIFByb3ZpZGVzIHJ1bi10aW1lIGNvbmZpZ3VyYXRpb24gZm9yIHRoZSBBUEkgc3VjaCBhcyBBdXRoLlxuICovXG5leHBvcnQgY2xhc3MgSW5pdCBleHRlbmRzIENvbnN0cnVjdCB7XG4gIHJlYWRvbmx5IGxhbWJkYTogRnVuY3Rpb247XG4gIGNvbnN0cnVjdG9yKGFwaTogQXBpKSB7XG4gICAgc3VwZXIoYXBpLCAnSW5pdCcpO1xuXG4gICAgY29uc3QgeyBzdGFja05hbWUsIHJlZ2lvbiwgYWNjb3VudCB9ID0gU3RhY2sub2YodGhpcyk7XG5cbiAgICB0aGlzLmxhbWJkYSA9IGNyZWF0ZUxhbWJkYTxJbml0RW52Pih0aGlzLCAnSGFuZGxlcicsIHtcbiAgICAgIGVudmlyb25tZW50OiB7XG4gICAgICAgIHN0YWNrTmFtZTogc3RhY2tOYW1lLFxuICAgICAgICB1c2VyUG9vbElkOiBhcGkuc3RhY2sucHJvcHMuY29nbml0by51c2VyUG9vbElkLFxuICAgICAgfSxcbiAgICAgIHRpbWVvdXQ6IER1cmF0aW9uLnNlY29uZHMoMTUpLFxuICAgICAgaW5pdGlhbFBvbGljeTogW1xuICAgICAgICBuZXcgUG9saWN5U3RhdGVtZW50KHtcbiAgICAgICAgICBlZmZlY3Q6IEVmZmVjdC5BTExPVyxcbiAgICAgICAgICBhY3Rpb25zOiBbJ2Nsb3VkZm9ybWF0aW9uOkRlc2NyaWJlU3RhY2tzJ10sXG4gICAgICAgICAgcmVzb3VyY2VzOiBbXG4gICAgICAgICAgICBgYXJuOmF3czpjbG91ZGZvcm1hdGlvbjoke3JlZ2lvbn06JHthY2NvdW50fTpzdGFjay8ke3N0YWNrTmFtZX0vKmAsXG4gICAgICAgICAgXSxcbiAgICAgICAgfSksXG4gICAgICBdLFxuICAgICAgYWxlcnRUb3BpYzogYXBpLnN0YWNrLmFsZXJ0VG9waWMsXG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
36
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9pbmZyYXN0cnVjdHVyZS9hcGkvSW5pdC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSwyQ0FBdUM7QUFJdkMsNkNBQThDO0FBQzlDLGlEQUE4RDtBQUM5RCxxREFBa0Q7QUFFbEQ7O0dBRUc7QUFDSCxNQUFhLElBQUssU0FBUSxzQkFBUztJQUVqQyxZQUFZLEdBQVE7UUFDbEIsS0FBSyxDQUFDLEdBQUcsRUFBRSxNQUFNLENBQUMsQ0FBQztRQUVuQixNQUFNLEVBQUUsU0FBUyxFQUFFLE1BQU0sRUFBRSxPQUFPLEVBQUUsR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUV0RCxJQUFJLENBQUMsTUFBTSxHQUFHLElBQUEsMkJBQVksRUFBVSxJQUFJLEVBQUUsU0FBUyxFQUFFO1lBQ25ELFdBQVcsRUFBRTtnQkFDWCxTQUFTLEVBQUUsU0FBUztnQkFDcEIsVUFBVSxFQUFFLEdBQUcsQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxVQUFVO2dCQUM5QyxRQUFRLEVBQUUsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxRQUFRLElBQUksSUFBSSxDQUFDLENBQUMsUUFBUSxFQUFFO2FBQ3hEO1lBQ0QsT0FBTyxFQUFFLHNCQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUM3QixhQUFhLEVBQUU7Z0JBQ2IsSUFBSSx5QkFBZSxDQUFDO29CQUNsQixNQUFNLEVBQUUsZ0JBQU0sQ0FBQyxLQUFLO29CQUNwQixPQUFPLEVBQUUsQ0FBQywrQkFBK0IsQ0FBQztvQkFDMUMsU0FBUyxFQUFFO3dCQUNULDBCQUEwQixNQUFNLElBQUksT0FBTyxVQUFVLFNBQVMsSUFBSTtxQkFDbkU7aUJBQ0YsQ0FBQzthQUNIO1lBQ0QsVUFBVSxFQUFFLEdBQUcsQ0FBQyxLQUFLLENBQUMsVUFBVTtTQUNqQyxDQUFDLENBQUM7SUFDTCxDQUFDO0NBQ0Y7QUExQkQsb0JBMEJDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5pbXBvcnQgeyBGdW5jdGlvbiB9IGZyb20gJ2F3cy1jZGstbGliL2F3cy1sYW1iZGEnO1xuaW1wb3J0IHsgSW5pdEVudiB9IGZyb20gJy4vSW5pdC5pbnRlcmZhY2UnO1xuaW1wb3J0IHsgQXBpIH0gZnJvbSAnLi4vQXBpJztcbmltcG9ydCB7IER1cmF0aW9uLCBTdGFjayB9IGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCB7IEVmZmVjdCwgUG9saWN5U3RhdGVtZW50IH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWlhbSc7XG5pbXBvcnQgeyBjcmVhdGVMYW1iZGEgfSBmcm9tICcuLi8uLi9jcmVhdGVMYW1iZGEnO1xuXG4vKipcbiAqIFByb3ZpZGVzIHJ1bi10aW1lIGNvbmZpZ3VyYXRpb24gZm9yIHRoZSBBUEkgc3VjaCBhcyBBdXRoLlxuICovXG5leHBvcnQgY2xhc3MgSW5pdCBleHRlbmRzIENvbnN0cnVjdCB7XG4gIHJlYWRvbmx5IGxhbWJkYTogRnVuY3Rpb247XG4gIGNvbnN0cnVjdG9yKGFwaTogQXBpKSB7XG4gICAgc3VwZXIoYXBpLCAnSW5pdCcpO1xuXG4gICAgY29uc3QgeyBzdGFja05hbWUsIHJlZ2lvbiwgYWNjb3VudCB9ID0gU3RhY2sub2YodGhpcyk7XG5cbiAgICB0aGlzLmxhbWJkYSA9IGNyZWF0ZUxhbWJkYTxJbml0RW52Pih0aGlzLCAnSGFuZGxlcicsIHtcbiAgICAgIGVudmlyb25tZW50OiB7XG4gICAgICAgIHN0YWNrTmFtZTogc3RhY2tOYW1lLFxuICAgICAgICB1c2VyUG9vbElkOiBhcGkuc3RhY2sucHJvcHMuY29nbml0by51c2VyUG9vbElkLFxuICAgICAgICBicmFuZGluZzogKGFwaS5zdGFjay5wcm9wcy5icmFuZGluZyA/PyB0cnVlKS50b1N0cmluZygpLFxuICAgICAgfSxcbiAgICAgIHRpbWVvdXQ6IER1cmF0aW9uLnNlY29uZHMoMTUpLFxuICAgICAgaW5pdGlhbFBvbGljeTogW1xuICAgICAgICBuZXcgUG9saWN5U3RhdGVtZW50KHtcbiAgICAgICAgICBlZmZlY3Q6IEVmZmVjdC5BTExPVyxcbiAgICAgICAgICBhY3Rpb25zOiBbJ2Nsb3VkZm9ybWF0aW9uOkRlc2NyaWJlU3RhY2tzJ10sXG4gICAgICAgICAgcmVzb3VyY2VzOiBbXG4gICAgICAgICAgICBgYXJuOmF3czpjbG91ZGZvcm1hdGlvbjoke3JlZ2lvbn06JHthY2NvdW50fTpzdGFjay8ke3N0YWNrTmFtZX0vKmAsXG4gICAgICAgICAgXSxcbiAgICAgICAgfSksXG4gICAgICBdLFxuICAgICAgYWxlcnRUb3BpYzogYXBpLnN0YWNrLmFsZXJ0VG9waWMsXG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,3 @@
1
+ export interface SettingsEnv {
2
+ TABLE_NAME: string;
3
+ }
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiU2V0dGluZ3MuaW50ZXJmYWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vaW5mcmFzdHJ1Y3R1cmUvYXBpL1NldHRpbmdzL1NldHRpbmdzLmludGVyZmFjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBTZXR0aW5nc0VudiB7XG4gIFRBQkxFX05BTUU6IHN0cmluZztcbn1cbiJdfQ==
@@ -0,0 +1,7 @@
1
+ import { Construct } from 'constructs';
2
+ import { Function } from 'aws-cdk-lib/aws-lambda';
3
+ import { Api } from '../Api';
4
+ export declare class Settings extends Construct {
5
+ readonly lambda: Function;
6
+ constructor(api: Api);
7
+ }
@@ -0,0 +1,21 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Settings = void 0;
4
+ const constructs_1 = require("constructs");
5
+ const createLambda_1 = require("../../createLambda");
6
+ class Settings extends constructs_1.Construct {
7
+ constructor(api) {
8
+ super(api, 'Settings');
9
+ // Create Lambda function
10
+ this.lambda = (0, createLambda_1.createLambda)(this, 'Handler', {
11
+ environment: {
12
+ TABLE_NAME: api.stack.table.tableName,
13
+ },
14
+ alertTopic: api.stack.alertTopic,
15
+ });
16
+ // Grant DynamoDB permissions
17
+ api.stack.table.grantReadWriteData(this.lambda);
18
+ }
19
+ }
20
+ exports.Settings = Settings;
21
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9pbmZyYXN0cnVjdHVyZS9hcGkvU2V0dGluZ3MvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsMkNBQXVDO0FBQ3ZDLHFEQUFrRDtBQUtsRCxNQUFhLFFBQVMsU0FBUSxzQkFBUztJQUdyQyxZQUFZLEdBQVE7UUFDbEIsS0FBSyxDQUFDLEdBQUcsRUFBRSxVQUFVLENBQUMsQ0FBQztRQUV2Qix5QkFBeUI7UUFDekIsSUFBSSxDQUFDLE1BQU0sR0FBRyxJQUFBLDJCQUFZLEVBQWMsSUFBSSxFQUFFLFNBQVMsRUFBRTtZQUN2RCxXQUFXLEVBQUU7Z0JBQ1gsVUFBVSxFQUFFLEdBQUcsQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLFNBQVM7YUFDdEM7WUFDRCxVQUFVLEVBQUUsR0FBRyxDQUFDLEtBQUssQ0FBQyxVQUFVO1NBQ2pDLENBQUMsQ0FBQztRQUVILDZCQUE2QjtRQUM3QixHQUFHLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDbEQsQ0FBQztDQUNGO0FBakJELDRCQWlCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuaW1wb3J0IHsgY3JlYXRlTGFtYmRhIH0gZnJvbSAnLi4vLi4vY3JlYXRlTGFtYmRhJztcbmltcG9ydCB7IEZ1bmN0aW9uIH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWxhbWJkYSc7XG5pbXBvcnQgeyBTZXR0aW5nc0VudiB9IGZyb20gJy4vU2V0dGluZ3MuaW50ZXJmYWNlJztcbmltcG9ydCB7IEFwaSB9IGZyb20gJy4uL0FwaSc7XG5cbmV4cG9ydCBjbGFzcyBTZXR0aW5ncyBleHRlbmRzIENvbnN0cnVjdCB7XG4gIHJlYWRvbmx5IGxhbWJkYTogRnVuY3Rpb247XG5cbiAgY29uc3RydWN0b3IoYXBpOiBBcGkpIHtcbiAgICBzdXBlcihhcGksICdTZXR0aW5ncycpO1xuXG4gICAgLy8gQ3JlYXRlIExhbWJkYSBmdW5jdGlvblxuICAgIHRoaXMubGFtYmRhID0gY3JlYXRlTGFtYmRhPFNldHRpbmdzRW52Pih0aGlzLCAnSGFuZGxlcicsIHtcbiAgICAgIGVudmlyb25tZW50OiB7XG4gICAgICAgIFRBQkxFX05BTUU6IGFwaS5zdGFjay50YWJsZS50YWJsZU5hbWUsXG4gICAgICB9LFxuICAgICAgYWxlcnRUb3BpYzogYXBpLnN0YWNrLmFsZXJ0VG9waWMsXG4gICAgfSk7XG5cbiAgICAvLyBHcmFudCBEeW5hbW9EQiBwZXJtaXNzaW9uc1xuICAgIGFwaS5zdGFjay50YWJsZS5ncmFudFJlYWRXcml0ZURhdGEodGhpcy5sYW1iZGEpO1xuICB9XG59XG4iXX0=