@btc-embedded/cdk-extensions 0.22.22 → 0.22.24

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 (64) hide show
  1. package/.jsii +905 -207
  2. package/API.md +777 -52
  3. package/CHANGELOG.md +16 -0
  4. package/assets/cli/catnip.js +1 -1
  5. package/lib/constructs/EventPipe.js +1 -1
  6. package/lib/constructs/ExportedService.js +1 -1
  7. package/lib/constructs/S3Bucket.js +1 -1
  8. package/lib/constructs/SecureRestApi.js +1 -1
  9. package/lib/constructs/SecureRestApiV2.js +1 -1
  10. package/lib/constructs/api-keys/ApiKeyClientAuthorization.js +1 -1
  11. package/lib/constructs/api-keys/ApiKeyManagement.js +1 -1
  12. package/lib/constructs/api-keys/ApiKeyPreTokenHandler.js +1 -1
  13. package/lib/constructs/api-keys/ApiKeyStore.js +1 -1
  14. package/lib/extensions/ApiGatewayExtension.d.ts +17 -0
  15. package/lib/extensions/ApiGatewayExtension.js +31 -32
  16. package/lib/extensions/ApplicationContainer.js +1 -1
  17. package/lib/extensions/ApplicationLoadBalancerExtension.d.ts +16 -0
  18. package/lib/extensions/ApplicationLoadBalancerExtension.js +28 -28
  19. package/lib/extensions/ApplicationLoadBalancerExtensionV2.d.ts +27 -7
  20. package/lib/extensions/ApplicationLoadBalancerExtensionV2.js +16 -7
  21. package/lib/extensions/CloudMapExtension.d.ts +15 -0
  22. package/lib/extensions/CloudMapExtension.js +7 -3
  23. package/lib/extensions/DeactivatableServiceExtension.js +1 -1
  24. package/lib/extensions/DeploymentConfigExtension.js +1 -1
  25. package/lib/extensions/DocumentDbAccessExtension.d.ts +14 -2
  26. package/lib/extensions/DocumentDbAccessExtension.js +9 -3
  27. package/lib/extensions/DomainEventMessagingExtension.js +1 -1
  28. package/lib/extensions/EfsMountExtension.js +8 -2
  29. package/lib/extensions/ExtraContainerExtension.js +1 -1
  30. package/lib/extensions/HTTPApiExtension.d.ts +30 -1
  31. package/lib/extensions/HTTPApiExtension.js +14 -7
  32. package/lib/extensions/LogExtension.js +1 -1
  33. package/lib/extensions/ModifyContainerDefinitionExtension.js +1 -1
  34. package/lib/extensions/ModifyTaskDefinitionExtension.js +1 -1
  35. package/lib/extensions/OpenIdExtension.d.ts +14 -0
  36. package/lib/extensions/OpenIdExtension.js +9 -3
  37. package/lib/extensions/OpenTelemetryExtension.js +1 -1
  38. package/lib/extensions/PostgresDbAccessExtension.d.ts +30 -7
  39. package/lib/extensions/PostgresDbAccessExtension.js +24 -29
  40. package/lib/extensions/SharedVolumeExtension.js +1 -1
  41. package/lib/extensions/TcpKeepAliveExtension.js +1 -1
  42. package/lib/platform/ApiGateway.d.ts +29 -1
  43. package/lib/platform/ApiGateway.js +31 -2
  44. package/lib/platform/ApiGatewayVpcLink.js +2 -2
  45. package/lib/platform/ApplicationLoadBalancer.js +1 -1
  46. package/lib/platform/ApplicationLoadBalancerV2.js +2 -2
  47. package/lib/platform/AuroraPostgresDB.d.ts +63 -3
  48. package/lib/platform/AuroraPostgresDB.js +45 -7
  49. package/lib/platform/BTCLogGroup.js +1 -1
  50. package/lib/platform/CognitoUserPool.js +2 -2
  51. package/lib/platform/DefaultUserPoolClients.js +1 -1
  52. package/lib/platform/DocumentDB.d.ts +10 -2
  53. package/lib/platform/DocumentDB.js +12 -7
  54. package/lib/platform/EcsCluster.js +1 -1
  55. package/lib/platform/EfsFileSystem.js +5 -15
  56. package/lib/platform/HostedZone.js +1 -1
  57. package/lib/platform/PrivateDnsNamespace.js +1 -1
  58. package/lib/platform/ResourceServer.js +1 -1
  59. package/lib/platform/Vpc.js +1 -1
  60. package/lib/platform/VpcV2.js +1 -1
  61. package/lib/stacks/ApplicationStack.js +1 -1
  62. package/lib/utils/BasePlatformStackResolver.js +1 -1
  63. package/lib/utils/StackParameter.js +1 -1
  64. package/package.json +1 -1
@@ -24,30 +24,6 @@ class ApiGatewayExtension extends ecs_service_extensions_1.ServiceExtension {
24
24
  this.props = props;
25
25
  }
26
26
  useService(service) {
27
- const vpcLinkIdParameter = new StackParameter_1.StackParameter({
28
- prefix: "api-gateway",
29
- key: "vpc-link-id",
30
- });
31
- const apiGatewayIdParameter = new StackParameter_1.StackParameter({
32
- prefix: "api-gateway",
33
- key: "id",
34
- });
35
- const cognitoFullAccessClientParameter = new StackParameter_1.StackParameter({
36
- prefix: "cognito",
37
- key: "full-access-client-id",
38
- });
39
- const cognitoUserPoolClientParameter = new StackParameter_1.StackParameter({
40
- prefix: "cognito",
41
- key: "user-pool-client-id",
42
- });
43
- const cognitoUserPoolIdParameter = new StackParameter_1.StackParameter({
44
- prefix: "cognito",
45
- key: "user-pool-id",
46
- });
47
- const fqdnParameter = new StackParameter_1.StackParameter({
48
- prefix: "fqdn",
49
- key: "fqdn",
50
- });
51
27
  if (!service.cloudMapService) {
52
28
  throw new Error("This extension requires the CloudMap extension to be added");
53
29
  }
@@ -55,21 +31,42 @@ class ApiGatewayExtension extends ecs_service_extensions_1.ServiceExtension {
55
31
  if (!container || !container.trafficPort) {
56
32
  throw new Error("Cannot apply API Gateway Extension, no main app container found");
57
33
  }
34
+ // VPC link ID and API ID: use direct values if provided, else import from base platform
35
+ const vpcLinkId = this.props.apiGateway?.vpcLinkId ??
36
+ new StackParameter_1.StackParameter({ prefix: "api-gateway", key: "vpc-link-id" }).import(service, this.props.basePlatformStackName);
37
+ const apiGatewayId = this.props.apiGateway?.apiId ??
38
+ new StackParameter_1.StackParameter({ prefix: "api-gateway", key: "id" }).import(service, this.props.basePlatformStackName);
58
39
  const vpcLink = aws_apigatewayv2_1.VpcLink.fromVpcLinkAttributes(this.scope, `${this.parentService.id}VpcLink`, {
59
40
  vpc: this.parentService.vpc,
60
- vpcLinkId: vpcLinkIdParameter.import(service, this.props.basePlatformStackName),
41
+ vpcLinkId,
61
42
  });
62
43
  const httpApi = aws_apigatewayv2_1.HttpApi.fromHttpApiAttributes(this.scope, `${this.parentService.id}HttpApi`, {
63
- httpApiId: apiGatewayIdParameter.import(service, this.props.basePlatformStackName),
44
+ httpApiId: apiGatewayId,
64
45
  });
65
46
  const integration = new aws_apigatewayv2_integrations_1.HttpServiceDiscoveryIntegration(`${this.parentService.id}ServiceDiscoveryIntegration`, service.cloudMapService, {
66
47
  method: aws_apigatewayv2_1.HttpMethod.ANY,
67
48
  vpcLink,
68
49
  parameterMapping: new aws_apigatewayv2_1.ParameterMapping().overwritePath(aws_apigatewayv2_1.MappingValue.custom("/$request.path.proxy")),
69
50
  });
70
- const userPoolClient = aws_cognito_1.UserPoolClient.fromUserPoolClientId(this.scope, `${this.parentService.id}ApiUserPoolClient`, cognitoFullAccessClientParameter.import(service, this.props.basePlatformStackName));
71
- const appUserPoolClient = aws_cognito_1.UserPoolClient.fromUserPoolClientId(this.scope, `${this.parentService.id}AppUserPoolClient`, cognitoUserPoolClientParameter.import(service, this.props.basePlatformStackName));
72
- const userPool = aws_cognito_1.UserPool.fromUserPoolId(this.scope, `${this.parentService.id}ApiGWUserPool`, cognitoUserPoolIdParameter.import(service, this.props.basePlatformStackName));
51
+ // Cognito: use injected construct if provided, else import from base platform
52
+ const userPoolClient = this.props.cognitoUserPool
53
+ ? this.props.cognitoUserPool.apiUserPoolClient
54
+ : aws_cognito_1.UserPoolClient.fromUserPoolClientId(this.scope, `${this.parentService.id}ApiUserPoolClient`, new StackParameter_1.StackParameter({
55
+ prefix: "cognito",
56
+ key: "full-access-client-id",
57
+ }).import(service, this.props.basePlatformStackName));
58
+ const appUserPoolClient = this.props.cognitoUserPool
59
+ ? this.props.cognitoUserPool.appUserPoolClient
60
+ : aws_cognito_1.UserPoolClient.fromUserPoolClientId(this.scope, `${this.parentService.id}AppUserPoolClient`, new StackParameter_1.StackParameter({
61
+ prefix: "cognito",
62
+ key: "user-pool-client-id",
63
+ }).import(service, this.props.basePlatformStackName));
64
+ const userPool = this.props.cognitoUserPool
65
+ ? this.props.cognitoUserPool.userPool
66
+ : aws_cognito_1.UserPool.fromUserPoolId(this.scope, `${this.parentService.id}ApiGWUserPool`, new StackParameter_1.StackParameter({
67
+ prefix: "cognito",
68
+ key: "user-pool-id",
69
+ }).import(service, this.props.basePlatformStackName));
73
70
  const authorizer = new aws_apigatewayv2_authorizers_1.HttpUserPoolAuthorizer(`${this.parentService.id}Authorizer`, userPool, {
74
71
  userPoolClients: [userPoolClient, appUserPoolClient],
75
72
  });
@@ -81,11 +78,13 @@ class ApiGatewayExtension extends ecs_service_extensions_1.ServiceExtension {
81
78
  });
82
79
  const apiGwSecurityGroup = aws_ec2_1.SecurityGroup.fromSecurityGroupId(this.scope, `${this.parentService.id}ApiGatewaySecurityGroup`, this.props.securityGroupId, { allowAllOutbound: false });
83
80
  apiGwSecurityGroup.connections.allowTo(service, aws_ec2_1.Port.tcp(container.trafficPort), `Allow traffic from API GW to ${this.parentService.id}`);
84
- const fqdn = fqdnParameter.import(service, this.props.basePlatformStackName);
81
+ // FQDN: use direct value if provided, else import from base platform
82
+ const fqdn = this.props.fqdn ??
83
+ new StackParameter_1.StackParameter({ prefix: "fqdn", key: "fqdn" }).import(service, this.props.basePlatformStackName);
85
84
  this.parentService.addURL("public", `https://api.${fqdn}${this.props.prefix}`);
86
85
  }
87
86
  }
88
87
  exports.ApiGatewayExtension = ApiGatewayExtension;
89
88
  _a = JSII_RTTI_SYMBOL_1;
