@cxbuilder/flow-config 1.0.2 → 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 (35) hide show
  1. package/.jsii +145 -68
  2. package/CHANGELOG.md +38 -0
  3. package/README.md +8 -5
  4. package/dist/backend/FlowConfig/index.js +111 -16
  5. package/dist/backend/FlowConfig/index.js.map +3 -3
  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 +36 -14
  16. package/dist/infrastructure/FlowConfigStack.js +54 -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/docs/Permissions-v1.md +132 -0
  33. package/docs/{Permissions.md → Permissions-v2.md} +15 -15
  34. package/package.json +1 -1
  35. package/dist/backend/Static/static/assets/index-Bx9Z3cF9.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-Bx9Z3cF9.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.
88
+ */
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.
80
93
  */
81
- readonly vpc?: VpcConfig;
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;
@@ -105,6 +123,10 @@ export declare class FlowConfigStack extends cdk.Stack {
105
123
  get appUrl(): string;
106
124
  constructor(scope: Construct, id: string, props: FlowConfigStackProps);
107
125
  createUserPoolClient(): UserPoolClient;
126
+ /**
127
+ * Create Cognito User Groups for role-based access control
128
+ */
129
+ createUserPoolGroups(): void;
108
130
  /**
109
131
  * Associated Voicemail as Agent Workspace app
110
132
  */
@@ -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', {
@@ -48,6 +49,7 @@ class FlowConfigStack extends cdk.Stack {
48
49
  const getConfig = new GetConfig_1.GetConfig(this);
49
50
  this.api = new Api_1.Api(this);
50
51
  this.userPoolClient = this.createUserPoolClient();
52
+ this.createUserPoolGroups();
51
53
  this.associate3pApp();
52
54
  new cdk.CfnOutput(this, 'AppUrl', {
53
55
  value: this.appUrl,
@@ -83,6 +85,32 @@ class FlowConfigStack extends cdk.Stack {
83
85
  });
84
86
  return client;
85
87
  }
88
+ /**
89
+ * Create Cognito User Groups for role-based access control
90
+ */
91
+ createUserPoolGroups() {
92
+ // FlowConfigAdmin - Full CRUD access
93
+ new aws_cognito_1.CfnUserPoolGroup(this, 'FlowConfigAdminGroup', {
94
+ userPoolId: this.userPool.userPoolId,
95
+ groupName: 'FlowConfigAdmin',
96
+ description: 'Full administrative access to all flow configs - can create, read, update, and delete flow configs and all their properties',
97
+ precedence: 1,
98
+ });
99
+ // FlowConfigEdit - Edit variable/prompt values only
100
+ new aws_cognito_1.CfnUserPoolGroup(this, 'FlowConfigEditGroup', {
101
+ userPoolId: this.userPool.userPoolId,
102
+ groupName: 'FlowConfigEdit',
103
+ description: 'Edit access to flow configs - can read and modify variable values and prompt content but cannot add/remove fields or delete configs',
104
+ precedence: 2,
105
+ });
106
+ // FlowConfigRead - Read-only access
107
+ new aws_cognito_1.CfnUserPoolGroup(this, 'FlowConfigReadGroup', {
108
+ userPoolId: this.userPool.userPoolId,
109
+ groupName: 'FlowConfigRead',
110
+ description: 'Read-only access to all flow configs for reporting and reference purposes',
111
+ precedence: 3,
112
+ });
113
+ }
86
114
  /**
87
115
  * Associated Voicemail as Agent Workspace app
88
116
  */
@@ -133,13 +161,13 @@ class FlowConfigStack extends cdk.Stack {
133
161
  const tableName = prefix;
134
162
  if (!globalTable) {
135
163
  // Single-region table
136
- return new dynamodb.Table(this, 'FlowConfigsTable', {
164
+ return new dynamodb.TableV2(this, 'Table', {
137
165
  tableName,
138
166
  partitionKey: {
139
167
  name: 'id',
140
168
  type: dynamodb.AttributeType.STRING,
141
169
  },
142
- billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,
170
+ billing: dynamodb.Billing.onDemand(),
143
171
  pointInTimeRecoverySpecification: {
144
172
  pointInTimeRecoveryEnabled: true,
145
173
  },
@@ -151,50 +179,58 @@ class FlowConfigStack extends cdk.Stack {
151
179
  }
152
180
  if (globalTable.isPrimaryRegion) {
153
181
  // Primary region creates global table with replicas
154
- return new dynamodb.Table(this, 'FlowConfigsTable', {
182
+ const table = new dynamodb.TableV2(this, 'Table', {
155
183
  tableName,
156
184
  partitionKey: {
157
185
  name: 'id',
158
186
  type: dynamodb.AttributeType.STRING,
159
187
  },
160
- billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,
188
+ billing: dynamodb.Billing.onDemand(),
161
189
  pointInTimeRecoverySpecification: {
162
190
  pointInTimeRecoveryEnabled: true,
163
191
  },
164
- replicationRegions: globalTable.replicaRegions || [],
165
192
  deletionProtection: prod,
166
193
  removalPolicy: prod
167
194
  ? cdk.RemovalPolicy.RETAIN
168
195
  : cdk.RemovalPolicy.DESTROY,
169
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;
170
209
  }
171
210
  else {
172
211
  // Secondary region references existing global table
173
- 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}`);
174
213
  }
175
214
  }
176
215
  /**
177
216
  * Resolve VPC configuration string IDs to CDK objects
178
217
  */
179
218
  resolveVpcConfig(vpcConfig) {
180
- const { vpcId, lambdaSecurityGroupIds, privateSubnetIds, vpcEndpointSecurityGroupIds, } = vpcConfig;
219
+ const { vpcId, securityGroupIds, subnetIds } = vpcConfig;
181
220
  // Resolve VPC
182
221
  const vpc = ec2.Vpc.fromLookup(this, 'ResolvedVpc', { vpcId });
183
222
  // Resolve Lambda security groups
184
- 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));
185
224
  // Resolve private subnets
186
- const privateSubnets = privateSubnetIds.map((subnetId, index) => ec2.Subnet.fromSubnetId(this, `PrivateSubnet${index}`, subnetId));
187
- // Resolve VPC endpoint security groups
188
- 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));
189
226
  return {
190
227
  vpc,
191
228
  lambdaSecurityGroups,
192
229
  privateSubnets,
193
- vpcEndpointSecurityGroups,
194
230
  };
195
231
  }
196
232
  }
197
233
  exports.FlowConfigStack = FlowConfigStack;
198
234
  _a = JSII_RTTI_SYMBOL_1;
199
- FlowConfigStack[_a] = { fqn: "@cxbuilder/flow-config.FlowConfigStack", version: "1.0.2" };
200
- //# 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} 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\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   * 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":"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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  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
+ }