@cxbuilder/flow-config 1.1.0 → 2.1.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.
- package/.jsii +151 -66
- package/CHANGELOG.md +24 -0
- package/README.md +8 -5
- package/dist/backend/FlowConfig/index.js +2 -1
- package/dist/backend/FlowConfig/index.js.map +2 -2
- package/dist/backend/GetConfig/index.js +1 -1
- package/dist/backend/GetConfig/index.js.map +2 -2
- package/dist/backend/Init/index.js +2 -1
- package/dist/backend/Init/index.js.map +2 -2
- package/dist/backend/Settings/index.js +255 -0
- package/dist/backend/Settings/index.js.map +7 -0
- package/dist/backend/Static/static/assets/index-Cejunttu.js +61 -0
- package/dist/backend/Static/static/assets/{index-NRh8x3FI.css → index-SZuscj14.css} +1 -1
- package/dist/backend/Static/static/index.html +3 -3
- package/dist/infrastructure/FlowConfigStack.d.ts +39 -15
- package/dist/infrastructure/FlowConfigStack.js +32 -20
- package/dist/infrastructure/GetConfig/index.js +2 -2
- package/dist/infrastructure/api/Api.d.ts +5 -2
- package/dist/infrastructure/api/Api.js +21 -15
- package/dist/infrastructure/api/Init/Init.interface.d.ts +4 -0
- package/dist/infrastructure/api/Init/Init.interface.js +1 -1
- package/dist/infrastructure/api/Init/index.js +2 -1
- package/dist/infrastructure/api/Settings/Settings.interface.d.ts +3 -0
- package/dist/infrastructure/api/Settings/Settings.interface.js +3 -0
- package/dist/infrastructure/api/Settings/index.d.ts +7 -0
- package/dist/infrastructure/api/Settings/index.js +21 -0
- package/dist/infrastructure/api/spec.yaml +122 -0
- package/dist/infrastructure/createLambda.js +1 -1
- package/dist/infrastructure/index.d.ts +1 -1
- package/dist/infrastructure/index.js +1 -1
- package/dist/infrastructure/tsconfig.tsbuildinfo +1 -1
- package/package.json +1 -1
- 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>
|
|
7
|
-
<script type="module" crossorigin src="/prod/assets/index-
|
|
8
|
-
<link rel="stylesheet" crossorigin href="/prod/assets/index-
|
|
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
|
|
21
|
+
* VPC configuration for API Gateway
|
|
22
|
+
* If provided, the API will be deployed in a private VPC.
|
|
22
23
|
*/
|
|
23
|
-
export interface
|
|
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
|
|
45
|
+
readonly securityGroupIds: string[];
|
|
32
46
|
/**
|
|
33
47
|
* Private subnet IDs for Lambda functions
|
|
34
48
|
*/
|
|
35
|
-
readonly
|
|
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,31 @@ export interface FlowConfigStackProps extends cdk.StackProps {
|
|
|
74
84
|
readonly alertEmails: string[];
|
|
75
85
|
readonly prod?: boolean;
|
|
76
86
|
/**
|
|
77
|
-
*
|
|
78
|
-
|
|
79
|
-
|
|
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
|
|
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;
|
|
106
|
+
/**
|
|
107
|
+
* Whether to associate the app with the Connect Agent Workspace.
|
|
108
|
+
* Set to false to disable automatic association.
|
|
109
|
+
* @default true
|
|
110
|
+
*/
|
|
111
|
+
readonly associate3pApp?: boolean;
|
|
88
112
|
}
|
|
89
113
|
export declare class FlowConfigStack extends cdk.Stack {
|
|
90
114
|
props: FlowConfigStackProps;
|
|
@@ -110,7 +134,7 @@ export declare class FlowConfigStack extends cdk.Stack {
|
|
|
110
134
|
*/
|
|
111
135
|
createUserPoolGroups(): void;
|
|
112
136
|
/**
|
|
113
|
-
*
|
|
137
|
+
* Associate FlowConfig as Agent Workspace app
|
|
114
138
|
*/
|
|
115
139
|
associate3pApp(): void;
|
|
116
140
|
/**
|
|
@@ -23,11 +23,10 @@ class FlowConfigStack extends cdk.Stack {
|
|
|
23
23
|
return this._resolvedVpcConfig;
|
|
24
24
|
}
|
|
25
25
|
get appUrl() {
|
|
26
|
-
|
|
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,
|
|
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 =
|
|
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', {
|
|
@@ -111,10 +112,13 @@ class FlowConfigStack extends cdk.Stack {
|
|
|
111
112
|
});
|
|
112
113
|
}
|
|
113
114
|
/**
|
|
114
|
-
*
|
|
115
|
+
* Associate FlowConfig as Agent Workspace app
|
|
115
116
|
*/
|
|
116
117
|
associate3pApp() {
|
|
117
|
-
const { props: { prefix, connectInstanceArn }, appUrl: url, } = this;
|
|
118
|
+
const { props: { prefix, connectInstanceArn }, appUrl: url, associate3pApp = true, } = this;
|
|
119
|
+
if (!associate3pApp) {
|
|
120
|
+
return;
|
|
121
|
+
}
|
|
118
122
|
let namespace = `cxbuilder.${prefix}`;
|
|
119
123
|
if (namespace.length > 32 && !cdk.Token.isUnresolved(namespace)) {
|
|
120
124
|
namespace = namespace.substring(0, 32);
|
|
@@ -160,13 +164,13 @@ class FlowConfigStack extends cdk.Stack {
|
|
|
160
164
|
const tableName = prefix;
|
|
161
165
|
if (!globalTable) {
|
|
162
166
|
// Single-region table
|
|
163
|
-
return new dynamodb.
|
|
167
|
+
return new dynamodb.TableV2(this, 'Table', {
|
|
164
168
|
tableName,
|
|
165
169
|
partitionKey: {
|
|
166
170
|
name: 'id',
|
|
167
171
|
type: dynamodb.AttributeType.STRING,
|
|
168
172
|
},
|
|
169
|
-
|
|
173
|
+
billing: dynamodb.Billing.onDemand(),
|
|
170
174
|
pointInTimeRecoverySpecification: {
|
|
171
175
|
pointInTimeRecoveryEnabled: true,
|
|
172
176
|
},
|
|
@@ -178,50 +182,58 @@ class FlowConfigStack extends cdk.Stack {
|
|
|
178
182
|
}
|
|
179
183
|
if (globalTable.isPrimaryRegion) {
|
|
180
184
|
// Primary region creates global table with replicas
|
|
181
|
-
|
|
185
|
+
const table = new dynamodb.TableV2(this, 'Table', {
|
|
182
186
|
tableName,
|
|
183
187
|
partitionKey: {
|
|
184
188
|
name: 'id',
|
|
185
189
|
type: dynamodb.AttributeType.STRING,
|
|
186
190
|
},
|
|
187
|
-
|
|
191
|
+
billing: dynamodb.Billing.onDemand(),
|
|
188
192
|
pointInTimeRecoverySpecification: {
|
|
189
193
|
pointInTimeRecoveryEnabled: true,
|
|
190
194
|
},
|
|
191
|
-
replicationRegions: globalTable.replicaRegions || [],
|
|
192
195
|
deletionProtection: prod,
|
|
193
196
|
removalPolicy: prod
|
|
194
197
|
? cdk.RemovalPolicy.RETAIN
|
|
195
198
|
: cdk.RemovalPolicy.DESTROY,
|
|
196
199
|
});
|
|
200
|
+
if (globalTable.replicaRegions) {
|
|
201
|
+
for (const region of globalTable.replicaRegions) {
|
|
202
|
+
table.addReplica({
|
|
203
|
+
region,
|
|
204
|
+
deletionProtection: prod,
|
|
205
|
+
pointInTimeRecoverySpecification: {
|
|
206
|
+
pointInTimeRecoveryEnabled: true,
|
|
207
|
+
},
|
|
208
|
+
});
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
return table;
|
|
197
212
|
}
|
|
198
213
|
else {
|
|
199
214
|
// Secondary region references existing global table
|
|
200
|
-
return dynamodb.Table.fromTableArn(this, '
|
|
215
|
+
return dynamodb.Table.fromTableArn(this, 'Table', `arn:aws:dynamodb:${this.region}:${this.account}:table/${tableName}`);
|
|
201
216
|
}
|
|
202
217
|
}
|
|
203
218
|
/**
|
|
204
219
|
* Resolve VPC configuration string IDs to CDK objects
|
|
205
220
|
*/
|
|
206
221
|
resolveVpcConfig(vpcConfig) {
|
|
207
|
-
const { vpcId,
|
|
222
|
+
const { vpcId, securityGroupIds, subnetIds } = vpcConfig;
|
|
208
223
|
// Resolve VPC
|
|
209
224
|
const vpc = ec2.Vpc.fromLookup(this, 'ResolvedVpc', { vpcId });
|
|
210
225
|
// Resolve Lambda security groups
|
|
211
|
-
const lambdaSecurityGroups =
|
|
226
|
+
const lambdaSecurityGroups = securityGroupIds.map((sgId, index) => ec2.SecurityGroup.fromSecurityGroupId(this, `LambdaSecurityGroup${index}`, sgId));
|
|
212
227
|
// Resolve private subnets
|
|
213
|
-
const privateSubnets =
|
|
214
|
-
// Resolve VPC endpoint security groups
|
|
215
|
-
const vpcEndpointSecurityGroups = vpcEndpointSecurityGroupIds.map((sgId, index) => ec2.SecurityGroup.fromSecurityGroupId(this, `VpcEndpointSecurityGroup${index}`, sgId));
|
|
228
|
+
const privateSubnets = subnetIds.map((subnetId, index) => ec2.Subnet.fromSubnetId(this, `PrivateSubnet${index}`, subnetId));
|
|
216
229
|
return {
|
|
217
230
|
vpc,
|
|
218
231
|
lambdaSecurityGroups,
|
|
219
232
|
privateSubnets,
|
|
220
|
-
vpcEndpointSecurityGroups,
|
|
221
233
|
};
|
|
222
234
|
}
|
|
223
235
|
}
|
|
224
236
|
exports.FlowConfigStack = FlowConfigStack;
|
|
225
237
|
_a = JSII_RTTI_SYMBOL_1;
|
|
226
|
-
FlowConfigStack[_a] = { fqn: "@cxbuilder/flow-config.FlowConfigStack", version: "
|
|
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"]}
|
|
238
|
+
FlowConfigStack[_a] = { fqn: "@cxbuilder/flow-config.FlowConfigStack", version: "2.1.0" };
|
|
239
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FlowConfigStack.js","sourceRoot":"","sources":["../../infrastructure/FlowConfigStack.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  IUserPool,\n  UserPool,\n  UserPoolClient,\n  UserPoolClientIdentityProvider,\n  CfnUserPoolGroup,\n} from 'aws-cdk-lib/aws-cognito';\nimport * as dynamodb from 'aws-cdk-lib/aws-dynamodb';\nimport { IVpc, ISecurityGroup, ISubnet } from 'aws-cdk-lib/aws-ec2';\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\nimport { Construct } from 'constructs';\nimport { Api } from './api/Api';\nimport { Topic } from 'aws-cdk-lib/aws-sns';\nimport { EmailSubscription } from 'aws-cdk-lib/aws-sns-subscriptions';\nimport { SecurityProfileProvider } from './SecurityProfileProvider';\nimport { CfnIntegrationAssociation } from 'aws-cdk-lib/aws-connect';\nimport { CfnApplication } from 'aws-cdk-lib/aws-appintegrations';\nimport { GetConfig } from './GetConfig';\n\n/**\n * Cognito configuration for FlowConfig stack\n */\nexport interface CognitoConfig {\n  readonly userPoolId: string;\n\n  /**\n   * Full domain name\n   */\n  readonly domain: string;\n\n  /**\n   * If provided, client will auth to SSO. Otherwise will auth to user pool\n   */\n  readonly ssoProviderName?: string;\n}\n\n/**\n * VPC configuration for API Gateway\n * If provided, the API will be deployed in a private VPC.\n */\nexport interface ApiVpcConfig {\n  /**\n   * The VPC ID to use for the API\n   */\n  readonly vpcId: string;\n  /**\n   * The VPC endpoint ID to use for the API\n   */\n  readonly vpcEndpointId: string;\n}\n\n/**\n * Lambda VPC configuration\n */\nexport interface LambdaVpcConfig {\n  /**\n   * The VPC ID to deploy resources into\n   */\n  readonly vpcId: string;\n\n  /**\n   * Security group IDs for Lambda functions\n   */\n  readonly securityGroupIds: string[];\n\n  /**\n   * Private subnet IDs for Lambda functions\n   */\n  readonly subnetIds: string[];\n}\n\n/**\n * Global table configuration for multi-region deployments\n */\nexport interface GlobalTableConfig {\n  /**\n   * Whether this is the primary region that creates the global table\n   */\n  readonly isPrimaryRegion: boolean;\n\n  /**\n   * List of all regions that should have replicas\n   * Only used by the primary region\n   */\n  readonly replicaRegions?: string[];\n}\n\n/**\n * Internal interface for resolved VPC resources\n */\nexport interface ResolvedVpcConfig {\n  readonly vpc: IVpc;\n  readonly lambdaSecurityGroups: ISecurityGroup[];\n  readonly privateSubnets: ISubnet[];\n}\n\nexport interface FlowConfigStackProps extends cdk.StackProps {\n  /**\n   * Used for resource naming. Will also be the name of the Connect Lambda\n   * @example `cxbuilder-flow-config`\n   */\n  readonly prefix: string;\n  readonly cognito: CognitoConfig;\n  readonly connectInstanceArn: string;\n\n  /**\n   * Who to notify for unhandled exceptions\n   */\n  readonly alertEmails: string[];\n  readonly prod?: boolean;\n\n  /**\n   * If provided, the API will be deployed in a VPC.\n   */\n  readonly apiVpcConfig?: ApiVpcConfig;\n\n  /**\n   * If provided, the Lambda functions will be deployed in a VPC.\n   * Note: VPC should contain endpoints to: CloudFormation, Lambda, DynamoDB, SNS, and Polly.\n   */\n  readonly lambdaVpcConfig?: LambdaVpcConfig;\n\n  /**\n   * Global table configuration for multi-region deployments.\n   * If provided, enables global table support.\n   * If undefined, creates a single-region table.\n   */\n  readonly globalTable?: GlobalTableConfig;\n\n  /**\n   * Set to false to remove CXBuilder branding from the web app.\n   * @default true\n   */\n  readonly branding?: boolean;\n\n  /**\n   * Whether to associate the app with the Connect Agent Workspace.\n   * Set to false to disable automatic association.\n   * @default true\n   */\n  readonly associate3pApp?: boolean;\n}\n\nexport class FlowConfigStack extends cdk.Stack {\n  userPool: IUserPool;\n  private api: Api;\n  userPoolClient: UserPoolClient;\n  alertTopic: Topic;\n  table: cdk.aws_dynamodb.ITable;\n\n  /**\n   * Resolved VPC configuration if private deployment is enabled\n   */\n  private readonly _resolvedVpcConfig?: ResolvedVpcConfig;\n\n  /**\n   * Get resolved VPC configuration for child constructs\n   * @internal\n   */\n  public _getResolvedVpcConfig(): ResolvedVpcConfig | undefined {\n    return this._resolvedVpcConfig;\n  }\n\n  get appUrl(): string {\n    return this.api.url;\n  }\n\n  constructor(\n    scope: Construct,\n    id: string,\n    public props: FlowConfigStackProps\n  ) {\n    const {\n      prefix,\n      cognito,\n      alertEmails,\n      prod = false,\n      lambdaVpcConfig,\n      globalTable,\n    } = props;\n    super(scope, id, {\n      ...props,\n      stackName: props.stackName ?? prefix,\n      description:\n        props.description ??\n        'Web-based interface for managing flow variables and prompts that are dynamically retrieved by Amazon Connect during customer interactions',\n      terminationProtection: prod,\n    });\n\n    // Resolve VPC configuration if provided\n    this._resolvedVpcConfig = lambdaVpcConfig\n      ? this.resolveVpcConfig(lambdaVpcConfig)\n      : undefined;\n\n    // DynamoDB table for storing flow configs\n    this.table = this.createTable(prefix, prod, globalTable);\n\n    this.alertTopic = new Topic(this, 'AlertTopic', {\n      topicName: `${prefix}-error-alerts`,\n    });\n    alertEmails.forEach((e) =>\n      this.alertTopic.addSubscription(new EmailSubscription(e))\n    );\n\n    this.userPool = UserPool.fromUserPoolId(\n      this,\n      'UserPool',\n      cognito.userPoolId\n    );\n\n    const getConfig = new GetConfig(this);\n    this.api = new Api(this);\n\n    this.userPoolClient = this.createUserPoolClient();\n    this.createUserPoolGroups();\n\n    this.associate3pApp();\n\n    new cdk.CfnOutput(this, 'AppUrl', {\n      value: this.appUrl,\n      description: 'Base URL for the Flow Config application',\n    });\n    new cdk.CfnOutput(this, 'ApiUrl', {\n      value: `${this.appUrl}/api`,\n      description: 'Base URL for the Flow Config API',\n    });\n    new cdk.CfnOutput(this, 'GetConfigLambdaName', {\n      value: getConfig.function.functionName,\n      description:\n        'Lambda function name for accessing flow configs from Connect',\n    });\n  }\n\n  createUserPoolClient(): UserPoolClient {\n    const { prefix, cognito } = this.props;\n\n    const domains = ['http://localhost:3000', this.appUrl];\n\n    const oAuth = {\n      callbackUrls: domains.map((x) => `${x}/popup.html`),\n      logoutUrls: domains,\n    };\n    const client = new UserPoolClient(this, 'UserPoolClient', {\n      userPoolClientName: prefix,\n      userPool: this.userPool,\n      supportedIdentityProviders: cognito.ssoProviderName\n        ? [UserPoolClientIdentityProvider.custom(cognito.ssoProviderName)]\n        : [UserPoolClientIdentityProvider.COGNITO],\n      generateSecret: false,\n      oAuth,\n    });\n\n    new cdk.CfnOutput(this, 'UserPoolClientId', {\n      value: client.userPoolClientId,\n    });\n    return client;\n  }\n\n  /**\n   * Create Cognito User Groups for role-based access control\n   */\n  createUserPoolGroups(): void {\n    // FlowConfigAdmin - Full CRUD access\n    new CfnUserPoolGroup(this, 'FlowConfigAdminGroup', {\n      userPoolId: this.userPool.userPoolId,\n      groupName: 'FlowConfigAdmin',\n      description:\n        'Full administrative access to all flow configs - can create, read, update, and delete flow configs and all their properties',\n      precedence: 1,\n    });\n\n    // FlowConfigEdit - Edit variable/prompt values only\n    new CfnUserPoolGroup(this, 'FlowConfigEditGroup', {\n      userPoolId: this.userPool.userPoolId,\n      groupName: 'FlowConfigEdit',\n      description:\n        'Edit access to flow configs - can read and modify variable values and prompt content but cannot add/remove fields or delete configs',\n      precedence: 2,\n    });\n\n    // FlowConfigRead - Read-only access\n    new CfnUserPoolGroup(this, 'FlowConfigReadGroup', {\n      userPoolId: this.userPool.userPoolId,\n      groupName: 'FlowConfigRead',\n      description:\n        'Read-only access to all flow configs for reporting and reference purposes',\n      precedence: 3,\n    });\n  }\n\n  /**\n   * Associate FlowConfig as Agent Workspace app\n   */\n  associate3pApp() {\n    const {\n      props: { prefix, connectInstanceArn },\n      appUrl: url,\n      associate3pApp = true,\n    } = this;\n\n    if (!associate3pApp) {\n      return;\n    }\n\n    let namespace = `cxbuilder.${prefix}`;\n    if (namespace.length > 32 && !cdk.Token.isUnresolved(namespace)) {\n      namespace = namespace.substring(0, 32);\n    }\n\n    const app = new CfnApplication(this, '3pApp', {\n      name: this.props.prefix,\n      description:\n        'Agent Workspace app for configuring contact flow variables and prompts',\n      namespace,\n      applicationSourceConfig: {\n        externalUrlConfig: {\n          accessUrl: url,\n        },\n      },\n      // docs: https://docs.aws.amazon.com/connect/latest/adminguide/3p-apps-events-requests.html\n      permissions: [\n        'User.Details.View',\n        'User.Configuration.View',\n        'User.Status.View',\n        'Contact.Details.View',\n        'Contact.CustomerDetails.View',\n        'Contact.Variables.View',\n      ],\n    });\n\n    const association = new CfnIntegrationAssociation(\n      this,\n      '3pAppAssociation',\n      {\n        instanceId: connectInstanceArn,\n        integrationType: 'APPLICATION',\n        integrationArn: app.attrApplicationArn,\n      }\n    );\n\n    const provider = new SecurityProfileProvider(\n      this,\n      'SecurityProfileProvider',\n      {\n        connectInstanceArn,\n      }\n    );\n    provider.node.addDependency(association);\n\n    // Automatically associate with security profiles\n    const securityProfiles = ['Admin'];\n    for (const profile of securityProfiles) {\n      provider.allowApplication(`Allow${profile}`, profile, namespace);\n    }\n  }\n\n  /**\n   * Create DynamoDB table with optional global table support\n   */\n  private createTable(\n    prefix: string,\n    prod: boolean,\n    globalTable?: GlobalTableConfig\n  ): dynamodb.ITable {\n    const tableName = prefix;\n\n    if (!globalTable) {\n      // Single-region table\n      return new dynamodb.TableV2(this, 'Table', {\n        tableName,\n        partitionKey: {\n          name: 'id',\n          type: dynamodb.AttributeType.STRING,\n        },\n        billing: dynamodb.Billing.onDemand(),\n        pointInTimeRecoverySpecification: {\n          pointInTimeRecoveryEnabled: true,\n        },\n        deletionProtection: prod,\n        removalPolicy: prod\n          ? cdk.RemovalPolicy.RETAIN\n          : cdk.RemovalPolicy.DESTROY,\n      });\n    }\n\n    if (globalTable.isPrimaryRegion) {\n      // Primary region creates global table with replicas\n      const table = new dynamodb.TableV2(this, 'Table', {\n        tableName,\n        partitionKey: {\n          name: 'id',\n          type: dynamodb.AttributeType.STRING,\n        },\n        billing: dynamodb.Billing.onDemand(),\n        pointInTimeRecoverySpecification: {\n          pointInTimeRecoveryEnabled: true,\n        },\n        deletionProtection: prod,\n        removalPolicy: prod\n          ? cdk.RemovalPolicy.RETAIN\n          : cdk.RemovalPolicy.DESTROY,\n      });\n      if (globalTable.replicaRegions) {\n        for (const region of globalTable.replicaRegions) {\n          table.addReplica({\n            region,\n            deletionProtection: prod,\n            pointInTimeRecoverySpecification: {\n              pointInTimeRecoveryEnabled: true,\n            },\n          });\n        }\n      }\n      return table;\n    } else {\n      // Secondary region references existing global table\n      return dynamodb.Table.fromTableArn(\n        this,\n        'Table',\n        `arn:aws:dynamodb:${this.region}:${this.account}:table/${tableName}`\n      );\n    }\n  }\n\n  /**\n   * Resolve VPC configuration string IDs to CDK objects\n   */\n  private resolveVpcConfig(vpcConfig: LambdaVpcConfig): ResolvedVpcConfig {\n    const { vpcId, securityGroupIds, subnetIds } = vpcConfig;\n\n    // Resolve VPC\n    const vpc = ec2.Vpc.fromLookup(this, 'ResolvedVpc', { vpcId });\n\n    // Resolve Lambda security groups\n    const lambdaSecurityGroups = securityGroupIds.map((sgId, index) =>\n      ec2.SecurityGroup.fromSecurityGroupId(\n        this,\n        `LambdaSecurityGroup${index}`,\n        sgId\n      )\n    );\n\n    // Resolve private subnets\n    const privateSubnets = subnetIds.map((subnetId, index) =>\n      ec2.Subnet.fromSubnetId(this, `PrivateSubnet${index}`, subnetId)\n    );\n\n    return {\n      vpc,\n      lambdaSecurityGroups,\n      privateSubnets,\n    };\n  }\n}\n"]}
|
|
@@ -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:
|
|
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,
|
|
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
|
-
|
|
23
|
-
|
|
24
|
-
|
|
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:
|
|
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(
|
|
37
|
+
policy: this.createVpcEndpointPolicy(apiVpcEndpoint.vpcId),
|
|
39
38
|
}
|
|
40
39
|
: {
|
|
41
|
-
restApiName:
|
|
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"]}
|
|
@@ -1,3 +1,3 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
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,
|
|
36
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9pbmZyYXN0cnVjdHVyZS9hcGkvSW5pdC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSwyQ0FBdUM7QUFJdkMsNkNBQThDO0FBQzlDLGlEQUE4RDtBQUM5RCxxREFBa0Q7QUFFbEQ7O0dBRUc7QUFDSCxNQUFhLElBQUssU0FBUSxzQkFBUztJQUVqQyxZQUFZLEdBQVE7UUFDbEIsS0FBSyxDQUFDLEdBQUcsRUFBRSxNQUFNLENBQUMsQ0FBQztRQUVuQixNQUFNLEVBQUUsU0FBUyxFQUFFLE1BQU0sRUFBRSxPQUFPLEVBQUUsR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUV0RCxJQUFJLENBQUMsTUFBTSxHQUFHLElBQUEsMkJBQVksRUFBVSxJQUFJLEVBQUUsU0FBUyxFQUFFO1lBQ25ELFdBQVcsRUFBRTtnQkFDWCxTQUFTLEVBQUUsU0FBUztnQkFDcEIsVUFBVSxFQUFFLEdBQUcsQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxVQUFVO2dCQUM5QyxRQUFRLEVBQUUsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxRQUFRLElBQUksSUFBSSxDQUFDLENBQUMsUUFBUSxFQUFFO2FBQ3hEO1lBQ0QsT0FBTyxFQUFFLHNCQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUM3QixhQUFhLEVBQUU7Z0JBQ2IsSUFBSSx5QkFBZSxDQUFDO29CQUNsQixNQUFNLEVBQUUsZ0JBQU0sQ0FBQyxLQUFLO29CQUNwQixPQUFPLEVBQUUsQ0FBQywrQkFBK0IsQ0FBQztvQkFDMUMsU0FBUyxFQUFFO3dCQUNULDBCQUEwQixNQUFNLElBQUksT0FBTyxVQUFVLFNBQVMsSUFBSTtxQkFDbkU7aUJBQ0YsQ0FBQzthQUNIO1lBQ0QsVUFBVSxFQUFFLEdBQUcsQ0FBQyxLQUFLLENBQUMsVUFBVTtTQUNqQyxDQUFDLENBQUM7SUFDTCxDQUFDO0NBQ0Y7QUExQkQsb0JBMEJDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5pbXBvcnQgeyBGdW5jdGlvbiB9IGZyb20gJ2F3cy1jZGstbGliL2F3cy1sYW1iZGEnO1xuaW1wb3J0IHsgSW5pdEVudiB9IGZyb20gJy4vSW5pdC5pbnRlcmZhY2UnO1xuaW1wb3J0IHsgQXBpIH0gZnJvbSAnLi4vQXBpJztcbmltcG9ydCB7IER1cmF0aW9uLCBTdGFjayB9IGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCB7IEVmZmVjdCwgUG9saWN5U3RhdGVtZW50IH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWlhbSc7XG5pbXBvcnQgeyBjcmVhdGVMYW1iZGEgfSBmcm9tICcuLi8uLi9jcmVhdGVMYW1iZGEnO1xuXG4vKipcbiAqIFByb3ZpZGVzIHJ1bi10aW1lIGNvbmZpZ3VyYXRpb24gZm9yIHRoZSBBUEkgc3VjaCBhcyBBdXRoLlxuICovXG5leHBvcnQgY2xhc3MgSW5pdCBleHRlbmRzIENvbnN0cnVjdCB7XG4gIHJlYWRvbmx5IGxhbWJkYTogRnVuY3Rpb247XG4gIGNvbnN0cnVjdG9yKGFwaTogQXBpKSB7XG4gICAgc3VwZXIoYXBpLCAnSW5pdCcpO1xuXG4gICAgY29uc3QgeyBzdGFja05hbWUsIHJlZ2lvbiwgYWNjb3VudCB9ID0gU3RhY2sub2YodGhpcyk7XG5cbiAgICB0aGlzLmxhbWJkYSA9IGNyZWF0ZUxhbWJkYTxJbml0RW52Pih0aGlzLCAnSGFuZGxlcicsIHtcbiAgICAgIGVudmlyb25tZW50OiB7XG4gICAgICAgIHN0YWNrTmFtZTogc3RhY2tOYW1lLFxuICAgICAgICB1c2VyUG9vbElkOiBhcGkuc3RhY2sucHJvcHMuY29nbml0by51c2VyUG9vbElkLFxuICAgICAgICBicmFuZGluZzogKGFwaS5zdGFjay5wcm9wcy5icmFuZGluZyA/PyB0cnVlKS50b1N0cmluZygpLFxuICAgICAgfSxcbiAgICAgIHRpbWVvdXQ6IER1cmF0aW9uLnNlY29uZHMoMTUpLFxuICAgICAgaW5pdGlhbFBvbGljeTogW1xuICAgICAgICBuZXcgUG9saWN5U3RhdGVtZW50KHtcbiAgICAgICAgICBlZmZlY3Q6IEVmZmVjdC5BTExPVyxcbiAgICAgICAgICBhY3Rpb25zOiBbJ2Nsb3VkZm9ybWF0aW9uOkRlc2NyaWJlU3RhY2tzJ10sXG4gICAgICAgICAgcmVzb3VyY2VzOiBbXG4gICAgICAgICAgICBgYXJuOmF3czpjbG91ZGZvcm1hdGlvbjoke3JlZ2lvbn06JHthY2NvdW50fTpzdGFjay8ke3N0YWNrTmFtZX0vKmAsXG4gICAgICAgICAgXSxcbiAgICAgICAgfSksXG4gICAgICBdLFxuICAgICAgYWxlcnRUb3BpYzogYXBpLnN0YWNrLmFsZXJ0VG9waWMsXG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiU2V0dGluZ3MuaW50ZXJmYWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vaW5mcmFzdHJ1Y3R1cmUvYXBpL1NldHRpbmdzL1NldHRpbmdzLmludGVyZmFjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBTZXR0aW5nc0VudiB7XG4gIFRBQkxFX05BTUU6IHN0cmluZztcbn1cbiJdfQ==
|
|
@@ -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=
|