90
- ApiGatewayExtension[_a] = { fqn: "@btc-embedded/cdk-extensions.ApiGatewayExtension", version: "0.22.22" };
91
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ApiGatewayExtension.js","sourceRoot":"","sources":["../../src/extensions/ApiGatewayExtension.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Container,\n  ServiceExtension,\n} from \"@aws-cdk-containers/ecs-service-extensions\";\nimport {\n  HttpApi,\n  HttpMethod,\n  HttpRoute,\n  HttpRouteKey,\n  MappingValue,\n  ParameterMapping,\n  VpcLink,\n} from \"aws-cdk-lib/aws-apigatewayv2\";\nimport { HttpUserPoolAuthorizer } from \"aws-cdk-lib/aws-apigatewayv2-authorizers\";\nimport { HttpServiceDiscoveryIntegration } from \"aws-cdk-lib/aws-apigatewayv2-integrations\";\nimport { UserPool, UserPoolClient } from \"aws-cdk-lib/aws-cognito\";\nimport { Port, SecurityGroup } from \"aws-cdk-lib/aws-ec2\";\nimport { Ec2Service, FargateService } from \"aws-cdk-lib/aws-ecs\";\nimport { StackParameter } from \"../utils/StackParameter\";\n\nexport interface ApiGatewayExtensionProps {\n  /**\n   * The prefix for the API Gateway endpoint.\n   */\n  readonly prefix: string;\n\n  /**\n   * The name of the base platform stack.\n   */\n  readonly basePlatformStackName?: string;\n\n  /**\n   * The security group ID of the API Gateway which should be allowed to connect to the service.\n   */\n  readonly securityGroupId: string;\n}\n\n/**\n * Configures the service to be reachable via an API Gateway.\n * @deprecated Use {@link HTTPApiExtension} instead\n */\nexport class ApiGatewayExtension extends ServiceExtension {\n  readonly props: ApiGatewayExtensionProps;\n\n  /**\n   *\n   * @param props\n   */\n  constructor(props: ApiGatewayExtensionProps) {\n    super(\"api-gateway-extension\");\n    this.props = props;\n  }\n\n  useService(service: Ec2Service | FargateService): void {\n    const vpcLinkIdParameter = new StackParameter({\n      prefix: \"api-gateway\",\n      key: \"vpc-link-id\",\n    });\n    const apiGatewayIdParameter = new StackParameter({\n      prefix: \"api-gateway\",\n      key: \"id\",\n    });\n    const cognitoFullAccessClientParameter = new StackParameter({\n      prefix: \"cognito\",\n      key: \"full-access-client-id\",\n    });\n    const cognitoUserPoolClientParameter = new StackParameter({\n      prefix: \"cognito\",\n      key: \"user-pool-client-id\",\n    });\n    const cognitoUserPoolIdParameter = new StackParameter({\n      prefix: \"cognito\",\n      key: \"user-pool-id\",\n    });\n    const fqdnParameter = new StackParameter({\n      prefix: \"fqdn\",\n      key: \"fqdn\",\n    });\n\n    if (!service.cloudMapService) {\n      throw new Error(\n        \"This extension requires the CloudMap extension to be added\",\n      );\n    }\n\n    const container = this.parentService.serviceDescription.get(\n      \"service-container\",\n    ) as Container;\n    if (!container || !container.trafficPort) {\n      throw new Error(\n        \"Cannot apply API Gateway Extension, no main app container found\",\n      );\n    }\n\n    const vpcLink = VpcLink.fromVpcLinkAttributes(\n      this.scope,\n      `${this.parentService.id}VpcLink`,\n      {\n        vpc: this.parentService.vpc,\n        vpcLinkId: vpcLinkIdParameter.import(\n          service,\n          this.props.basePlatformStackName,\n        ),\n      },\n    );\n\n    const httpApi = HttpApi.fromHttpApiAttributes(\n      this.scope,\n      `${this.parentService.id}HttpApi`,\n      {\n        httpApiId: apiGatewayIdParameter.import(\n          service,\n          this.props.basePlatformStackName,\n        ),\n      },\n    );\n\n    const integration = new HttpServiceDiscoveryIntegration(\n      `${this.parentService.id}ServiceDiscoveryIntegration`,\n      service.cloudMapService,\n      {\n        method: HttpMethod.ANY,\n        vpcLink,\n        parameterMapping: new ParameterMapping().overwritePath(\n          MappingValue.custom(\"/$request.path.proxy\"),\n        ),\n      },\n    );\n\n    const userPoolClient = UserPoolClient.fromUserPoolClientId(\n      this.scope,\n      `${this.parentService.id}ApiUserPoolClient`,\n      cognitoFullAccessClientParameter.import(\n        service,\n        this.props.basePlatformStackName,\n      ),\n    );\n\n    const appUserPoolClient = UserPoolClient.fromUserPoolClientId(\n      this.scope,\n      `${this.parentService.id}AppUserPoolClient`,\n      cognitoUserPoolClientParameter.import(\n        service,\n        this.props.basePlatformStackName,\n      ),\n    );\n\n    const userPool = UserPool.fromUserPoolId(\n      this.scope,\n      `${this.parentService.id}ApiGWUserPool`,\n      cognitoUserPoolIdParameter.import(\n        service,\n        this.props.basePlatformStackName,\n      ),\n    );\n\n    const authorizer = new HttpUserPoolAuthorizer(\n      `${this.parentService.id}Authorizer`,\n      userPool,\n      {\n        userPoolClients: [userPoolClient, appUserPoolClient],\n      },\n    );\n\n    new HttpRoute(this.scope, `${this.parentService.id}HttpRoute`, {\n      httpApi,\n      integration,\n      routeKey: HttpRouteKey.with(\n        `${this.props.prefix}/{proxy+}`,\n        HttpMethod.ANY,\n      ),\n      authorizer,\n    });\n\n    const apiGwSecurityGroup = SecurityGroup.fromSecurityGroupId(\n      this.scope,\n      `${this.parentService.id}ApiGatewaySecurityGroup`,\n      this.props.securityGroupId,\n      { allowAllOutbound: false },\n    );\n\n    apiGwSecurityGroup.connections.allowTo(\n      service,\n      Port.tcp(container.trafficPort),\n      `Allow traffic from API GW to ${this.parentService.id}`,\n    );\n\n    const fqdn = fqdnParameter.import(\n      service,\n      this.props.basePlatformStackName,\n    );\n\n    this.parentService.addURL(\n      \"public\",\n      `https://api.${fqdn}${this.props.prefix}`,\n    );\n  }\n}\n"]}
89
+ ApiGatewayExtension[_a] = { fqn: "@btc-embedded/cdk-extensions.ApiGatewayExtension", version: "0.22.24" };
90
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ApiGatewayExtension.js","sourceRoot":"","sources":["../../src/extensions/ApiGatewayExtension.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Container,\n  ServiceExtension,\n} from \"@aws-cdk-containers/ecs-service-extensions\";\nimport {\n  HttpApi,\n  HttpMethod,\n  HttpRoute,\n  HttpRouteKey,\n  MappingValue,\n  ParameterMapping,\n  VpcLink,\n} from \"aws-cdk-lib/aws-apigatewayv2\";\nimport { HttpUserPoolAuthorizer } from \"aws-cdk-lib/aws-apigatewayv2-authorizers\";\nimport { HttpServiceDiscoveryIntegration } from \"aws-cdk-lib/aws-apigatewayv2-integrations\";\nimport { UserPool, UserPoolClient } from \"aws-cdk-lib/aws-cognito\";\nimport { Port, SecurityGroup } from \"aws-cdk-lib/aws-ec2\";\nimport { Ec2Service, FargateService } from \"aws-cdk-lib/aws-ecs\";\nimport { IApiGateway } from \"../platform/ApiGateway\";\nimport { ICognitoUserPool } from \"../platform/CognitoUserPool\";\nimport { StackParameter } from \"../utils/StackParameter\";\n\nexport interface ApiGatewayExtensionProps {\n  /**\n   * The prefix for the API Gateway endpoint.\n   */\n  readonly prefix: string;\n\n  /**\n   * The name of the base platform stack.\n   */\n  readonly basePlatformStackName?: string;\n\n  /**\n   * The security group ID of the API Gateway which should be allowed to connect to the service.\n   */\n  readonly securityGroupId: string;\n\n  /**\n   * Provide a direct {@link IApiGateway} reference for same-stack usage.\n   * When omitted the VPC link ID and API ID are imported from the base platform stack.\n   */\n  readonly apiGateway?: IApiGateway;\n\n  /**\n   * Provide a direct Cognito user pool reference for same-stack usage.\n   * When omitted the user pool attributes are imported from the base platform stack.\n   */\n  readonly cognitoUserPool?: ICognitoUserPool;\n\n  /**\n   * Provide the FQDN directly for same-stack usage.\n   * When omitted the FQDN is imported from the base platform stack.\n   */\n  readonly fqdn?: string;\n}\n\n/**\n * Configures the service to be reachable via an API Gateway.\n * @deprecated Use {@link HTTPApiExtension} instead\n */\nexport class ApiGatewayExtension extends ServiceExtension {\n  readonly props: ApiGatewayExtensionProps;\n\n  /**\n   *\n   * @param props\n   */\n  constructor(props: ApiGatewayExtensionProps) {\n    super(\"api-gateway-extension\");\n    this.props = props;\n  }\n\n  useService(service: Ec2Service | FargateService): void {\n    if (!service.cloudMapService) {\n      throw new Error(\n        \"This extension requires the CloudMap extension to be added\",\n      );\n    }\n\n    const container = this.parentService.serviceDescription.get(\n      \"service-container\",\n    ) as Container;\n    if (!container || !container.trafficPort) {\n      throw new Error(\n        \"Cannot apply API Gateway Extension, no main app container found\",\n      );\n    }\n\n    // VPC link ID and API ID: use direct values if provided, else import from base platform\n    const vpcLinkId =\n      this.props.apiGateway?.vpcLinkId ??\n      new StackParameter({ prefix: \"api-gateway\", key: \"vpc-link-id\" }).import(\n        service,\n        this.props.basePlatformStackName,\n      );\n\n    const apiGatewayId =\n      this.props.apiGateway?.apiId ??\n      new StackParameter({ prefix: \"api-gateway\", key: \"id\" }).import(\n        service,\n        this.props.basePlatformStackName,\n      );\n\n    const vpcLink = VpcLink.fromVpcLinkAttributes(\n      this.scope,\n      `${this.parentService.id}VpcLink`,\n      {\n        vpc: this.parentService.vpc,\n        vpcLinkId,\n      },\n    );\n\n    const httpApi = HttpApi.fromHttpApiAttributes(\n      this.scope,\n      `${this.parentService.id}HttpApi`,\n      {\n        httpApiId: apiGatewayId,\n      },\n    );\n\n    const integration = new HttpServiceDiscoveryIntegration(\n      `${this.parentService.id}ServiceDiscoveryIntegration`,\n      service.cloudMapService,\n      {\n        method: HttpMethod.ANY,\n        vpcLink,\n        parameterMapping: new ParameterMapping().overwritePath(\n          MappingValue.custom(\"/$request.path.proxy\"),\n        ),\n      },\n    );\n\n    // Cognito: use injected construct if provided, else import from base platform\n    const userPoolClient = this.props.cognitoUserPool\n      ? this.props.cognitoUserPool.apiUserPoolClient\n      : UserPoolClient.fromUserPoolClientId(\n          this.scope,\n          `${this.parentService.id}ApiUserPoolClient`,\n          new StackParameter({\n            prefix: \"cognito\",\n            key: \"full-access-client-id\",\n          }).import(service, this.props.basePlatformStackName),\n        );\n\n    const appUserPoolClient = this.props.cognitoUserPool\n      ? this.props.cognitoUserPool.appUserPoolClient\n      : UserPoolClient.fromUserPoolClientId(\n          this.scope,\n          `${this.parentService.id}AppUserPoolClient`,\n          new StackParameter({\n            prefix: \"cognito\",\n            key: \"user-pool-client-id\",\n          }).import(service, this.props.basePlatformStackName),\n        );\n\n    const userPool = this.props.cognitoUserPool\n      ? this.props.cognitoUserPool.userPool\n      : UserPool.fromUserPoolId(\n          this.scope,\n          `${this.parentService.id}ApiGWUserPool`,\n          new StackParameter({\n            prefix: \"cognito\",\n            key: \"user-pool-id\",\n          }).import(service, this.props.basePlatformStackName),\n        );\n\n    const authorizer = new HttpUserPoolAuthorizer(\n      `${this.parentService.id}Authorizer`,\n      userPool,\n      {\n        userPoolClients: [userPoolClient, appUserPoolClient],\n      },\n    );\n\n    new HttpRoute(this.scope, `${this.parentService.id}HttpRoute`, {\n      httpApi,\n      integration,\n      routeKey: HttpRouteKey.with(\n        `${this.props.prefix}/{proxy+}`,\n        HttpMethod.ANY,\n      ),\n      authorizer,\n    });\n\n    const apiGwSecurityGroup = SecurityGroup.fromSecurityGroupId(\n      this.scope,\n      `${this.parentService.id}ApiGatewaySecurityGroup`,\n      this.props.securityGroupId,\n      { allowAllOutbound: false },\n    );\n\n    apiGwSecurityGroup.connections.allowTo(\n      service,\n      Port.tcp(container.trafficPort),\n      `Allow traffic from API GW to ${this.parentService.id}`,\n    );\n\n    // FQDN: use direct value if provided, else import from base platform\n    const fqdn =\n      this.props.fqdn ??\n      new StackParameter({ prefix: \"fqdn\", key: \"fqdn\" }).import(\n        service,\n        this.props.basePlatformStackName,\n      );\n\n    this.parentService.addURL(\n      \"public\",\n      `https://api.${fqdn}${this.props.prefix}`,\n    );\n  }\n}\n"]}
@@ -103,5 +103,5 @@ class ApplicationContainer extends ecs_service_extensions_1.Container {
103
103
  }
104
104
  exports.ApplicationContainer = ApplicationContainer;
105
105
  _a = JSII_RTTI_SYMBOL_1;
