@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.
Files changed (33) hide show
  1. package/.jsii +151 -66
  2. package/CHANGELOG.md +24 -0
  3. package/README.md +8 -5
  4. package/dist/backend/FlowConfig/index.js +2 -1
  5. package/dist/backend/FlowConfig/index.js.map +2 -2
  6. package/dist/backend/GetConfig/index.js +1 -1
  7. package/dist/backend/GetConfig/index.js.map +2 -2
  8. package/dist/backend/Init/index.js +2 -1
  9. package/dist/backend/Init/index.js.map +2 -2
  10. package/dist/backend/Settings/index.js +255 -0
  11. package/dist/backend/Settings/index.js.map +7 -0
  12. package/dist/backend/Static/static/assets/index-Cejunttu.js +61 -0
  13. package/dist/backend/Static/static/assets/{index-NRh8x3FI.css → index-SZuscj14.css} +1 -1
  14. package/dist/backend/Static/static/index.html +3 -3
  15. package/dist/infrastructure/FlowConfigStack.d.ts +39 -15
  16. package/dist/infrastructure/FlowConfigStack.js +32 -20
  17. package/dist/infrastructure/GetConfig/index.js +2 -2
  18. package/dist/infrastructure/api/Api.d.ts +5 -2
  19. package/dist/infrastructure/api/Api.js +21 -15
  20. package/dist/infrastructure/api/Init/Init.interface.d.ts +4 -0
  21. package/dist/infrastructure/api/Init/Init.interface.js +1 -1
  22. package/dist/infrastructure/api/Init/index.js +2 -1
  23. package/dist/infrastructure/api/Settings/Settings.interface.d.ts +3 -0
  24. package/dist/infrastructure/api/Settings/Settings.interface.js +3 -0
  25. package/dist/infrastructure/api/Settings/index.d.ts +7 -0
  26. package/dist/infrastructure/api/Settings/index.js +21 -0
  27. package/dist/infrastructure/api/spec.yaml +122 -0
  28. package/dist/infrastructure/createLambda.js +1 -1
  29. package/dist/infrastructure/index.d.ts +1 -1
  30. package/dist/infrastructure/index.js +1 -1
  31. package/dist/infrastructure/tsconfig.tsbuildinfo +1 -1
  32. package/package.json +1 -1
  33. package/dist/backend/Static/static/assets/index-FHwnAA8f.js +0 -61
@@ -23,6 +23,7 @@ components:
23
23
  - region
24
24
  - userPoolId
25
25
  - clientId
26
+ - branding
26
27
  properties:
27
28
  region:
28
29
  type: string
@@ -30,6 +31,8 @@ components:
30
31
  type: string
31
32
  clientId:
32
33
  type: string
34
+ branding:
35
+ type: boolean
33
36
  User:
34
37
  type: object
35
38
  description: Describe the Connect user
@@ -189,6 +192,45 @@ components:
189
192
  type: string
190
193
  description: Channel type for the prompt
191
194
  enum: [voice, chat]
195
+ Locale:
196
+ type: object
197
+ description: Locale configuration with available voices
198
+ required:
199
+ - code
200
+ - name
201
+ - voices
202
+ properties:
203
+ code:
204
+ type: string
205
+ description: Amazon Polly language code (e.g., en-US, arb, cmn-CN)
206
+ pattern: '^([a-z]{2,3}(-[A-Z]{2})?(-[A-Z]{3})?|arb)$'
207
+ example: en-US
208
+ name:
209
+ type: string
210
+ description: Human-readable display name for the locale
211
+ example: English (United States)
212
+ maxLength: 100
213
+ voices:
214
+ type: array
215
+ description: Array of Amazon Polly voice IDs available for this locale
216
+ items:
217
+ type: string
218
+ description: Amazon Polly voice ID
219
+ example: Joanna
220
+ maxLength: 50
221
+ maxItems: 50
222
+ Settings:
223
+ type: object
224
+ description: Application settings including locale and voice configurations
225
+ required:
226
+ - locales
227
+ properties:
228
+ locales:
229
+ type: array
230
+ description: Array of configured locales with their available voices
231
+ items:
232
+ $ref: '#/components/schemas/Locale'
233
+ maxItems: 100
192
234
  paths:
