@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.
- package/.jsii +145 -68
- package/CHANGELOG.md +38 -0
- package/README.md +8 -5
- package/dist/backend/FlowConfig/index.js +111 -16
- package/dist/backend/FlowConfig/index.js.map +3 -3
- 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 +36 -14
- package/dist/infrastructure/FlowConfigStack.js +54 -18
- 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/docs/Permissions-v1.md +132 -0
- package/docs/{Permissions.md → Permissions-v2.md} +15 -15
- package/package.json +1 -1
- 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>
|
|
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,25 @@ 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;
|
|
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
|
-
|
|
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', {
|
|
@@ -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.
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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, '
|
|
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,
|
|
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 =
|
|
223
|
+
const lambdaSecurityGroups = securityGroupIds.map((sgId, index) => ec2.SecurityGroup.fromSecurityGroupId(this, `LambdaSecurityGroup${index}`, sgId));
|
|
185
224
|
// Resolve private subnets
|
|
186
|
-
const privateSubnets =
|
|
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: "
|
|
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:
|
|
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==
|