106
- ApplicationContainer[_a] = { fqn: "@btc-embedded/cdk-extensions.ApplicationContainer", version: "0.22.22" };
106
+ ApplicationContainer[_a] = { fqn: "@btc-embedded/cdk-extensions.ApplicationContainer", version: "0.22.24" };
107
107
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQXBwbGljYXRpb25Db250YWluZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZXh0ZW5zaW9ucy9BcHBsaWNhdGlvbkNvbnRhaW5lci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLHVGQUdvRDtBQUdwRCxzREFBMEQ7QUFFMUQ7O0dBRUc7QUFDSCxNQUFNLDZCQUE4QixTQUFRLDhDQUFxQjtJQUMvRCxZQUE2QixhQUF3QztRQUNuRSxLQUFLLEVBQUUsQ0FBQztRQURtQixrQkFBYSxHQUFiLGFBQWEsQ0FBMkI7SUFFckUsQ0FBQztJQUVELHlCQUF5QixDQUN2QixPQUF1QztRQUV2QyxNQUFNLEVBQ0osU0FBUyxFQUNULFdBQVcsRUFDWCxRQUFRLEVBQ1IsbUJBQW1CLEVBQ25CLHFCQUFxQixFQUNyQixHQUFHLFNBQVMsRUFDYixHQUFHLElBQUksQ0FBQyxhQUFhLENBQUM7UUFFdkIsaURBQWlEO1FBQ2pELG9FQUFvRTtRQUNwRSxNQUFNLG1CQUFtQixHQUFHLElBQUEsZ0NBQWtCLEVBQzVDLG1CQUFtQixFQUNuQixTQUFTLENBQUMsV0FBVyxFQUNyQixXQUFZLENBQ2IsQ0FBQztRQUVGLDhDQUE4QztRQUM5QyxPQUFPO1lBQ0wsR0FBRyxPQUFPO1lBQ1YsR0FBRyxTQUFTO1lBQ1osR0FBRyxDQUFDLG1CQUFtQixJQUFJLEVBQUUsV0FBVyxFQUFFLG1CQUFtQixFQUFFLENBQUM7U0FDakUsQ0FBQztJQUNKLENBQUM7Q0FDRjtBQWNEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FnREc7QUFDSCxNQUFhLG9CQUFxQixTQUFRLGtDQUFTO0lBQ2pELFlBQVksS0FBZ0M7UUFDMUMsK0VBQStFO1FBQy9FLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxFQUFFLENBQUM7WUFDZixNQUFNLElBQUksS0FBSyxDQUFDLHlDQUF5QyxDQUFDLENBQUM7UUFDN0QsQ0FBQztRQUNELElBQUksQ0FBQyxLQUFLLENBQUMsU0FBUyxFQUFFLENBQUM7WUFDckIsTUFBTSxJQUFJLEtBQUssQ0FBQywrQ0FBK0MsQ0FBQyxDQUFDO1FBQ25FLENBQUM7UUFDRCxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsRUFBRSxDQUFDO1lBQ3ZCLE1BQU0sSUFBSSxLQUFLLENBQUMsaURBQWlELENBQUMsQ0FBQztRQUNyRSxDQUFDO1FBRUQsNkRBQTZEO1FBQzdELElBQUEsZ0NBQWtCLEVBQ2hCLEtBQUssQ0FBQyxtQkFBbUIsRUFDekIsS0FBSyxDQUFDLFdBQVcsRUFDakIsS0FBSyxDQUFDLFdBQVcsQ0FDbEIsQ0FBQztRQUVGLDhDQUE4QztRQUM5QyxLQUFLLENBQUM7WUFDSixHQUFHLEVBQUUsS0FBSyxDQUFDLEdBQUc7WUFDZCxTQUFTLEVBQUUsS0FBSyxDQUFDLFNBQVM7WUFDMUIsV0FBVyxFQUFFLEtBQUssQ0FBQyxXQUFXO1lBQzlCLEtBQUssRUFBRSxLQUFLLENBQUMsS0FBSztZQUNsQixRQUFRLEVBQUUsS0FBSyxDQUFDLFFBQVE7U0FDekIsQ0FBQyxDQUFDO1FBRUgscURBQXFEO1FBQ3JELElBQUksQ0FBQyx3QkFBd0IsQ0FBQyxJQUFJLDZCQUE2QixDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7SUFDMUUsQ0FBQzs7QUEvQkgsb0RBZ0NDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQ29udGFpbmVyLFxuICBDb250YWluZXJNdXRhdGluZ0hvb2ssXG59IGZyb20gXCJAYXdzLWNkay1jb250YWluZXJzL2Vjcy1zZXJ2aWNlLWV4dGVuc2lvbnNcIjtcbmltcG9ydCAqIGFzIGVjcyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWVjc1wiO1xuaW1wb3J0IHsgRW5oYW5jZWRDb250YWluZXJQcm9wcyB9IGZyb20gXCIuL2NvbnRhaW5lci1jb21tb25cIjtcbmltcG9ydCB7IHJlc29sdmVIZWFsdGhDaGVjayB9IGZyb20gXCIuLi91dGlscy9oZWFsdGhjaGVja1wiO1xuXG4vKipcbiAqIEhvb2sgdG8gYXBwbHkgYWxsIGNvbnRhaW5lciBkZWZpbml0aW9uIG9wdGlvbnMuXG4gKi9cbmNsYXNzIEVuaGFuY2VkQ29udGFpbmVyTXV0YXRpbmdIb29rIGV4dGVuZHMgQ29udGFpbmVyTXV0YXRpbmdIb29rIHtcbiAgY29uc3RydWN0b3IocHJpdmF0ZSByZWFkb25seSBlbmhhbmNlZFByb3BzOiBBcHBsaWNhdGlvbkNvbnRhaW5lclByb3BzKSB7XG4gICAgc3VwZXIoKTtcbiAgfVxuXG4gIG11dGF0ZUNvbnRhaW5lckRlZmluaXRpb24oXG4gICAgb3B0aW9uczogZWNzLkNvbnRhaW5lckRlZmluaXRpb25PcHRpb25zLFxuICApOiBlY3MuQ29udGFpbmVyRGVmaW5pdGlvbk9wdGlvbnMge1xuICAgIGNvbnN0IHtcbiAgICAgIG1lbW9yeU1pQixcbiAgICAgIHRyYWZmaWNQb3J0LFxuICAgICAgbG9nR3JvdXAsXG4gICAgICBoZWFsdGhDaGVja0VuZHBvaW50LFxuICAgICAgY29udGFpbmVyRGVwZW5kZW5jaWVzLFxuICAgICAgLi4ucmVzdFByb3BzXG4gICAgfSA9IHRoaXMuZW5oYW5jZWRQcm9wcztcblxuICAgIC8vIFJlc29sdmUgaGVhbHRoIGNoZWNrIChlbmRwb2ludCBvciBmdWxsIGNvbmZpZylcbiAgICAvLyBVc2Ugbm9uLW51bGwgYXNzZXJ0aW9uIGFzIHRyYWZmaWNQb3J0IGlzIHZhbGlkYXRlZCBpbiBjb25zdHJ1Y3RvclxuICAgIGNvbnN0IHJlc29sdmVkSGVhbHRoQ2hlY2sgPSByZXNvbHZlSGVhbHRoQ2hlY2soXG4gICAgICBoZWFsdGhDaGVja0VuZHBvaW50LFxuICAgICAgcmVzdFByb3BzLmhlYWx0aENoZWNrLFxuICAgICAgdHJhZmZpY1BvcnQhLFxuICAgICk7XG5cbiAgICAvLyBNZXJnZSBhbGwgcHJvcHMsIHdpdGggcmVzb2x2ZWQgaGVhbHRoIGNoZWNrXG4gICAgcmV0dXJuIHtcbiAgICAgIC4uLm9wdGlvbnMsXG4gICAgICAuLi5yZXN0UHJvcHMsXG4gICAgICAuLi4ocmVzb2x2ZWRIZWFsdGhDaGVjayAmJiB7IGhlYWx0aENoZWNrOiByZXNvbHZlZEhlYWx0aENoZWNrIH0pLFxuICAgIH07XG4gIH1cbn1cblxuLyoqXG4gKiBQcm9wZXJ0aWVzIGZvciBBcHBsaWNhdGlvbkNvbnRhaW5lci5cbiAqIEFsbCBwcm9wZXJ0aWVzIGZyb20gRW5oYW5jZWRDb250YWluZXJQcm9wcyBhcmUgc3VwcG9ydGVkLlxuICpcbiAqICoqUmVxdWlyZWQgcHJvcGVydGllczoqKlxuICogLSBjcHU6IEhvdyBtdWNoIENQVSB0aGUgY29udGFpbmVyIHJlcXVpcmVzXG4gKiAtIG1lbW9yeU1pQjogSG93IG11Y2ggbWVtb3J5IGluIG1lZ2FieXRlcyB0aGUgY29udGFpbmVyIHJlcXVpcmVzXG4gKiAtIGltYWdlOiBUaGUgaW1hZ2UgdG8gcnVuXG4gKiAtIHRyYWZmaWNQb3J0OiBXaGF0IHBvcnQgdGhlIGltYWdlIGxpc3RlbnMgZm9yIHRyYWZmaWMgb25cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBBcHBsaWNhdGlvbkNvbnRhaW5lclByb3BzIGV4dGVuZHMgRW5oYW5jZWRDb250YWluZXJQcm9wcyB7fVxuXG4vKipcbiAqIEFuIGFwcGxpY2F0aW9uIGNvbnRhaW5lciB3aXRoIGJ1aWx0LWluIHN1cHBvcnQgZm9yIHNpbXBsaWZpZWQgaGVhbHRoIGNoZWNrc1xuICogYW5kIGFsbCBjb250YWluZXIgZGVmaW5pdGlvbiBvcHRpb25zLlxuICpcbiAqIFRoaXMgZXh0ZW5zaW9uIGV4dGVuZHMgdGhlIHVwc3RyZWFtIENvbnRhaW5lciBjbGFzcyBhbmQgcHJvdmlkZXM6XG4gKiAxLiBTaW1wbGlmaWVkIGhlYWx0aCBjaGVjayBBUEkgdmlhIGBoZWFsdGhDaGVja0VuZHBvaW50YFxuICogMi4gRnVsbCBhY2Nlc3MgdG8gYWxsIENvbnRhaW5lckRlZmluaXRpb25PcHRpb25zICh1c2VyLCBwcml2aWxlZ2VkLCB3b3JraW5nRGlyZWN0b3J5LCBldGMuKVxuICogMy4gRHJvcC1pbiByZXBsYWNlbWVudCBmb3IgQ29udGFpbmVyIHdpdGggYWRkaXRpb25hbCBmZWF0dXJlc1xuICpcbiAqICoqUmVwbGFjZXMgdGhlIG5lZWQgZm9yOioqIENvbnRhaW5lciArIE1vZGlmeUNvbnRhaW5lckRlZmluaXRpb25FeHRlbnNpb25cbiAqXG4gKiBAZXhhbXBsZVxuICogLy8gU2ltcGxlIGhlYWx0aCBjaGVja1xuICogbmV3IEFwcGxpY2F0aW9uQ29udGFpbmVyKHtcbiAqICAgY3B1OiAxMDI0LFxuICogICBtZW1vcnlNaUI6IDIwNDgsXG4gKiAgIGltYWdlOiBFY3JJbWFnZS5mcm9tRG9ja2VySW1hZ2VBc3NldChpbWFnZSksXG4gKiAgIHRyYWZmaWNQb3J0OiAzMDAwLFxuICogICBoZWFsdGhDaGVja0VuZHBvaW50OiAnL2hlYWx0aCcsXG4gKiB9KVxuICpcbiAqIEBleGFtcGxlXG4gKiAvLyBXaXRoIGFkZGl0aW9uYWwgY29udGFpbmVyIG9wdGlvbnNcbiAqIG5ldyBBcHBsaWNhdGlvbkNvbnRhaW5lcih7XG4gKiAgIGNwdTogMTAyNCxcbiAqICAgbWVtb3J5TWlCOiAyMDQ4LFxuICogICBpbWFnZTogRWNySW1hZ2UuZnJvbURvY2tlckltYWdlQXNzZXQoaW1hZ2UpLFxuICogICB0cmFmZmljUG9ydDogMzAwMCxcbiAqICAgaGVhbHRoQ2hlY2tFbmRwb2ludDogJy9oZWFsdGgnLFxuICogICB1c2VyOiAnMTAwMDoxMDAwJyxcbiAqICAgd29ya2luZ0RpcmVjdG9yeTogJy9hcHAnLFxuICogICBlbnZpcm9ubWVudDoge1xuICogICAgIE5PREVfRU5WOiAncHJvZHVjdGlvbicsXG4gKiAgIH0sXG4gKiB9KVxuICpcbiAqIEBleGFtcGxlXG4gKiAvLyBGdWxsIGhlYWx0aCBjaGVjayBjb250cm9sIChlc2NhcGUgaGF0Y2gpXG4gKiBuZXcgQXBwbGljYXRpb25Db250YWluZXIoe1xuICogICBjcHU6IDEwMjQsXG4gKiAgIG1lbW9yeU1pQjogMjA0OCxcbiAqICAgaW1hZ2U6IEVjckltYWdlLmZyb21Eb2NrZXJJbWFnZUFzc2V0KGltYWdlKSxcbiAqICAgdHJhZmZpY1BvcnQ6IDMwMDAsXG4gKiAgIGhlYWx0aENoZWNrOiB7XG4gKiAgICAgY29tbWFuZDogWydDTUQtU0hFTEwnLCAnY3VzdG9tLWhlYWx0aC1jaGVjay5zaCddLFxuICogICAgIGludGVydmFsOiBEdXJhdGlvbi5zZWNvbmRzKDMwKSxcbiAqICAgfSxcbiAqIH0pO1xuICovXG5leHBvcnQgY2xhc3MgQXBwbGljYXRpb25Db250YWluZXIgZXh0ZW5kcyBDb250YWluZXIge1xuICBjb25zdHJ1Y3Rvcihwcm9wczogQXBwbGljYXRpb25Db250YWluZXJQcm9wcykge1xuICAgIC8vIFZhbGlkYXRlIHJlcXVpcmVkIHByb3BlcnRpZXMgdGhhdCBhcmUgb3B0aW9uYWwgaW4gQ29udGFpbmVyRGVmaW5pdGlvbk9wdGlvbnNcbiAgICBpZiAoIXByb3BzLmNwdSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFwiQXBwbGljYXRpb25Db250YWluZXI6ICdjcHUnIGlzIHJlcXVpcmVkXCIpO1xuICAgIH1cbiAgICBpZiAoIXByb3BzLm1lbW9yeU1pQikge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFwiQXBwbGljYXRpb25Db250YWluZXI6ICdtZW1vcnlNaUInIGlzIHJlcXVpcmVkXCIpO1xuICAgIH1cbiAgICBpZiAoIXByb3BzLnRyYWZmaWNQb3J0KSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJBcHBsaWNhdGlvbkNvbnRhaW5lcjogJ3RyYWZmaWNQb3J0JyBpcyByZXF1aXJlZFwiKTtcbiAgICB9XG5cbiAgICAvLyBWYWxpZGF0ZSBoZWFsdGggY2hlY2sgY29uZmlndXJhdGlvbiAoYmVmb3JlIGNhbGxpbmcgc3VwZXIpXG4gICAgcmVzb2x2ZUhlYWx0aENoZWNrKFxuICAgICAgcHJvcHMuaGVhbHRoQ2hlY2tFbmRwb2ludCxcbiAgICAgIHByb3BzLmhlYWx0aENoZWNrLFxuICAgICAgcHJvcHMudHJhZmZpY1BvcnQsXG4gICAgKTtcblxuICAgIC8vIENhbGwgcGFyZW50IGNvbnN0cnVjdG9yIHdpdGggcmVxdWlyZWQgcHJvcHNcbiAgICBzdXBlcih7XG4gICAgICBjcHU6IHByb3BzLmNwdSxcbiAgICAgIG1lbW9yeU1pQjogcHJvcHMubWVtb3J5TWlCLFxuICAgICAgdHJhZmZpY1BvcnQ6IHByb3BzLnRyYWZmaWNQb3J0LFxuICAgICAgaW1hZ2U6IHByb3BzLmltYWdlLFxuICAgICAgbG9nR3JvdXA6IHByb3BzLmxvZ0dyb3VwLFxuICAgIH0pO1xuXG4gICAgLy8gQWRkIGhvb2sgdG8gYXBwbHkgYWxsIGNvbnRhaW5lciBkZWZpbml0aW9uIG9wdGlvbnNcbiAgICB0aGlzLmFkZENvbnRhaW5lck11dGF0aW5nSG9vayhuZXcgRW5oYW5jZWRDb250YWluZXJNdXRhdGluZ0hvb2socHJvcHMpKTtcbiAgfVxufVxuIl19
@@ -3,14 +3,30 @@ import { IUserPool, IUserPoolClient, IUserPoolDomain } from "aws-cdk-lib/aws-cog
3
3
  import { Ec2Service, FargateService } from "aws-cdk-lib/aws-ecs";