193
235
  /api/init:
194
236
  get:
@@ -456,6 +498,86 @@ paths:
456
498
  application/json:
457
499
  schema:
458
500
  $ref: '#/components/schemas/Error'
501
+ /api/settings:
502
+ get:
503
+ summary: Get application settings
504
+ description: Returns the current application settings including configured locales and voices
505
+ x-amazon-apigateway-integration:
506
+ httpMethod: 'POST'
507
+ uri: SettingsHandler
508
+ passthroughBehavior: 'when_no_match'
509
+ type: 'aws_proxy'
510
+ security:
511
+ - CognitoAuthorizer: []
512
+ operationId: getSettings
513
+ responses:
514
+ '200':
515
+ description: Successful operation
516
+ content:
517
+ application/json:
518
+ schema:
519
+ $ref: '#/components/schemas/Settings'
520
+ '401':
521
+ description: Unauthorized
522
+ content:
523
+ application/json:
524
+ schema:
525
+ $ref: '#/components/schemas/Error'
526
+ '403':
527
+ description: Forbidden - Admin access required
528
+ content:
529
+ application/json:
530
+ schema:
531
+ $ref: '#/components/schemas/Error'
532
+ post:
533
+ summary: Update application settings
534
+ description: Updates the application settings including configured locales and voices. Admin access required.
535
+ x-amazon-apigateway-integration:
536
+ httpMethod: 'POST'
537
+ uri: SettingsHandler
538
+ passthroughBehavior: 'when_no_match'
539
+ type: 'aws_proxy'
540
+ security:
541
+ - CognitoAuthorizer: []
542
+ operationId: updateSettings
543
+ requestBody:
544
+ description: Settings object
545
+ required: true
546
+ content:
547
+ application/json:
548
+ schema:
549
+ $ref: '#/components/schemas/Settings'
550
+ responses:
551
+ '200':
552
+ description: Settings updated successfully
553
+ content:
554
+ application/json:
555
+ schema:
556
+ $ref: '#/components/schemas/Settings'
557
+ '400':
558
+ description: Invalid input
559
+ content:
560
+ application/json:
561
+ schema:
562
+ $ref: '#/components/schemas/Error'
563
+ '401':
564
+ description: Unauthorized
565
+ content:
566
+ application/json:
567
+ schema:
568
+ $ref: '#/components/schemas/Error'
569
+ '403':
570
+ description: Forbidden - Admin access required
571
+ content:
572
+ application/json:
573
+ schema:
574
+ $ref: '#/components/schemas/Error'
575
+ '413':
576
+ description: Payload too large
577
+ content:
578
+ application/json:
579
+ schema:
580
+ $ref: '#/components/schemas/Error'
459
581
  /api/preview-speech:
460
582
  post:
461
583
  summary: Preview speech using Amazon Polly
@@ -82,4 +82,4 @@ function createLambda(scope, id, props) {
82
82
  });
83
83
  return func;
84
84
  }