4
4
  import { IApplicationListener } from "aws-cdk-lib/aws-elasticloadbalancingv2";
5
5
  import { Construct } from "constructs";
6
+ import { ICognitoUserPool } from "../platform";
6
7
  export interface ApplicationLoadBalancerExtensionProps {
7
8
  readonly basePlatformStackName?: string;
8
9
  readonly userPoolClientSecret: string;
9
10
  readonly securityGroupId: string;
10
11
  readonly clientScope: string;
11
12
  readonly healtCheckPath?: string;
13
+ /** Path to redirect the client after successful authentication. */
12
14
  readonly path?: string;
15
+ /**
16
+ * Provide a direct ALB listener reference for same-stack usage.
17
+ * When omitted the listener ARN is imported from the base platform stack.
18
+ */
13
19
  readonly listener?: IApplicationListener;
20
+ /**
21
+ * Provide a direct Cognito user pool reference for same-stack usage.
22
+ * When omitted the user pool attributes are imported from the base platform stack.
23
+ */
24
+ readonly cognitoUserPool?: ICognitoUserPool;
25
+ /**
26
+ * Provide the FQDN directly for same-stack usage.
27
+ * When omitted the FQDN is imported from the base platform stack.
28
+ */
29
+ readonly fqdn?: string;
14
30
  }
15
31
  /**
16
32
  * @deprecated Use {@link ApplicationLoadBalancerExtensionV2} instead
@@ -52,40 +52,40 @@ class ApplicationLoadBalancerExtension extends ecs_service_extensions_1.ServiceE
52
52
  }
53
53
  prehook(service, _scope) {
54
54
  this.parentService = service;
55
- const fqdnParameter = new StackParameter_1.StackParameter({
56
- key: "fqdn",
57
- prefix: "fqdn",
58
- });
59
- const albListenerArnParameter = new StackParameter_1.StackParameter({
60
- prefix: ["alb"],
61
- key: "listener-arn",
62
- });
63
- const cognitoUserPoolIdParameter = new StackParameter_1.StackParameter({
64
- prefix: "cognito",
65
- key: "user-pool-id",
66
- });
67
- const cognitoUserPoolClientParameter = new StackParameter_1.StackParameter({
68
- prefix: "cognito",
69
- key: "user-pool-client-id",
70
- });
71
- const cognitoDomainPrefixParameter = new StackParameter_1.StackParameter({
72
- prefix: "cognito",
73
- key: "domain-prefix",
74
- });
75
- this.fqdn = fqdnParameter.import(service, this.props.basePlatformStackName);
76
- const albListenerArn = albListenerArnParameter.import(service, this.props.basePlatformStackName);
55
+ // FQDN: use direct value if provided, else import from base platform stack
56
+ this.fqdn =
57
+ this.props.fqdn ??
58
+ new StackParameter_1.StackParameter({ key: "fqdn", prefix: "fqdn" }).import(service, this.props.basePlatformStackName);
77
59
  if (this.props.listener) {
78
60
  this.listener = this.props.listener;
79
61
  }
80
62
  else {
81
63
  this.listener = aws_elasticloadbalancingv2_1.ApplicationListener.fromApplicationListenerAttributes(service, "ALBListener", {
82
- listenerArn: albListenerArn,
64
+ listenerArn: new StackParameter_1.StackParameter({
65
+ prefix: ["alb"],
66
+ key: "listener-arn",
67
+ }).import(service, this.props.basePlatformStackName),
83
68
  securityGroup: aws_ec2_1.SecurityGroup.fromSecurityGroupId(service, "ALBSecurityGroup", this.props.securityGroupId, { allowAllOutbound: false }),
84
69
  });
85
70
  }
86
- this.userPool = aws_cognito_1.UserPool.fromUserPoolId(service, "ALBUserPool", cognitoUserPoolIdParameter.import(service, this.props.basePlatformStackName));
87
- this.userPoolClient = aws_cognito_1.UserPoolClient.fromUserPoolClientId(service, "UserPoolClient", cognitoUserPoolClientParameter.import(service, this.props.basePlatformStackName));
88
- this.userPoolDomain = aws_cognito_1.UserPoolDomain.fromDomainName(service, "UserPoolDomain", cognitoDomainPrefixParameter.import(service, this.props.basePlatformStackName));
71
+ if (this.props.cognitoUserPool) {
72
+ // Same-stack: use the injected construct directly
73
+ this.userPool = this.props.cognitoUserPool.userPool;
74
+ this.userPoolClient = this.props.cognitoUserPool.appUserPoolClient;
75
+ this.userPoolDomain = this.props.cognitoUserPool.cognitoUserPoolDomain;
76
+ }
77
+ else {
78
+ // Cross-stack: import attributes from the base platform stack
79
+ this.userPool = aws_cognito_1.UserPool.fromUserPoolId(service, "ALBUserPool", new StackParameter_1.StackParameter({ prefix: "cognito", key: "user-pool-id" }).import(service, this.props.basePlatformStackName));
80
+ this.userPoolClient = aws_cognito_1.UserPoolClient.fromUserPoolClientId(service, "UserPoolClient", new StackParameter_1.StackParameter({
81
+ prefix: "cognito",
82
+ key: "user-pool-client-id",
83
+ }).import(service, this.props.basePlatformStackName));
84
+ this.userPoolDomain = aws_cognito_1.UserPoolDomain.fromDomainName(service, "UserPoolDomain", new StackParameter_1.StackParameter({
85
+ prefix: "cognito",
86
+ key: "domain-prefix",
87
+ }).import(service, this.props.basePlatformStackName));
88
+ }
89
89
  }
90
90
  modifyServiceProps(props) {
91
91
  return {
@@ -125,5 +125,5 @@ class ApplicationLoadBalancerExtension extends ecs_service_extensions_1.ServiceE
125
125
  }
126
126
  exports.ApplicationLoadBalancerExtension = ApplicationLoadBalancerExtension;
127
127
  _a = JSII_RTTI_SYMBOL_1;
128
- ApplicationLoadBalancerExtension[_a] = { fqn: "@btc-embedded/cdk-extensions.ApplicationLoadBalancerExtension", version: "0.22.22" };
129
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ApplicationLoadBalancerExtension.js","sourceRoot":"","sources":["../../src/extensions/ApplicationLoadBalancerExtension.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Container,\n  ContainerMutatingHook,\n  Service,\n  ServiceBuild,\n  ServiceExtension,\n} from \"@aws-cdk-containers/ecs-service-extensions\";\nimport { Aws, Duration } from \"aws-cdk-lib\";\nimport {\n  IUserPool,\n  IUserPoolClient,\n  IUserPoolDomain,\n  UserPool,\n  UserPoolClient,\n  UserPoolDomain,\n} from \"aws-cdk-lib/aws-cognito\";\nimport { SecurityGroup } from \"aws-cdk-lib/aws-ec2\";\nimport {\n  ContainerDefinitionOptions,\n  Ec2Service,\n  FargateService,\n} from \"aws-cdk-lib/aws-ecs\";\nimport {\n  ApplicationListener,\n  ApplicationProtocol,\n  ApplicationTargetGroup,\n  IApplicationListener,\n  ListenerAction,\n  ListenerCondition,\n  UnauthenticatedAction,\n} from \"aws-cdk-lib/aws-elasticloadbalancingv2\";\nimport { AuthenticateCognitoAction } from \"aws-cdk-lib/aws-elasticloadbalancingv2-actions\";\nimport { Construct } from \"constructs\";\nimport { StackParameter } from \"../utils/StackParameter\";\n\nexport interface ApplicationLoadBalancerExtensionProps {\n  readonly basePlatformStackName?: string;\n  readonly userPoolClientSecret: string;\n  readonly securityGroupId: string;\n  readonly clientScope: string;\n  readonly healtCheckPath?: string;\n  readonly path?: string; // path to redirect the client after successful authentication\n  readonly listener?: IApplicationListener;\n}\n\ninterface OidcVarsHookProps {\n  readonly issuerUrl: string;\n  readonly clientId: string;\n  readonly clientSecret: string;\n  readonly clientScope: string;\n  readonly redirectUri: string;\n}\n\nclass AddOidcVarsHook extends ContainerMutatingHook {\n  readonly props: OidcVarsHookProps;\n  constructor(props: OidcVarsHookProps) {\n    super();\n    this.props = props;\n  }\n\n  mutateContainerDefinition(\n    props: ContainerDefinitionOptions,\n  ): ContainerDefinitionOptions {\n    return {\n      ...props,\n      environment: {\n        ...props.environment,\n        OIDC_ISSUER_URL: this.props.issuerUrl,\n        OIDC_CLIENT_ID: this.props.clientId,\n        OIDC_CLIENT_SECRET: this.props.clientSecret,\n        OIDC_POST_LOGOUT_REDIRECT_URI: this.props.redirectUri,\n        OIDC_SCOPE: this.props.clientScope,\n      },\n    };\n  }\n}\n/**\n * @deprecated Use {@link ApplicationLoadBalancerExtensionV2} instead\n */\nexport class ApplicationLoadBalancerExtension extends ServiceExtension {\n  private listener!: IApplicationListener;\n  private readonly props: ApplicationLoadBalancerExtensionProps;\n\n  userPool!: IUserPool;\n  userPoolClient!: IUserPoolClient;\n  userPoolDomain!: IUserPoolDomain;\n  fqdn!: string;\n\n  constructor(props: ApplicationLoadBalancerExtensionProps) {\n    super(\"application-load-balancer\");\n    this.props = props;\n  }\n\n  addHooks(): void {\n    const container = this.parentService.serviceDescription.get(\n      \"service-container\",\n    ) as Container;\n\n    if (!container || !container.trafficPort) {\n      throw new Error(\n        \"Cannot apply ALB Extension, no main app container found\",\n      );\n    }\n\n    container.addContainerMutatingHook(\n      new AddOidcVarsHook({\n        clientId: this.userPoolClient.userPoolClientId,\n        clientSecret: this.props.userPoolClientSecret,\n        clientScope: this.props.clientScope,\n        issuerUrl: `https://cognito-idp.${Aws.REGION}.amazonaws.com/${this.userPool.userPoolId}`,\n        redirectUri: `https://${this.fqdn}${this.props.path ?? \"\"}`,\n      }),\n    );\n  }\n\n  prehook(service: Service, _scope: Construct): void {\n    this.parentService = service;\n\n    const fqdnParameter = new StackParameter({\n      key: \"fqdn\",\n      prefix: \"fqdn\",\n    });\n    const albListenerArnParameter = new StackParameter({\n      prefix: [\"alb\"],\n      key: \"listener-arn\",\n    });\n    const cognitoUserPoolIdParameter = new StackParameter({\n      prefix: \"cognito\",\n      key: \"user-pool-id\",\n    });\n    const cognitoUserPoolClientParameter = new StackParameter({\n      prefix: \"cognito\",\n      key: \"user-pool-client-id\",\n    });\n    const cognitoDomainPrefixParameter = new StackParameter({\n      prefix: \"cognito\",\n      key: \"domain-prefix\",\n    });\n\n    this.fqdn = fqdnParameter.import(service, this.props.basePlatformStackName);\n\n    const albListenerArn = albListenerArnParameter.import(\n      service,\n      this.props.basePlatformStackName,\n    );\n\n    if (this.props.listener) {\n      this.listener = this.props.listener;\n    } else {\n      this.listener = ApplicationListener.fromApplicationListenerAttributes(\n        service,\n        \"ALBListener\",\n        {\n          listenerArn: albListenerArn,\n          securityGroup: SecurityGroup.fromSecurityGroupId(\n            service,\n            \"ALBSecurityGroup\",\n            this.props.securityGroupId,\n            { allowAllOutbound: false },\n          ),\n        },\n      );\n    }\n\n    this.userPool = UserPool.fromUserPoolId(\n      service,\n      \"ALBUserPool\",\n      cognitoUserPoolIdParameter.import(\n        service,\n        this.props.basePlatformStackName,\n      ),\n    );\n\n    this.userPoolClient = UserPoolClient.fromUserPoolClientId(\n      service,\n      \"UserPoolClient\",\n      cognitoUserPoolClientParameter.import(\n        service,\n        this.props.basePlatformStackName,\n      ),\n    );\n\n    this.userPoolDomain = UserPoolDomain.fromDomainName(\n      service,\n      \"UserPoolDomain\",\n      cognitoDomainPrefixParameter.import(\n        service,\n        this.props.basePlatformStackName,\n      ),\n    );\n  }\n\n  modifyServiceProps(props: ServiceBuild): ServiceBuild {\n    return {\n      ...props,\n      healthCheckGracePeriod: Duration.minutes(1),\n    } satisfies ServiceBuild;\n  }\n\n  public useService(service: Ec2Service | FargateService): void {\n    const container = this.parentService.serviceDescription.get(\n      \"service-container\",\n    ) as Container;\n    if (!container || !container.trafficPort) {\n      throw new Error(\n        \"Cannot apply ALB Extension, no main app container found\",\n      );\n    }\n\n    const targetGroup = new ApplicationTargetGroup(\n      this.parentService,\n      `${this.parentService.id}TargetGroup`,\n      {\n        vpc: this.parentService.vpc,\n        port: container.trafficPort,\n        protocol: ApplicationProtocol.HTTP,\n        deregistrationDelay: Duration.seconds(10),\n        targets: [service],\n        healthCheck: {\n          path: this.props.healtCheckPath,\n        },\n      },\n    );\n\n    this.listener.addAction(`${this.parentService.id}Action`, {\n      priority: 55,\n      conditions: [ListenerCondition.hostHeaders([this.fqdn])],\n      action: new AuthenticateCognitoAction({\n        userPool: this.userPool,\n        userPoolClient: this.userPoolClient,\n        userPoolDomain: this.userPoolDomain,\n        next: ListenerAction.forward([targetGroup]),\n        onUnauthenticatedRequest: UnauthenticatedAction.AUTHENTICATE,\n      }),\n    });\n\n    this.parentService.targetGroup = targetGroup;\n    this.parentService.addURL(\"public\", `https://${this.fqdn}`);\n  }\n}\n"]}
128
+ ApplicationLoadBalancerExtension[_a] = { fqn: "@btc-embedded/cdk-extensions.ApplicationLoadBalancerExtension", version: "0.22.24" };
129
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ApplicationLoadBalancerExtension.js","sourceRoot":"","sources":["../../src/extensions/ApplicationLoadBalancerExtension.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Container,\n  ContainerMutatingHook,\n  Service,\n  ServiceBuild,\n  ServiceExtension,\n} from \"@aws-cdk-containers/ecs-service-extensions\";\nimport { Aws, Duration } from \"aws-cdk-lib\";\nimport {\n  IUserPool,\n  IUserPoolClient,\n  IUserPoolDomain,\n  UserPool,\n  UserPoolClient,\n  UserPoolDomain,\n} from \"aws-cdk-lib/aws-cognito\";\nimport { SecurityGroup } from \"aws-cdk-lib/aws-ec2\";\nimport {\n  ContainerDefinitionOptions,\n  Ec2Service,\n  FargateService,\n} from \"aws-cdk-lib/aws-ecs\";\nimport {\n  ApplicationListener,\n  ApplicationProtocol,\n  ApplicationTargetGroup,\n  IApplicationListener,\n  ListenerAction,\n  ListenerCondition,\n  UnauthenticatedAction,\n} from \"aws-cdk-lib/aws-elasticloadbalancingv2\";\nimport { AuthenticateCognitoAction } from \"aws-cdk-lib/aws-elasticloadbalancingv2-actions\";\nimport { Construct } from \"constructs\";\nimport { ICognitoUserPool } from \"../platform\";\nimport { StackParameter } from \"../utils/StackParameter\";\n\nexport interface ApplicationLoadBalancerExtensionProps {\n  readonly basePlatformStackName?: string;\n  readonly userPoolClientSecret: string;\n  readonly securityGroupId: string;\n  readonly clientScope: string;\n  readonly healtCheckPath?: string;\n  /** Path to redirect the client after successful authentication. */\n  readonly path?: string;\n  /**\n   * Provide a direct ALB listener reference for same-stack usage.\n   * When omitted the listener ARN is imported from the base platform stack.\n   */\n  readonly listener?: IApplicationListener;\n  /**\n   * Provide a direct Cognito user pool reference for same-stack usage.\n   * When omitted the user pool attributes are imported from the base platform stack.\n   */\n  readonly cognitoUserPool?: ICognitoUserPool;\n  /**\n   * Provide the FQDN directly for same-stack usage.\n   * When omitted the FQDN is imported from the base platform stack.\n   */\n  readonly fqdn?: string;\n}\n\ninterface OidcVarsHookProps {\n  readonly issuerUrl: string;\n  readonly clientId: string;\n  readonly clientSecret: string;\n  readonly clientScope: string;\n  readonly redirectUri: string;\n}\n\nclass AddOidcVarsHook extends ContainerMutatingHook {\n  readonly props: OidcVarsHookProps;\n  constructor(props: OidcVarsHookProps) {\n    super();\n    this.props = props;\n  }\n\n  mutateContainerDefinition(\n    props: ContainerDefinitionOptions,\n  ): ContainerDefinitionOptions {\n    return {\n      ...props,\n      environment: {\n        ...props.environment,\n        OIDC_ISSUER_URL: this.props.issuerUrl,\n        OIDC_CLIENT_ID: this.props.clientId,\n        OIDC_CLIENT_SECRET: this.props.clientSecret,\n        OIDC_POST_LOGOUT_REDIRECT_URI: this.props.redirectUri,\n        OIDC_SCOPE: this.props.clientScope,\n      },\n    };\n  }\n}\n/**\n * @deprecated Use {@link ApplicationLoadBalancerExtensionV2} instead\n */\nexport class ApplicationLoadBalancerExtension extends ServiceExtension {\n  private listener!: IApplicationListener;\n  private readonly props: ApplicationLoadBalancerExtensionProps;\n\n  userPool!: IUserPool;\n  userPoolClient!: IUserPoolClient;\n  userPoolDomain!: IUserPoolDomain;\n  fqdn!: string;\n\n  constructor(props: ApplicationLoadBalancerExtensionProps) {\n    super(\"application-load-balancer\");\n    this.props = props;\n  }\n\n  addHooks(): void {\n    const container = this.parentService.serviceDescription.get(\n      \"service-container\",\n    ) as Container;\n\n    if (!container || !container.trafficPort) {\n      throw new Error(\n        \"Cannot apply ALB Extension, no main app container found\",\n      );\n    }\n\n    container.addContainerMutatingHook(\n      new AddOidcVarsHook({\n        clientId: this.userPoolClient.userPoolClientId,\n        clientSecret: this.props.userPoolClientSecret,\n        clientScope: this.props.clientScope,\n        issuerUrl: `https://cognito-idp.${Aws.REGION}.amazonaws.com/${this.userPool.userPoolId}`,\n        redirectUri: `https://${this.fqdn}${this.props.path ?? \"\"}`,\n      }),\n    );\n  }\n\n  prehook(service: Service, _scope: Construct): void {\n    this.parentService = service;\n\n    // FQDN: use direct value if provided, else import from base platform stack\n    this.fqdn =\n      this.props.fqdn ??\n      new StackParameter({ key: \"fqdn\", prefix: \"fqdn\" }).import(\n        service,\n        this.props.basePlatformStackName,\n      );\n\n    if (this.props.listener) {\n      this.listener = this.props.listener;\n    } else {\n      this.listener = ApplicationListener.fromApplicationListenerAttributes(\n        service,\n        \"ALBListener\",\n        {\n          listenerArn: new StackParameter({\n            prefix: [\"alb\"],\n            key: \"listener-arn\",\n          }).import(service, this.props.basePlatformStackName),\n          securityGroup: SecurityGroup.fromSecurityGroupId(\n            service,\n            \"ALBSecurityGroup\",\n            this.props.securityGroupId,\n            { allowAllOutbound: false },\n          ),\n        },\n      );\n    }\n\n    if (this.props.cognitoUserPool) {\n      // Same-stack: use the injected construct directly\n      this.userPool = this.props.cognitoUserPool.userPool;\n      this.userPoolClient = this.props.cognitoUserPool.appUserPoolClient;\n      this.userPoolDomain = this.props.cognitoUserPool.cognitoUserPoolDomain;\n    } else {\n      // Cross-stack: import attributes from the base platform stack\n      this.userPool = UserPool.fromUserPoolId(\n        service,\n        \"ALBUserPool\",\n        new StackParameter({ prefix: \"cognito\", key: \"user-pool-id\" }).import(\n          service,\n          this.props.basePlatformStackName,\n        ),\n      );\n\n      this.userPoolClient = UserPoolClient.fromUserPoolClientId(\n        service,\n        \"UserPoolClient\",\n        new StackParameter({\n          prefix: \"cognito\",\n          key: \"user-pool-client-id\",\n        }).import(service, this.props.basePlatformStackName),\n      );\n\n      this.userPoolDomain = UserPoolDomain.fromDomainName(\n        service,\n        \"UserPoolDomain\",\n        new StackParameter({\n          prefix: \"cognito\",\n          key: \"domain-prefix\",\n        }).import(service, this.props.basePlatformStackName),\n      );\n    }\n  }\n\n  modifyServiceProps(props: ServiceBuild): ServiceBuild {\n    return {\n      ...props,\n      healthCheckGracePeriod: Duration.minutes(1),\n    } satisfies ServiceBuild;\n  }\n\n  public useService(service: Ec2Service | FargateService): void {\n    const container = this.parentService.serviceDescription.get(\n      \"service-container\",\n    ) as Container;\n    if (!container || !container.trafficPort) {\n      throw new Error(\n        \"Cannot apply ALB Extension, no main app container found\",\n      );\n    }\n\n    const targetGroup = new ApplicationTargetGroup(\n      this.parentService,\n      `${this.parentService.id}TargetGroup`,\n      {\n        vpc: this.parentService.vpc,\n        port: container.trafficPort,\n        protocol: ApplicationProtocol.HTTP,\n        deregistrationDelay: Duration.seconds(10),\n        targets: [service],\n        healthCheck: {\n          path: this.props.healtCheckPath,\n        },\n      },\n    );\n\n    this.listener.addAction(`${this.parentService.id}Action`, {\n      priority: 55,\n      conditions: [ListenerCondition.hostHeaders([this.fqdn])],\n      action: new AuthenticateCognitoAction({\n        userPool: this.userPool,\n        userPoolClient: this.userPoolClient,\n        userPoolDomain: this.userPoolDomain,\n        next: ListenerAction.forward([targetGroup]),\n        onUnauthenticatedRequest: UnauthenticatedAction.AUTHENTICATE,\n      }),\n    });\n\n    this.parentService.targetGroup = targetGroup;\n    this.parentService.addURL(\"public\", `https://${this.fqdn}`);\n  }\n}\n"]}
@@ -1,21 +1,29 @@
1
1
  import { Service, ServiceBuild, ServiceExtension } from "@aws-cdk-containers/ecs-service-extensions";
2
2
  import { Ec2Service, FargateService } from "aws-cdk-lib/aws-ecs";
3
3
  import { Construct } from "constructs";
4
- import { ApplicationLoadBalancerV2 } from "../platform";
4
+ import { IApplicationLoadBalancerV2 } from "../platform";
5
5
  /**
6
6
  * Properties for the ApplicationLoadBalancerExtensionV2.
7
7
  */
8
8
  export interface ApplicationLoadBalancerExtensionPropsV2 {
9
9
  /**
10
10
  * The name of the platform stack to use for the load balancer.
11
- * Either this or `loadBalancer` must be provided.
11
+ *
12
+ * Used when {@link ApplicationLoadBalancerExtensionPropsV2.loadBalancer} is not provided. Falls back to
13
+ * `BasePlatformStackResolver` (CDK context) when also omitted.
12
14
  */
13
15
  readonly basePlatformStackName?: string;
14
16
  /**
15
- * The Application Load Balancer to use.
16
- * Either this or `platformStackName` must be provided.
17
+ * The Application Load Balancer to use directly.
18
+ *
19
+ * When provided, the extension uses this construct as-is and no cross-stack
20
+ * import via {@link ApplicationLoadBalancerV2.fromBasePlatform} is performed.
21
+ * Use this when the ALB resides in the same CDK stack as the ECS service.
22
+ *
23
+ * When omitted, the extension imports the ALB from the base platform stack
24
+ * identified by {@link ApplicationLoadBalancerExtensionPropsV2.basePlatformStackName}.
17
25
  */
18
- readonly loadBalancer?: ApplicationLoadBalancerV2;
26
+ readonly loadBalancer?: IApplicationLoadBalancerV2;
19
27
  }
20
28
  /**
21
29
  * A service extension that adds an Application Load Balancer to an ECS service.
@@ -24,11 +32,23 @@ export declare class ApplicationLoadBalancerExtensionV2 extends ServiceExtension
24
32
  private readonly props;
25
33
  /**
26
34
  * Creates a new ApplicationLoadBalancerExtensionV2.
27
- * @param props - The properties for configuring the load balancer extension.
28
- * @throws Error if neither platformStackName nor loadBalancer is provided.
35
+ *
36
+ * At least one of {@link ApplicationLoadBalancerExtensionPropsV2.basePlatformStackName}
37
+ * or {@link ApplicationLoadBalancerExtensionPropsV2.loadBalancer} must be
38
+ * provided. If neither is given, CDK context
39
+ * (`cdk-extensions:basePlatformStackName`) is used as a final fallback;
40
+ * synthesis will fail if that is also absent.
29
41
  */
30
42
  constructor(props: ApplicationLoadBalancerExtensionPropsV2);
31
43
  prehook(service: Service, _scope: Construct): void;
32
44
  modifyServiceProps(props: ServiceBuild): ServiceBuild;
45
+ /**
46
+ * Wires the ECS service to the Application Load Balancer.
47
+ *
48
+ * Uses the directly injected {@link ApplicationLoadBalancerExtensionPropsV2.loadBalancer}
49
+ * when available (same-stack), otherwise imports from the base platform stack. Falls back to
50
+ * `BasePlatformStackResolver` (CDK context) when
51
+ * {@link ApplicationLoadBalancerExtensionPropsV2.basePlatformStackName} is also absent.
52
+ */
33
53
  useService(service: Ec2Service | FargateService): void;
34
54
  }
@@ -12,14 +12,15 @@ const platform_1 = require("../platform");
12
12
  class ApplicationLoadBalancerExtensionV2 extends ecs_service_extensions_1.ServiceExtension {
13
13
  /**
14
14
  * Creates a new ApplicationLoadBalancerExtensionV2.
15
- * @param props - The properties for configuring the load balancer extension.
16
- * @throws Error if neither platformStackName nor loadBalancer is provided.
15
+ *
16
+ * At least one of {@link ApplicationLoadBalancerExtensionPropsV2.basePlatformStackName}
17
+ * or {@link ApplicationLoadBalancerExtensionPropsV2.loadBalancer} must be
18
+ * provided. If neither is given, CDK context
19
+ * (`cdk-extensions:basePlatformStackName`) is used as a final fallback;
20
+ * synthesis will fail if that is also absent.
17
21
  */
18
22
  constructor(props) {
19
23
  super("application-load-balancer");
20
- if (!props.basePlatformStackName && !props.loadBalancer) {
21
- throw new Error("Either basePlatformStackName or loadBalancer must be provided");
22
- }
23
24
  this.props = props;
24
25
  }
25
26
  prehook(service, _scope) {
@@ -31,6 +32,14 @@ class ApplicationLoadBalancerExtensionV2 extends ecs_service_extensions_1.Servic
31
32
  healthCheckGracePeriod: aws_cdk_lib_1.Duration.minutes(1),
32
33
  };
33
34
  }