85
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createLambda.js","sourceRoot":"","sources":["../../infrastructure/createLambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, RemovalPolicy, Stack } from 'aws-cdk-lib';\nimport { ManagedPolicy } from 'aws-cdk-lib/aws-iam';\nimport {\n  Architecture,\n  Runtime,\n  Tracing,\n  Function,\n  FunctionProps,\n  Code,\n  LayerVersion,\n} from 'aws-cdk-lib/aws-lambda';\nimport { LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { ITopic } from 'aws-cdk-lib/aws-sns';\nimport { Construct } from 'constructs';\nimport { existsSync } from 'fs';\nimport { resolve } from 'path';\nimport { FlowConfigStack } from './FlowConfigStack';\n\ninterface CreateLambdaProps<TEnv>\n  extends Omit<FunctionProps, 'handler' | 'environment' | 'code' | 'runtime'> {\n  environment?: TEnv;\n\n  /**\n   * Notify someone when your lambda fails!\n   */\n  alertTopic?: ITopic;\n}\n/**\n * Utility function for creating a log group and lambda with app defaults.\n * Enforces environment variables via generic parameter TEnv\n */\nexport function createLambda<TEnv>(\n  scope: Construct,\n  id: string,\n  props: CreateLambdaProps<TEnv>\n) {\n  // Determine the code location based on logical ID\n  // If id is \"Handler\", use the parent construct's ID instead\n  const backendId = id === 'Handler' ? scope.node.id : id;\n  let codeLocation = resolve(__dirname, '..', 'backend', backendId);\n  let expectedIndexFile = resolve(codeLocation, 'index.js');\n\n  // Verify the bundled Lambda code exists, try ../dist/backend if not found\n  if (!existsSync(expectedIndexFile)) {\n    // Try alternative location in ../dist/backend\n    codeLocation = resolve(__dirname, '..', 'dist', 'backend', backendId);\n    expectedIndexFile = resolve(codeLocation, 'index.js');\n    \n    if (!existsSync(expectedIndexFile)) {\n      throw new Error(\n        `Lambda function ${backendId} bundled code not found at: ${expectedIndexFile} or ${resolve(__dirname, '..', 'backend', backendId, 'index.js')}. ` +\n          'Please run \"npm run build:lambdas\" to bundle the Lambda functions.'\n      );\n    }\n  }\n\n  console.log(\n    `📦 Using Lambda code for ${backendId} (id=${id}) from: ${codeLocation}`\n  );\n\n  // Get the current region for the Lambda Powertools layer\n  const region = Stack.of(scope).region;\n\n  // AWS Lambda Powertools layer ARN\n  // See: https://docs.powertools.aws.dev/lambda/typescript/latest/#lambda-layer\n  const powertoolsLayerArn = `arn:aws:lambda:${region}:094274105915:layer:AWSLambdaPowertoolsTypeScriptV2:22`;\n\n  // Check if we're in a FlowConfigStack and get VPC configuration\n  const stack = Stack.of(scope) as FlowConfigStack;\n  const vpcConfig = stack._getResolvedVpcConfig();\n\n  const func = new Function(scope, `${id}Function`, {\n    // Apply defaults\n    handler: 'index.handler',\n    runtime: Runtime.NODEJS_22_X,\n    architecture: Architecture.ARM_64,\n    tracing: Tracing.ACTIVE,\n    timeout: Duration.seconds(30),\n    code: Code.fromAsset(codeLocation),\n    layers: [\n      LayerVersion.fromLayerVersionArn(\n        scope,\n        `${id}PowertoolsLayer`,\n        powertoolsLayerArn\n      ),\n    ],\n    // VPC configuration if provided\n    ...(vpcConfig && {\n      vpc: vpcConfig.vpc,\n      vpcSubnets: { subnets: vpcConfig.privateSubnets },\n      securityGroups: vpcConfig.lambdaSecurityGroups,\n    }),\n    // Apply overrides\n    ...props,\n    // Ensure that the log retention custom resource is not created\n    logRetention: undefined,\n    environment: {\n      // see https://docs.aws.amazon.com/lambda/latest/dg/typescript-exceptions.html\n      NODE_OPTIONS: '--enable-source-maps',\n      // Log level. Do not emit DEBUG in prod environment because it will bloat your CW costs\n      POWERTOOLS_LOG_LEVEL: 'DEBUG',\n      // Log incoming event\n      POWERTOOLS_LOGGER_LOG_EVENT: 'true',\n      ALERT_TOPIC_ARN: props.alertTopic?.topicArn ?? 'N/A',\n      ...props.environment,\n    },\n  });\n\n  props.alertTopic?.grantPublish(func);\n\n  // Note: we are using this policy instead of log.grantWrite(func) because the func.functionName a circular dependency.\n  // To constrain LogGroup permissions further, you can use a static functionName with log.grantWrite\n  func.role?.addManagedPolicy(\n    ManagedPolicy.fromAwsManagedPolicyName(\n      vpcConfig\n        ? 'service-role/AWSLambdaVPCAccessExecutionRole'\n        : 'service-role/AWSLambdaBasicExecutionRole'\n    )\n  );\n\n  // Create a self-managed log group\n  new LogGroup(scope, `${id}LogGroup`, {\n    retention: props.logRetention ?? RetentionDays.ONE_MONTH,\n    logGroupName: `/aws/lambda/${func.functionName}`,\n    removalPolicy: RemovalPolicy.DESTROY,\n  });\n  return func;\n}\n"]}
85
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createLambda.js","sourceRoot":"","sources":["../../infrastructure/createLambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, RemovalPolicy, Stack } from 'aws-cdk-lib';\nimport { ManagedPolicy } from 'aws-cdk-lib/aws-iam';\nimport {\n  Architecture,\n  Runtime,\n  Tracing,\n  Function,\n  FunctionProps,\n  Code,\n  LayerVersion,\n} from 'aws-cdk-lib/aws-lambda';\nimport { LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { ITopic } from 'aws-cdk-lib/aws-sns';\nimport { Construct } from 'constructs';\nimport { existsSync } from 'fs';\nimport { resolve } from 'path';\nimport { FlowConfigStack } from './FlowConfigStack';\n\ninterface CreateLambdaProps<TEnv>\n  extends Omit<FunctionProps, 'handler' | 'environment' | 'code' | 'runtime'> {\n  environment?: TEnv;\n\n  /**\n   * Notify someone when your lambda fails!\n   */\n  alertTopic?: ITopic;\n}\n/**\n * Utility function for creating a log group and lambda with app defaults.\n * Enforces environment variables via generic parameter TEnv\n */\nexport function createLambda<TEnv>(\n  scope: Construct,\n  id: string,\n  props: CreateLambdaProps<TEnv>\n) {\n  // Determine the code location based on logical ID\n  // If id is \"Handler\", use the parent construct's ID instead\n  const backendId = id === 'Handler' ? scope.node.id : id;\n  let codeLocation = resolve(__dirname, '..', 'backend', backendId);\n  let expectedIndexFile = resolve(codeLocation, 'index.js');\n\n  // Verify the bundled Lambda code exists, try ../dist/backend if not found\n  if (!existsSync(expectedIndexFile)) {\n    // Try alternative location in ../dist/backend\n    codeLocation = resolve(__dirname, '..', 'dist', 'backend', backendId);\n    expectedIndexFile = resolve(codeLocation, 'index.js');\n\n    if (!existsSync(expectedIndexFile)) {\n      throw new Error(\n        `Lambda function ${backendId} bundled code not found at: ${expectedIndexFile} or ${resolve(\n          __dirname,\n          '..',\n          'backend',\n          backendId,\n          'index.js'\n        )}. ` +\n          'Please run \"npm run build:lambdas\" to bundle the Lambda functions.'\n      );\n    }\n  }\n\n  console.log(\n    `📦 Using Lambda code for ${backendId} (id=${id}) from: ${codeLocation}`\n  );\n\n  // Get the current region for the Lambda Powertools layer\n  const region = Stack.of(scope).region;\n\n  // AWS Lambda Powertools layer ARN\n  // See: https://docs.powertools.aws.dev/lambda/typescript/latest/#lambda-layer\n  const powertoolsLayerArn = `arn:aws:lambda:${region}:094274105915:layer:AWSLambdaPowertoolsTypeScriptV2:22`;\n\n  // Check if we're in a FlowConfigStack and get VPC configuration\n  const stack = Stack.of(scope) as FlowConfigStack;\n  const vpcConfig = stack._getResolvedVpcConfig();\n\n  const func = new Function(scope, `${id}Function`, {\n    // Apply defaults\n    handler: 'index.handler',\n    runtime: Runtime.NODEJS_22_X,\n    architecture: Architecture.ARM_64,\n    tracing: Tracing.ACTIVE,\n    timeout: Duration.seconds(30),\n    code: Code.fromAsset(codeLocation),\n    layers: [\n      LayerVersion.fromLayerVersionArn(\n        scope,\n        `${id}PowertoolsLayer`,\n        powertoolsLayerArn\n      ),\n    ],\n    // VPC configuration if provided\n    ...(vpcConfig && {\n      vpc: vpcConfig.vpc,\n      vpcSubnets: { subnets: vpcConfig.privateSubnets },\n      securityGroups: vpcConfig.lambdaSecurityGroups,\n    }),\n    // Apply overrides\n    ...props,\n    // Ensure that the log retention custom resource is not created\n    logRetention: undefined,\n    environment: {\n      // see https://docs.aws.amazon.com/lambda/latest/dg/typescript-exceptions.html\n      NODE_OPTIONS: '--enable-source-maps',\n      // Log level. Do not emit DEBUG in prod environment because it will bloat your CW costs\n      POWERTOOLS_LOG_LEVEL: 'DEBUG',\n      // Log incoming event\n      POWERTOOLS_LOGGER_LOG_EVENT: 'true',\n      ALERT_TOPIC_ARN: props.alertTopic?.topicArn ?? 'N/A',\n      ...props.environment,\n    },\n  });\n\n  props.alertTopic?.grantPublish(func);\n\n  // Note: we are using this policy instead of log.grantWrite(func) because the func.functionName a circular dependency.\n  // To constrain LogGroup permissions further, you can use a static functionName with log.grantWrite\n  func.role?.addManagedPolicy(\n    ManagedPolicy.fromAwsManagedPolicyName(\n      vpcConfig\n        ? 'service-role/AWSLambdaVPCAccessExecutionRole'\n        : 'service-role/AWSLambdaBasicExecutionRole'\n    )\n  );\n\n  // Create a self-managed log group\n  new LogGroup(scope, `${id}LogGroup`, {\n    retention: props.logRetention ?? RetentionDays.ONE_MONTH,\n    logGroupName: `/aws/lambda/${func.functionName}`,\n    removalPolicy: RemovalPolicy.DESTROY,\n  });\n  return func;\n}\n"]}
@@ -1 +1 @@
1
- export { FlowConfigStack, FlowConfigStackProps, CognitoConfig, VpcConfig, GlobalTableConfig, } from './FlowConfigStack';
1
+ export { FlowConfigStack, FlowConfigStackProps, CognitoConfig, ApiVpcConfig, LambdaVpcConfig, GlobalTableConfig, } from './FlowConfigStack';
@@ -4,4 +4,4 @@ exports.FlowConfigStack = void 0;
4
4
  // Export the main infrastructure stack and its props interface
5
5
  var FlowConfigStack_1 = require("./FlowConfigStack");
6
6
  Object.defineProperty(exports, "FlowConfigStack", { enumerable: true, get: function () { return FlowConfigStack_1.FlowConfigStack; } });
7
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9pbmZyYXN0cnVjdHVyZS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSwrREFBK0Q7QUFDL0QscURBTTJCO0FBTHpCLGtIQUFBLGVBQWUsT0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8vIEV4cG9ydCB0aGUgbWFpbiBpbmZyYXN0cnVjdHVyZSBzdGFjayBhbmQgaXRzIHByb3BzIGludGVyZmFjZVxuZXhwb3J0IHtcbiAgRmxvd0NvbmZpZ1N0YWNrLFxuICBGbG93Q29uZmlnU3RhY2tQcm9wcyxcbiAgQ29nbml0b0NvbmZpZyxcbiAgVnBjQ29uZmlnLFxuICBHbG9iYWxUYWJsZUNvbmZpZyxcbn0gZnJvbSAnLi9GbG93Q29uZmlnU3RhY2snO1xuIl19
7
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9pbmZyYXN0cnVjdHVyZS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSwrREFBK0Q7QUFDL0QscURBTzJCO0FBTnpCLGtIQUFBLGVBQWUsT0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8vIEV4cG9ydCB0aGUgbWFpbiBpbmZyYXN0cnVjdHVyZSBzdGFjayBhbmQgaXRzIHByb3BzIGludGVyZmFjZVxuZXhwb3J0IHtcbiAgRmxvd0NvbmZpZ1N0YWNrLFxuICBGbG93Q29uZmlnU3RhY2tQcm9wcyxcbiAgQ29nbml0b0NvbmZpZyxcbiAgQXBpVnBjQ29uZmlnLFxuICBMYW1iZGFWcGNDb25maWcsXG4gIEdsb2JhbFRhYmxlQ29uZmlnLFxufSBmcm9tICcuL0Zsb3dDb25maWdTdGFjayc7XG4iXX0=