35
+ /**
36
+ * Wires the ECS service to the Application Load Balancer.
37
+ *
38
+ * Uses the directly injected {@link ApplicationLoadBalancerExtensionPropsV2.loadBalancer}
39
+ * when available (same-stack), otherwise imports from the base platform stack. Falls back to
40
+ * `BasePlatformStackResolver` (CDK context) when
41
+ * {@link ApplicationLoadBalancerExtensionPropsV2.basePlatformStackName} is also absent.
42
+ */
34
43
  useService(service) {
35
44
  const container = this.parentService.serviceDescription.get("service-container");
36
45
  if (!container || !container.trafficPort) {
@@ -46,5 +55,5 @@ class ApplicationLoadBalancerExtensionV2 extends ecs_service_extensions_1.Servic
46
55
  }
47
56
  exports.ApplicationLoadBalancerExtensionV2 = ApplicationLoadBalancerExtensionV2;
48
57
  _a = JSII_RTTI_SYMBOL_1;
49
- ApplicationLoadBalancerExtensionV2[_a] = { fqn: "@btc-embedded/cdk-extensions.ApplicationLoadBalancerExtensionV2", version: "0.22.22" };
50
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJFeHRlbnNpb25WMi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9leHRlbnNpb25zL0FwcGxpY2F0aW9uTG9hZEJhbGFuY2VyRXh0ZW5zaW9uVjIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFBQSx1RkFLb0Q7QUFDcEQsNkNBQXVDO0FBR3ZDLDBDQUF3RDtBQW1CeEQ7O0dBRUc7QUFDSCxNQUFhLGtDQUFtQyxTQUFRLHlDQUFnQjtJQUd0RTs7OztPQUlHO0lBQ0gsWUFBWSxLQUE4QztRQUN4RCxLQUFLLENBQUMsMkJBQTJCLENBQUMsQ0FBQztRQUVuQyxJQUFJLENBQUMsS0FBSyxDQUFDLHFCQUFxQixJQUFJLENBQUMsS0FBSyxDQUFDLFlBQVksRUFBRSxDQUFDO1lBQ3hELE1BQU0sSUFBSSxLQUFLLENBQ2IsK0RBQStELENBQ2hFLENBQUM7UUFDSixDQUFDO1FBRUQsSUFBSSxDQUFDLEtBQUssR0FBRyxLQUFLLENBQUM7SUFDckIsQ0FBQztJQUVELE9BQU8sQ0FBQyxPQUFnQixFQUFFLE1BQWlCO1FBQ3pDLElBQUksQ0FBQyxhQUFhLEdBQUcsT0FBTyxDQUFDO0lBQy9CLENBQUM7SUFFRCxrQkFBa0IsQ0FBQyxLQUFtQjtRQUNwQyxPQUFPO1lBQ0wsR0FBRyxLQUFLO1lBQ1Isc0JBQXNCLEVBQUUsc0JBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDO1NBQ3JCLENBQUM7SUFDM0IsQ0FBQztJQUVNLFVBQVUsQ0FBQyxPQUFvQztRQUNwRCxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsYUFBYSxDQUFDLGtCQUFrQixDQUFDLEdBQUcsQ0FDekQsbUJBQW1CLENBQ1AsQ0FBQztRQUNmLElBQUksQ0FBQyxTQUFTLElBQUksQ0FBQyxTQUFTLENBQUMsV0FBVyxFQUFFLENBQUM7WUFDekMsTUFBTSxJQUFJLEtBQUssQ0FDYix5REFBeUQsQ0FDMUQsQ0FBQztRQUNKLENBQUM7UUFFRCxNQUFNLEdBQUcsR0FDUCxJQUFJLENBQUMsS0FBSyxDQUFDLFlBQVk7WUFDdkIsb0NBQXlCLENBQUMsZ0JBQWdCLENBQ3hDLE9BQU8sRUFDUCwyQkFBMkIsRUFDM0I7Z0JBQ0UscUJBQXFCLEVBQUUsSUFBSSxDQUFDLEtBQUssQ0FBQyxxQkFBcUI7YUFDeEQsQ0FDRixDQUFDO1FBRUosR0FBRyxDQUFDLFNBQVMsQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUV2QixJQUFJLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBQyxRQUFRLEVBQUUsV0FBVyxHQUFHLENBQUMsVUFBVSxFQUFFLENBQUMsQ0FBQztJQUNuRSxDQUFDOztBQXRESCxnRkF1REMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBDb250YWluZXIsXG4gIFNlcnZpY2UsXG4gIFNlcnZpY2VCdWlsZCxcbiAgU2VydmljZUV4dGVuc2lvbixcbn0gZnJvbSBcIkBhd3MtY2RrLWNvbnRhaW5lcnMvZWNzLXNlcnZpY2UtZXh0ZW5zaW9uc1wiO1xuaW1wb3J0IHsgRHVyYXRpb24gfSBmcm9tIFwiYXdzLWNkay1saWJcIjtcbmltcG9ydCB7IEVjMlNlcnZpY2UsIEZhcmdhdGVTZXJ2aWNlIH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1lY3NcIjtcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gXCJjb25zdHJ1Y3RzXCI7XG5pbXBvcnQgeyBBcHBsaWNhdGlvbkxvYWRCYWxhbmNlclYyIH0gZnJvbSBcIi4uL3BsYXRmb3JtXCI7XG5cbi8qKlxuICogUHJvcGVydGllcyBmb3IgdGhlIEFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyRXh0ZW5zaW9uVjIuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJFeHRlbnNpb25Qcm9wc1YyIHtcbiAgLyoqXG4gICAqIFRoZSBuYW1lIG9mIHRoZSBwbGF0Zm9ybSBzdGFjayB0byB1c2UgZm9yIHRoZSBsb2FkIGJhbGFuY2VyLlxuICAgKiBFaXRoZXIgdGhpcyBvciBgbG9hZEJhbGFuY2VyYCBtdXN0IGJlIHByb3ZpZGVkLlxuICAgKi9cbiAgcmVhZG9ubHkgYmFzZVBsYXRmb3JtU3RhY2tOYW1lPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBUaGUgQXBwbGljYXRpb24gTG9hZCBCYWxhbmNlciB0byB1c2UuXG4gICAqIEVpdGhlciB0aGlzIG9yIGBwbGF0Zm9ybVN0YWNrTmFtZWAgbXVzdCBiZSBwcm92aWRlZC5cbiAgICovXG4gIHJlYWRvbmx5IGxvYWRCYWxhbmNlcj86IEFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyVjI7XG59XG5cbi8qKlxuICogQSBzZXJ2aWNlIGV4dGVuc2lvbiB0aGF0IGFkZHMgYW4gQXBwbGljYXRpb24gTG9hZCBCYWxhbmNlciB0byBhbiBFQ1Mgc2VydmljZS5cbiAqL1xuZXhwb3J0IGNsYXNzIEFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyRXh0ZW5zaW9uVjIgZXh0ZW5kcyBTZXJ2aWNlRXh0ZW5zaW9uIHtcbiAgcHJpdmF0ZSByZWFkb25seSBwcm9wczogQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJFeHRlbnNpb25Qcm9wc1YyO1xuXG4gIC8qKlxuICAgKiBDcmVhdGVzIGEgbmV3IEFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyRXh0ZW5zaW9uVjIuXG4gICAqIEBwYXJhbSBwcm9wcyAtIFRoZSBwcm9wZXJ0aWVzIGZvciBjb25maWd1cmluZyB0aGUgbG9hZCBiYWxhbmNlciBleHRlbnNpb24uXG4gICAqIEB0aHJvd3MgRXJyb3IgaWYgbmVpdGhlciBwbGF0Zm9ybVN0YWNrTmFtZSBub3IgbG9hZEJhbGFuY2VyIGlzIHByb3ZpZGVkLlxuICAgKi9cbiAgY29uc3RydWN0b3IocHJvcHM6IEFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyRXh0ZW5zaW9uUHJvcHNWMikge1xuICAgIHN1cGVyKFwiYXBwbGljYXRpb24tbG9hZC1iYWxhbmNlclwiKTtcblxuICAgIGlmICghcHJvcHMuYmFzZVBsYXRmb3JtU3RhY2tOYW1lICYmICFwcm9wcy5sb2FkQmFsYW5jZXIpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgXCJFaXRoZXIgYmFzZVBsYXRmb3JtU3RhY2tOYW1lIG9yIGxvYWRCYWxhbmNlciBtdXN0IGJlIHByb3ZpZGVkXCIsXG4gICAgICApO1xuICAgIH1cblxuICAgIHRoaXMucHJvcHMgPSBwcm9wcztcbiAgfVxuXG4gIHByZWhvb2soc2VydmljZTogU2VydmljZSwgX3Njb3BlOiBDb25zdHJ1Y3QpOiB2b2lkIHtcbiAgICB0aGlzLnBhcmVudFNlcnZpY2UgPSBzZXJ2aWNlO1xuICB9XG5cbiAgbW9kaWZ5U2VydmljZVByb3BzKHByb3BzOiBTZXJ2aWNlQnVpbGQpOiBTZXJ2aWNlQnVpbGQge1xuICAgIHJldHVybiB7XG4gICAgICAuLi5wcm9wcyxcbiAgICAgIGhlYWx0aENoZWNrR3JhY2VQZXJpb2Q6IER1cmF0aW9uLm1pbnV0ZXMoMSksXG4gICAgfSBzYXRpc2ZpZXMgU2VydmljZUJ1aWxkO1xuICB9XG5cbiAgcHVibGljIHVzZVNlcnZpY2Uoc2VydmljZTogRWMyU2VydmljZSB8IEZhcmdhdGVTZXJ2aWNlKTogdm9pZCB7XG4gICAgY29uc3QgY29udGFpbmVyID0gdGhpcy5wYXJlbnRTZXJ2aWNlLnNlcnZpY2VEZXNjcmlwdGlvbi5nZXQoXG4gICAgICBcInNlcnZpY2UtY29udGFpbmVyXCIsXG4gICAgKSBhcyBDb250YWluZXI7XG4gICAgaWYgKCFjb250YWluZXIgfHwgIWNvbnRhaW5lci50cmFmZmljUG9ydCkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICBcIkNhbm5vdCBhcHBseSBBTEIgRXh0ZW5zaW9uLCBubyBtYWluIGFwcCBjb250YWluZXIgZm91bmRcIixcbiAgICAgICk7XG4gICAgfVxuXG4gICAgY29uc3QgYWxiID1cbiAgICAgIHRoaXMucHJvcHMubG9hZEJhbGFuY2VyID8/XG4gICAgICBBcHBsaWNhdGlvbkxvYWRCYWxhbmNlclYyLmZyb21CYXNlUGxhdGZvcm0oXG4gICAgICAgIHNlcnZpY2UsXG4gICAgICAgIFwiQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJWMlwiLFxuICAgICAgICB7XG4gICAgICAgICAgYmFzZVBsYXRmb3JtU3RhY2tOYW1lOiB0aGlzLnByb3BzLmJhc2VQbGF0Zm9ybVN0YWNrTmFtZSxcbiAgICAgICAgfSxcbiAgICAgICk7XG5cbiAgICBhbGIuYWRkVGFyZ2V0KHNlcnZpY2UpO1xuXG4gICAgdGhpcy5wYXJlbnRTZXJ2aWNlLmFkZFVSTChcInB1YmxpY1wiLCBgaHR0cHM6Ly8ke2FsYi5kb21haW5OYW1lfWApO1xuICB9XG59XG4iXX0=
58
+ ApplicationLoadBalancerExtensionV2[_a] = { fqn: "@btc-embedded/cdk-extensions.ApplicationLoadBalancerExtensionV2", version: "0.22.24" };
59
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJFeHRlbnNpb25WMi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9leHRlbnNpb25zL0FwcGxpY2F0aW9uTG9hZEJhbGFuY2VyRXh0ZW5zaW9uVjIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFBQSx1RkFLb0Q7QUFDcEQsNkNBQXVDO0FBR3ZDLDBDQUdxQjtBQTJCckI7O0dBRUc7QUFDSCxNQUFhLGtDQUFtQyxTQUFRLHlDQUFnQjtJQUd0RTs7Ozs7Ozs7T0FRRztJQUNILFlBQVksS0FBOEM7UUFDeEQsS0FBSyxDQUFDLDJCQUEyQixDQUFDLENBQUM7UUFDbkMsSUFBSSxDQUFDLEtBQUssR0FBRyxLQUFLLENBQUM7SUFDckIsQ0FBQztJQUVELE9BQU8sQ0FBQyxPQUFnQixFQUFFLE1BQWlCO1FBQ3pDLElBQUksQ0FBQyxhQUFhLEdBQUcsT0FBTyxDQUFDO0lBQy9CLENBQUM7SUFFRCxrQkFBa0IsQ0FBQyxLQUFtQjtRQUNwQyxPQUFPO1lBQ0wsR0FBRyxLQUFLO1lBQ1Isc0JBQXNCLEVBQUUsc0JBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDO1NBQ3JCLENBQUM7SUFDM0IsQ0FBQztJQUVEOzs7Ozs7O09BT0c7SUFDSSxVQUFVLENBQUMsT0FBb0M7UUFDcEQsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLGFBQWEsQ0FBQyxrQkFBa0IsQ0FBQyxHQUFHLENBQ3pELG1CQUFtQixDQUNQLENBQUM7UUFDZixJQUFJLENBQUMsU0FBUyxJQUFJLENBQUMsU0FBUyxDQUFDLFdBQVcsRUFBRSxDQUFDO1lBQ3pDLE1BQU0sSUFBSSxLQUFLLENBQ2IseURBQXlELENBQzFELENBQUM7UUFDSixDQUFDO1FBRUQsTUFBTSxHQUFHLEdBQ1AsSUFBSSxDQUFDLEtBQUssQ0FBQyxZQUFZO1lBQ3ZCLG9DQUF5QixDQUFDLGdCQUFnQixDQUN4QyxPQUFPLEVBQ1AsMkJBQTJCLEVBQzNCO2dCQUNFLHFCQUFxQixFQUFFLElBQUksQ0FBQyxLQUFLLENBQUMscUJBQXFCO2FBQ3hELENBQ0YsQ0FBQztRQUVKLEdBQUcsQ0FBQyxTQUFTLENBQUMsT0FBTyxDQUFDLENBQUM7UUFFdkIsSUFBSSxDQUFDLGFBQWEsQ0FBQyxNQUFNLENBQUMsUUFBUSxFQUFFLFdBQVcsR0FBRyxDQUFDLFVBQVUsRUFBRSxDQUFDLENBQUM7SUFDbkUsQ0FBQzs7QUEzREgsZ0ZBNERDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQ29udGFpbmVyLFxuICBTZXJ2aWNlLFxuICBTZXJ2aWNlQnVpbGQsXG4gIFNlcnZpY2VFeHRlbnNpb24sXG59IGZyb20gXCJAYXdzLWNkay1jb250YWluZXJzL2Vjcy1zZXJ2aWNlLWV4dGVuc2lvbnNcIjtcbmltcG9ydCB7IER1cmF0aW9uIH0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQgeyBFYzJTZXJ2aWNlLCBGYXJnYXRlU2VydmljZSB9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtZWNzXCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tIFwiY29uc3RydWN0c1wiO1xuaW1wb3J0IHtcbiAgQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJWMixcbiAgSUFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyVjIsXG59IGZyb20gXCIuLi9wbGF0Zm9ybVwiO1xuXG4vKipcbiAqIFByb3BlcnRpZXMgZm9yIHRoZSBBcHBsaWNhdGlvbkxvYWRCYWxhbmNlckV4dGVuc2lvblYyLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIEFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyRXh0ZW5zaW9uUHJvcHNWMiB7XG4gIC8qKlxuICAgKiBUaGUgbmFtZSBvZiB0aGUgcGxhdGZvcm0gc3RhY2sgdG8gdXNlIGZvciB0aGUgbG9hZCBiYWxhbmNlci5cbiAgICpcbiAgICogVXNlZCB3aGVuIHtAbGluayBBcHBsaWNhdGlvbkxvYWRCYWxhbmNlckV4dGVuc2lvblByb3BzVjIubG9hZEJhbGFuY2VyfSBpcyBub3QgcHJvdmlkZWQuIEZhbGxzIGJhY2sgdG9cbiAgICogYEJhc2VQbGF0Zm9ybVN0YWNrUmVzb2x2ZXJgIChDREsgY29udGV4dCkgd2hlbiBhbHNvIG9taXR0ZWQuXG4gICAqL1xuICByZWFkb25seSBiYXNlUGxhdGZvcm1TdGFja05hbWU/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIFRoZSBBcHBsaWNhdGlvbiBMb2FkIEJhbGFuY2VyIHRvIHVzZSBkaXJlY3RseS5cbiAgICpcbiAgICogV2hlbiBwcm92aWRlZCwgdGhlIGV4dGVuc2lvbiB1c2VzIHRoaXMgY29uc3RydWN0IGFzLWlzIGFuZCBubyBjcm9zcy1zdGFja1xuICAgKiBpbXBvcnQgdmlhIHtAbGluayBBcHBsaWNhdGlvbkxvYWRCYWxhbmNlclYyLmZyb21CYXNlUGxhdGZvcm19IGlzIHBlcmZvcm1lZC5cbiAgICogVXNlIHRoaXMgd2hlbiB0aGUgQUxCIHJlc2lkZXMgaW4gdGhlIHNhbWUgQ0RLIHN0YWNrIGFzIHRoZSBFQ1Mgc2VydmljZS5cbiAgICpcbiAgICogV2hlbiBvbWl0dGVkLCB0aGUgZXh0ZW5zaW9uIGltcG9ydHMgdGhlIEFMQiBmcm9tIHRoZSBiYXNlIHBsYXRmb3JtIHN0YWNrXG4gICAqIGlkZW50aWZpZWQgYnkge0BsaW5rIEFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyRXh0ZW5zaW9uUHJvcHNWMi5iYXNlUGxhdGZvcm1TdGFja05hbWV9LlxuICAgKi9cbiAgcmVhZG9ubHkgbG9hZEJhbGFuY2VyPzogSUFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyVjI7XG59XG5cbi8qKlxuICogQSBzZXJ2aWNlIGV4dGVuc2lvbiB0aGF0IGFkZHMgYW4gQXBwbGljYXRpb24gTG9hZCBCYWxhbmNlciB0byBhbiBFQ1Mgc2VydmljZS5cbiAqL1xuZXhwb3J0IGNsYXNzIEFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyRXh0ZW5zaW9uVjIgZXh0ZW5kcyBTZXJ2aWNlRXh0ZW5zaW9uIHtcbiAgcHJpdmF0ZSByZWFkb25seSBwcm9wczogQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJFeHRlbnNpb25Qcm9wc1YyO1xuXG4gIC8qKlxuICAgKiBDcmVhdGVzIGEgbmV3IEFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyRXh0ZW5zaW9uVjIuXG4gICAqXG4gICAqIEF0IGxlYXN0IG9uZSBvZiB7QGxpbmsgQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJFeHRlbnNpb25Qcm9wc1YyLmJhc2VQbGF0Zm9ybVN0YWNrTmFtZX1cbiAgICogb3Ige0BsaW5rIEFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyRXh0ZW5zaW9uUHJvcHNWMi5sb2FkQmFsYW5jZXJ9IG11c3QgYmVcbiAgICogcHJvdmlkZWQuIElmIG5laXRoZXIgaXMgZ2l2ZW4sIENESyBjb250ZXh0XG4gICAqIChgY2RrLWV4dGVuc2lvbnM6YmFzZVBsYXRmb3JtU3RhY2tOYW1lYCkgaXMgdXNlZCBhcyBhIGZpbmFsIGZhbGxiYWNrO1xuICAgKiBzeW50aGVzaXMgd2lsbCBmYWlsIGlmIHRoYXQgaXMgYWxzbyBhYnNlbnQuXG4gICAqL1xuICBjb25zdHJ1Y3Rvcihwcm9wczogQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJFeHRlbnNpb25Qcm9wc1YyKSB7XG4gICAgc3VwZXIoXCJhcHBsaWNhdGlvbi1sb2FkLWJhbGFuY2VyXCIpO1xuICAgIHRoaXMucHJvcHMgPSBwcm9wcztcbiAgfVxuXG4gIHByZWhvb2soc2VydmljZTogU2VydmljZSwgX3Njb3BlOiBDb25zdHJ1Y3QpOiB2b2lkIHtcbiAgICB0aGlzLnBhcmVudFNlcnZpY2UgPSBzZXJ2aWNlO1xuICB9XG5cbiAgbW9kaWZ5U2VydmljZVByb3BzKHByb3BzOiBTZXJ2aWNlQnVpbGQpOiBTZXJ2aWNlQnVpbGQge1xuICAgIHJldHVybiB7XG4gICAgICAuLi5wcm9wcyxcbiAgICAgIGhlYWx0aENoZWNrR3JhY2VQZXJpb2Q6IER1cmF0aW9uLm1pbnV0ZXMoMSksXG4gICAgfSBzYXRpc2ZpZXMgU2VydmljZUJ1aWxkO1xuICB9XG5cbiAgLyoqXG4gICAqIFdpcmVzIHRoZSBFQ1Mgc2VydmljZSB0byB0aGUgQXBwbGljYXRpb24gTG9hZCBCYWxhbmNlci5cbiAgICpcbiAgICogVXNlcyB0aGUgZGlyZWN0bHkgaW5qZWN0ZWQge0BsaW5rIEFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyRXh0ZW5zaW9uUHJvcHNWMi5sb2FkQmFsYW5jZXJ9XG4gICAqIHdoZW4gYXZhaWxhYmxlIChzYW1lLXN0YWNrKSwgb3RoZXJ3aXNlIGltcG9ydHMgZnJvbSB0aGUgYmFzZSBwbGF0Zm9ybSBzdGFjay4gRmFsbHMgYmFjayB0b1xuICAgKiBgQmFzZVBsYXRmb3JtU3RhY2tSZXNvbHZlcmAgKENESyBjb250ZXh0KSB3aGVuXG4gICAqIHtAbGluayBBcHBsaWNhdGlvbkxvYWRCYWxhbmNlckV4dGVuc2lvblByb3BzVjIuYmFzZVBsYXRmb3JtU3RhY2tOYW1lfSBpcyBhbHNvIGFic2VudC5cbiAgICovXG4gIHB1YmxpYyB1c2VTZXJ2aWNlKHNlcnZpY2U6IEVjMlNlcnZpY2UgfCBGYXJnYXRlU2VydmljZSk6IHZvaWQge1xuICAgIGNvbnN0IGNvbnRhaW5lciA9IHRoaXMucGFyZW50U2VydmljZS5zZXJ2aWNlRGVzY3JpcHRpb24uZ2V0KFxuICAgICAgXCJzZXJ2aWNlLWNvbnRhaW5lclwiLFxuICAgICkgYXMgQ29udGFpbmVyO1xuICAgIGlmICghY29udGFpbmVyIHx8ICFjb250YWluZXIudHJhZmZpY1BvcnQpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgXCJDYW5ub3QgYXBwbHkgQUxCIEV4dGVuc2lvbiwgbm8gbWFpbiBhcHAgY29udGFpbmVyIGZvdW5kXCIsXG4gICAgICApO1xuICAgIH1cblxuICAgIGNvbnN0IGFsYiA9XG4gICAgICB0aGlzLnByb3BzLmxvYWRCYWxhbmNlciA/P1xuICAgICAgQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJWMi5mcm9tQmFzZVBsYXRmb3JtKFxuICAgICAgICBzZXJ2aWNlLFxuICAgICAgICBcIkFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyVjJcIixcbiAgICAgICAge1xuICAgICAgICAgIGJhc2VQbGF0Zm9ybVN0YWNrTmFtZTogdGhpcy5wcm9wcy5iYXNlUGxhdGZvcm1TdGFja05hbWUsXG4gICAgICAgIH0sXG4gICAgICApO1xuXG4gICAgYWxiLmFkZFRhcmdldChzZXJ2aWNlKTtcblxuICAgIHRoaXMucGFyZW50U2VydmljZS5hZGRVUkwoXCJwdWJsaWNcIiwgYGh0dHBzOi8vJHthbGIuZG9tYWluTmFtZX1gKTtcbiAgfVxufVxuIl19
@@ -1,9 +1,13 @@
1
1
  import { ConnectToProps, Service, ServiceBuild, ServiceExtension } from "@aws-cdk-containers/ecs-service-extensions";
2
2
  import { Ec2Service, FargateService } from "aws-cdk-lib/aws-ecs";
3
3
  import { ExportedService } from "../constructs";
4
+ import { IPrivateDnsNamespace } from "../platform";
4
5
  export interface CloudMapExtensionProps {
5
6
  /**
6
7
  * The name of the platform stack to import the namespace parameter details from.
8
+ *
9
+ * Used when {@link privateDnsNamespace} is not provided. Falls back to
10
+ * `BasePlatformStackResolver` (CDK context) when also omitted.
7
11
  */
8
12
  readonly basePlatformStackName?: string;
9
13
  /**
@@ -16,6 +20,17 @@ export interface CloudMapExtensionProps {
16
20
  * @default false
17
21
  */
18
22
  readonly exportService?: boolean;
23
+ /**
24
+ * Private DNS namespace to use directly.
25
+ *
26
+ * When provided, the extension uses this construct as-is and no cross-stack
27
+ * import via {@link PrivateDnsNamespace.fromBasePlatform} is performed. Use
28
+ * this when the namespace resides in the same CDK stack as the ECS service.
29
+ *
30
+ * When omitted, the extension imports the namespace from the base platform
31
+ * stack identified by {@link basePlatformStackName}.
32
+ */
33
+ readonly privateDnsNamespace?: IPrivateDnsNamespace;
19
34
  }
20
35
  export declare class CloudMapExtension extends ServiceExtension {
21
36
  props: CloudMapExtensionProps;
@@ -21,7 +21,11 @@ class CloudMapExtension extends ecs_service_extensions_1.ServiceExtension {
21
21
  throw new Error("You must add a container before configuring the CloudMap extension");
22
22
  }
23
23
  this.serviceName = (0, common_1.kebabCase)(this.parentService.id);
24
- const namespace = platform_1.PrivateDnsNamespace.fromBasePlatform(this.scope, this.serviceName + "PrivateDnsNamespace", this.props.basePlatformStackName);
24
+ // Use the directly injected namespace when available (same-stack usage).
25
+ // Otherwise fall back to cross-stack import, which in turn falls back to
26
+ // BasePlatformStackResolver (CDK context) when basePlatformStackName is absent.
27
+ const namespace = this.props.privateDnsNamespace ??
28
+ platform_1.PrivateDnsNamespace.fromBasePlatform(this.scope, this.serviceName + "PrivateDnsNamespace", this.props.basePlatformStackName);
25
29
  return {
26
30
  ...props,
27
31
  cloudMapOptions: {
@@ -78,5 +82,5 @@ class CloudMapExtension extends ecs_service_extensions_1.ServiceExtension {
78
82
  }
79
83
  exports.CloudMapExtension = CloudMapExtension;
80
84
  _a = JSII_RTTI_SYMBOL_1;
81
- CloudMapExtension[_a] = { fqn: "@btc-embedded/cdk-extensions.CloudMapExtension", version: "0.22.22" };
82
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CloudMapExtension.js","sourceRoot":"","sources":["../../src/extensions/CloudMapExtension.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ConnectToProps,\n  Container,\n  Service,\n  ServiceBuild,\n  ServiceExtension,\n} from \"@aws-cdk-containers/ecs-service-extensions\";\nimport { Duration } from \"aws-cdk-lib\";\nimport { Port } from \"aws-cdk-lib/aws-ec2\";\nimport { Ec2Service, FargateService } from \"aws-cdk-lib/aws-ecs\";\nimport { CfnService, DnsRecordType } from \"aws-cdk-lib/aws-servicediscovery\";\nimport { ExportedService } from \"../constructs\";\nimport { PrivateDnsNamespace } from \"../platform\";\nimport { kebabCase } from \"../utils/common\";\n\nexport interface CloudMapExtensionProps {\n  /**\n   * The name of the platform stack to import the namespace parameter details from.\n   */\n  readonly basePlatformStackName?: string;\n\n  /**\n   * Whether to export this service for cross-stack consumption.\n   *\n   * When enabled, the service metadata (security group, namespace, port) is\n   * exported as stack parameters, allowing other stacks to import and connect\n   * to this service using ExportedService.fromAttributes().\n   *\n   * @default false\n   */\n  readonly exportService?: boolean;\n}\n\nexport class CloudMapExtension extends ServiceExtension {\n  props: CloudMapExtensionProps;\n\n  serviceName!: string;\n\n  /**\n   * The exported service construct, if exportService is enabled.\n   * Available after the service has been configured.\n   */\n  exportedServiceConstruct?: ExportedService;\n\n  constructor(props: CloudMapExtensionProps) {\n    super(\"cloudmap-extension\");\n\n    this.props = props;\n  }\n\n  modifyServiceProps(props: ServiceBuild): ServiceBuild {\n    const mainContainer = props.taskDefinition.findContainer(\"app\");\n\n    if (!mainContainer) {\n      throw new Error(\n        \"You must add a container before configuring the CloudMap extension\",\n      );\n    }\n\n    this.serviceName = kebabCase(this.parentService.id);\n\n    const namespace = PrivateDnsNamespace.fromBasePlatform(\n      this.scope,\n      this.serviceName + \"PrivateDnsNamespace\",\n      this.props.basePlatformStackName,\n    );\n\n    return {\n      ...props,\n      cloudMapOptions: {\n        name: this.serviceName,\n        dnsRecordType: DnsRecordType.SRV,\n        dnsTtl: Duration.seconds(60),\n        containerPort: mainContainer.containerPort,\n        cloudMapNamespace: namespace.namespace,\n      },\n    };\n  }\n\n  useService(service: Ec2Service | FargateService): void {\n    const container = this.parentService.serviceDescription.get(\n      \"service-container\",\n    ) as Container;\n\n    if (!service.cloudMapService) {\n      throw new Error(\"Expected the cloud map settings to be available\");\n    }\n\n    const {\n      cloudMapService: {\n        namespace: { namespaceName },\n      },\n    } = service;\n\n    this.parentService.addURL(\n      \"internal\",\n      `http://${this.serviceName}.${namespaceName}:${container.trafficPort}`,\n    );\n\n    // add hack to support SRV + A records\n\n    const cfnService = service.cloudMapService?.node.defaultChild as CfnService;\n    cfnService.dnsConfig = {\n      ...cfnService.dnsConfig,\n      dnsRecords: [\n        {\n          ttl: 60,\n          type: \"A\",\n        },\n        {\n          ttl: 60,\n          type: \"SRV\",\n        },\n      ],\n    };\n\n    // Export the service if configured\n    if (this.props.exportService) {\n      this.exportedServiceConstruct = new ExportedService(\n        this.scope,\n        `${this.serviceName}ExportedService`,\n        {\n          serviceName: this.serviceName,\n          securityGroup: service.connections.securityGroups[0],\n          namespaceName,\n        },\n      );\n    }\n  }\n\n  /**\n   * Adjust the security group of the given service to allow traffic from this service.\n   *\n   */\n  connectToService(service: Service, _connectToProps?: ConnectToProps): void {\n    const container = service.serviceDescription.get(\n      \"service-container\",\n    ) as Container;\n    if (!container || !container.trafficPort) {\n      throw new Error(\n        \"Cannot apply extension, no main app container found in target service\",\n      );\n    }\n\n    service.ecsService.connections.allowFrom(\n      this.parentService.ecsService,\n      Port.tcp(container.trafficPort),\n    );\n  }\n}\n"]}
85
+ CloudMapExtension[_a] = { fqn: "@btc-embedded/cdk-extensions.CloudMapExtension", version: "0.22.24" };
86
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CloudMapExtension.js","sourceRoot":"","sources":["../../src/extensions/CloudMapExtension.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ConnectToProps,\n  Container,\n  Service,\n  ServiceBuild,\n  ServiceExtension,\n} from \"@aws-cdk-containers/ecs-service-extensions\";\nimport { Duration } from \"aws-cdk-lib\";\nimport { Port } from \"aws-cdk-lib/aws-ec2\";\nimport { Ec2Service, FargateService } from \"aws-cdk-lib/aws-ecs\";\nimport { CfnService, DnsRecordType } from \"aws-cdk-lib/aws-servicediscovery\";\nimport { ExportedService } from \"../constructs\";\nimport { IPrivateDnsNamespace, PrivateDnsNamespace } from \"../platform\";\nimport { kebabCase } from \"../utils/common\";\n\nexport interface CloudMapExtensionProps {\n  /**\n   * The name of the platform stack to import the namespace parameter details from.\n   *\n   * Used when {@link privateDnsNamespace} is not provided. Falls back to\n   * `BasePlatformStackResolver` (CDK context) when also omitted.\n   */\n  readonly basePlatformStackName?: string;\n\n  /**\n   * Whether to export this service for cross-stack consumption.\n   *\n   * When enabled, the service metadata (security group, namespace, port) is\n   * exported as stack parameters, allowing other stacks to import and connect\n   * to this service using ExportedService.fromAttributes().\n   *\n   * @default false\n   */\n  readonly exportService?: boolean;\n\n  /**\n   * Private DNS namespace to use directly.\n   *\n   * When provided, the extension uses this construct as-is and no cross-stack\n   * import via {@link PrivateDnsNamespace.fromBasePlatform} is performed. Use\n   * this when the namespace resides in the same CDK stack as the ECS service.\n   *\n   * When omitted, the extension imports the namespace from the base platform\n   * stack identified by {@link basePlatformStackName}.\n   */\n  readonly privateDnsNamespace?: IPrivateDnsNamespace;\n}\n\nexport class CloudMapExtension extends ServiceExtension {\n  props: CloudMapExtensionProps;\n\n  serviceName!: string;\n\n  /**\n   * The exported service construct, if exportService is enabled.\n   * Available after the service has been configured.\n   */\n  exportedServiceConstruct?: ExportedService;\n\n  constructor(props: CloudMapExtensionProps) {\n    super(\"cloudmap-extension\");\n\n    this.props = props;\n  }\n\n  modifyServiceProps(props: ServiceBuild): ServiceBuild {\n    const mainContainer = props.taskDefinition.findContainer(\"app\");\n\n    if (!mainContainer) {\n      throw new Error(\n        \"You must add a container before configuring the CloudMap extension\",\n      );\n    }\n\n    this.serviceName = kebabCase(this.parentService.id);\n\n    // Use the directly injected namespace when available (same-stack usage).\n    // Otherwise fall back to cross-stack import, which in turn falls back to\n    // BasePlatformStackResolver (CDK context) when basePlatformStackName is absent.\n    const namespace =\n      this.props.privateDnsNamespace ??\n      PrivateDnsNamespace.fromBasePlatform(\n        this.scope,\n        this.serviceName + \"PrivateDnsNamespace\",\n        this.props.basePlatformStackName,\n      );\n\n    return {\n      ...props,\n      cloudMapOptions: {\n        name: this.serviceName,\n        dnsRecordType: DnsRecordType.SRV,\n        dnsTtl: Duration.seconds(60),\n        containerPort: mainContainer.containerPort,\n        cloudMapNamespace: namespace.namespace,\n      },\n    };\n  }\n\n  useService(service: Ec2Service | FargateService): void {\n    const container = this.parentService.serviceDescription.get(\n      \"service-container\",\n    ) as Container;\n\n    if (!service.cloudMapService) {\n      throw new Error(\"Expected the cloud map settings to be available\");\n    }\n\n    const {\n      cloudMapService: {\n        namespace: { namespaceName },\n      },\n    } = service;\n\n    this.parentService.addURL(\n      \"internal\",\n      `http://${this.serviceName}.${namespaceName}:${container.trafficPort}`,\n    );\n\n    // add hack to support SRV + A records\n\n    const cfnService = service.cloudMapService?.node.defaultChild as CfnService;\n    cfnService.dnsConfig = {\n      ...cfnService.dnsConfig,\n      dnsRecords: [\n        {\n          ttl: 60,\n          type: \"A\",\n        },\n        {\n          ttl: 60,\n          type: \"SRV\",\n        },\n      ],\n    };\n\n    // Export the service if configured\n    if (this.props.exportService) {\n      this.exportedServiceConstruct = new ExportedService(\n        this.scope,\n        `${this.serviceName}ExportedService`,\n        {\n          serviceName: this.serviceName,\n          securityGroup: service.connections.securityGroups[0],\n          namespaceName,\n        },\n      );\n    }\n  }\n\n  /**\n   * Adjust the security group of the given service to allow traffic from this service.\n   *\n   */\n  connectToService(service: Service, _connectToProps?: ConnectToProps): void {\n    const container = service.serviceDescription.get(\n      \"service-container\",\n    ) as Container;\n    if (!container || !container.trafficPort) {\n      throw new Error(\n        \"Cannot apply extension, no main app container found in target service\",\n      );\n    }\n\n    service.ecsService.connections.allowFrom(\n      this.parentService.ecsService,\n      Port.tcp(container.trafficPort),\n    );\n  }\n}\n"]}
@@ -28,5 +28,5 @@ class DeactivatableServiceExtension extends ecs_service_extensions_1.ServiceExte
28
28
  }
29
29
  exports.DeactivatableServiceExtension = DeactivatableServiceExtension;
30
30
  _a = JSII_RTTI_SYMBOL_1;
31
- DeactivatableServiceExtension[_a] = { fqn: "@btc-embedded/cdk-extensions.DeactivatableServiceExtension", version: "0.22.22" };
31
+ DeactivatableServiceExtension[_a] = { fqn: "@btc-embedded/cdk-extensions.DeactivatableServiceExtension", version: "0.22.24" };
32
32
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiRGVhY3RpdmF0YWJsZVNlcnZpY2VFeHRlbnNpb24uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZXh0ZW5zaW9ucy9EZWFjdGl2YXRhYmxlU2VydmljZUV4dGVuc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLHVGQUdvRDtBQUNwRCw2Q0FBMEM7QUFDMUMsd0VBQXVFO0FBRXZFOzs7O0dBSUc7QUFDSCxNQUFhLDZCQUE4QixTQUFRLHlDQUFnQjtJQUNqRTtRQUNFLEtBQUssQ0FBQyx1QkFBdUIsQ0FBQyxDQUFDO0lBQ2pDLENBQUM7SUFFRCxrQkFBa0IsQ0FBQyxLQUFtQjtRQUNwQyxJQUFJLElBQUEsNkNBQXNCLEVBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDO1lBQ2xELHlCQUFXLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsQ0FBQyxPQUFPLENBQ3hDLEdBQUcsSUFBSSxDQUFDLGFBQWEsQ0FBQyxFQUFFLDRCQUE0QixDQUNyRCxDQUFDO1lBQ0YsT0FBTztnQkFDTCxHQUFHLEtBQUs7Z0JBQ1IsWUFBWSxFQUFFLENBQUM7YUFDaEIsQ0FBQztRQUNKLENBQUM7UUFDRCxPQUFPLEtBQUssQ0FBQztJQUNmLENBQUM7O0FBaEJILHNFQWlCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIFNlcnZpY2VCdWlsZCxcbiAgU2VydmljZUV4dGVuc2lvbixcbn0gZnJvbSBcIkBhd3MtY2RrLWNvbnRhaW5lcnMvZWNzLXNlcnZpY2UtZXh0ZW5zaW9uc1wiO1xuaW1wb3J0IHsgQW5ub3RhdGlvbnMgfSBmcm9tIFwiYXdzLWNkay1saWJcIjtcbmltcG9ydCB7IGlzRGVhY3RpdmF0ZWRDb25zdHJ1Y3QgfSBmcm9tIFwiLi4vdXRpbHMvZGVhY3RpdmF0ZWRDb25zdHJ1Y3RcIjtcblxuLyoqXG4gKiBFeHRlbnNpb24gdGhhdCBhbGxvd3MgdGhlIHNlcnZpY2UgdG8gYmUgZGVhY3RpdmF0ZWQsIGlmIHRoZSBlbnZpcm9ubWVudCB2YXJpYWJsZSBjb250YWluc1xuICogaXRzIElELiBEZWFjdGl2YXRpb24gaXMgaW1wbGVtZW50ZWQgYnkgc2V0dGluZyB0aGUgZGVzaXJlZCBjb3VudCB0byB6ZXJvLCBpLmUuIHRoZVxuICogY29uc3RydWN0IGlzIHN0aWxsIHByZXNlbnQgaW4gdGhlIHN0YWNrLlxuICovXG5leHBvcnQgY2xhc3MgRGVhY3RpdmF0YWJsZVNlcnZpY2VFeHRlbnNpb24gZXh0ZW5kcyBTZXJ2aWNlRXh0ZW5zaW9uIHtcbiAgY29uc3RydWN0b3IoKSB7XG4gICAgc3VwZXIoXCJkZWFjdGl2YXRhYmxlLXNlcnZpY2VcIik7XG4gIH1cblxuICBtb2RpZnlTZXJ2aWNlUHJvcHMocHJvcHM6IFNlcnZpY2VCdWlsZCk6IFNlcnZpY2VCdWlsZCB7XG4gICAgaWYgKGlzRGVhY3RpdmF0ZWRDb25zdHJ1Y3QodGhpcy5wYXJlbnRTZXJ2aWNlLmlkKSkge1xuICAgICAgQW5ub3RhdGlvbnMub2YodGhpcy5wYXJlbnRTZXJ2aWNlKS5hZGRJbmZvKFxuICAgICAgICBgJHt0aGlzLnBhcmVudFNlcnZpY2UuaWR9IGlzIHNldCB0byBiZSBkZWFjdGl2YXRlZC5gLFxuICAgICAgKTtcbiAgICAgIHJldHVybiB7XG4gICAgICAgIC4uLnByb3BzLFxuICAgICAgICBkZXNpcmVkQ291bnQ6IDAsXG4gICAgICB9O1xuICAgIH1cbiAgICByZXR1cm4gcHJvcHM7XG4gIH1cbn1cbiJdfQ==