UNPKG

@cdklabs/cdk-amazonmq

Version:
160 lines 25.8 kB
"use strict"; var _a, _b; Object.defineProperty(exports, "__esModule", { value: true }); exports.RabbitMqCustomResource = exports.RabbitMqCustomResourcePolicy = exports.HttpMethods = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); /* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. SPDX-License-Identifier: Apache-2.0 */ const crypto_1 = require("crypto"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_ec2_1 = require("aws-cdk-lib/aws-ec2"); const custom_resources_1 = require("aws-cdk-lib/custom-resources"); const constructs_1 = require("constructs"); const rabbitmq_custom_resource_singleton_function_1 = require("./rabbitmq-custom-resource-singleton-function"); const HASH_LEN = 16; /** * All http request methods */ var HttpMethods; (function (HttpMethods) { HttpMethods["GET"] = "GET"; HttpMethods["POST"] = "POST"; HttpMethods["PUT"] = "PUT"; HttpMethods["DELETE"] = "DELETE"; })(HttpMethods || (exports.HttpMethods = HttpMethods = {})); /** * The IAM Policy that will be applied to the calls. */ class RabbitMqCustomResourcePolicy { /** * Explicit IAM Policy Statements. * * @param statements the statements to propagate to the SDK calls. */ static fromStatements(statements) { return new RabbitMqCustomResourcePolicy(statements); } /** * @param statements statements for explicit policy. * @param resources resources for auto-generated from SDK calls. */ constructor(statements) { this.statements = statements; } } exports.RabbitMqCustomResourcePolicy = RabbitMqCustomResourcePolicy; _a = JSII_RTTI_SYMBOL_1; RabbitMqCustomResourcePolicy[_a] = { fqn: "@cdklabs/cdk-amazonmq.RabbitMqCustomResourcePolicy", version: "0.1.8" }; /** * Use this constant to configure access to any resource. */ RabbitMqCustomResourcePolicy.ANY_RESOURCE = ["*"]; /** * @experimental * * Defines a custom resource that is materialized using specific RabbitMQ Management HTTP API calls. * * Use this to interact with the Amazon MQ for RabbitMQ broker. You can specify exactly which calls are invoked for the 'CREATE', 'UPDATE' and 'DELETE' life cycle events. */ class RabbitMqCustomResource extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); if (!props.onCreate && !props.onUpdate && !props.onDelete) { throw new Error("At least `onCreate`, `onUpdate` or `onDelete` must be specified."); } if (props.onCreate && !props.onCreate.physicalResourceId) { throw new Error("'physicalResourceId' must be specified for 'onCreate' call."); } if (!props.onCreate && props.onUpdate && !props.onUpdate.physicalResourceId) { throw new Error("'physicalResourceId' must be specified for 'onUpdate' call when 'onCreate' is omitted."); } let securityGroups = props.vpc ? props.securityGroups || [ new aws_ec2_1.SecurityGroup(this, "ProviderSG", { vpc: props.vpc }), ] : undefined; const uuid = this.renderUniqueId(props.broker, props.credentials, props.vpc, props.vpcSubnets, props.securityGroups); const provider = new rabbitmq_custom_resource_singleton_function_1.RabbitMqCustomResourceSingletonFunction(this, "Provider", { uuid, vpc: props.vpc, vpcSubnets: props.vpcSubnets, securityGroups: securityGroups, ...(props.logRetention ? { logRetention: props.logRetention } : {}), logGroup: props.logGroup, timeout: props.timeout || aws_cdk_lib_1.Duration.minutes(1), initialPolicy: props.policy?.statements, }); const onUpdate = props.onUpdate && this.formatSdkCall(props.onUpdate); const onCreate = (props.onCreate && this.formatSdkCall(props.onCreate)) || onUpdate; const onDelete = props.onDelete && this.formatSdkCall(props.onDelete); this.resource = new aws_cdk_lib_1.CustomResource(this, `Resource${uuid}`, { resourceType: "Custom::RMQAPI", serviceToken: provider.functionArn, pascalCaseProperties: true, properties: { url: props.broker.endpoints.console.url, credentials: props.credentials.secretArn, create: onCreate, update: onUpdate, delete: onDelete, }, }); this.connections = new aws_ec2_1.Connections({ securityGroups, }); props.credentials.grantRead(provider); this.grantPrincipal = provider.grantPrincipal; } getResponseField(key) { return this.resource.getAttString(key); } getResponseFieldReference(key) { return this.resource.getAtt(key); } formatSdkCall(sdkCall) { const { logging, ...call } = sdkCall; const renderedLogging = (logging ?? custom_resources_1.Logging.all())._render(this); return this.encodeJson({ ...call, ...renderedLogging, }); } encodeJson(obj) { return aws_cdk_lib_1.Lazy.uncachedString({ produce: () => aws_cdk_lib_1.Stack.of(this).toJsonString(obj), }); } renderUniqueId(broker, creds, vpc, subnets, securityGroups) { let hashContent = ""; const resourceBroker = broker; hashContent += aws_cdk_lib_1.Names.uniqueId(resourceBroker); hashContent += aws_cdk_lib_1.Names.uniqueId(creds); if (vpc) { hashContent += aws_cdk_lib_1.Names.uniqueId(vpc); if (subnets) { hashContent += vpc .selectSubnets(subnets) .subnets.map((s) => aws_cdk_lib_1.Names.uniqueId(s)) .join(""); } if (securityGroups) { hashContent += securityGroups.map((sg) => aws_cdk_lib_1.Names.uniqueId(sg)).join(""); } } // INFO: run this through the CDK team as in the S3 Bucket Deployment implementation there is no hashing, just verbatim value addition // see: https://github.com/aws/aws-cdk/blob/318eae6c9eca456e0c34ed21855dad9d2bfa2a0f/packages/aws-cdk-lib/aws-s3-deployment/lib/bucket-deployment.ts#L556 return (0, crypto_1.createHash)("sha256") .update(hashContent) .digest("hex") .slice(0, HASH_LEN) .toUpperCase(); } } exports.RabbitMqCustomResource = RabbitMqCustomResource; _b = JSII_RTTI_SYMBOL_1; RabbitMqCustomResource[_b] = { fqn: "@cdklabs/cdk-amazonmq.RabbitMqCustomResource", version: "0.1.8" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rabbitmq-custom-resource.js","sourceRoot":"","sources":["../../../src/rabbitmq/custom-resource/rabbitmq-custom-resource.ts"],"names":[],"mappings":"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","sourcesContent":["/*\nCopyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\nSPDX-License-Identifier: Apache-2.0\n*/\n\nimport { createHash } from \"crypto\";\nimport {\n  CustomResource,\n  Duration,\n  IResource,\n  Lazy,\n  Names,\n  Reference,\n  Stack,\n} from \"aws-cdk-lib\";\nimport {\n  Connections,\n  IConnectable,\n  ISecurityGroup,\n  IVpc,\n  SecurityGroup,\n  SubnetSelection,\n} from \"aws-cdk-lib/aws-ec2\";\nimport {\n  IGrantable,\n  IPrincipal,\n  IRole,\n  PolicyStatement,\n} from \"aws-cdk-lib/aws-iam\";\nimport { LogGroup, RetentionDays } from \"aws-cdk-lib/aws-logs\";\nimport { ISecret } from \"aws-cdk-lib/aws-secretsmanager\";\nimport { Logging, PhysicalResourceId } from \"aws-cdk-lib/custom-resources\";\nimport { Construct } from \"constructs\";\nimport { IRabbitMqBroker } from \"../rabbitmq-broker\";\nimport { RabbitMqCustomResourceSingletonFunction } from \"./rabbitmq-custom-resource-singleton-function\";\n\nconst HASH_LEN = 16;\n\n/**\n * All http request methods\n */\nexport enum HttpMethods {\n  GET = \"GET\",\n  POST = \"POST\",\n  PUT = \"PUT\",\n  DELETE = \"DELETE\",\n}\n\n/**\n * A RabbitMQ Management HTTP API call\n */\nexport interface RabbitMqApiCall {\n  /**\n   * The RabbitMQ Management HTTP API call path.\n   */\n  readonly path: string;\n\n  /**\n   * The HTTP Method used when invoking the RabbitMQ Management HTTP API call.\n   * @default GET\n   */\n  readonly method?: HttpMethods;\n\n  /**\n   * The payload expected by the RabbitMQ Management HTTP API call.\n   */\n  readonly payload?: { [key: string]: any };\n  /**\n   * The physical resource id of the custom resource for this call.\n   * Mandatory for onCreate call.\n   * In onUpdate, you can omit this to passthrough it from request.\n   *\n   * @default - no physical resource id\n   */\n  readonly physicalResourceId?: PhysicalResourceId;\n\n  /**\n   * Restrict the data returned by the custom resource to specific paths in the API response.\n   *\n   * Use this to limit the data returned by the custom resource if working with API calls that could potentially result in custom response objects exceeding the hard limit of 4096 bytes.\n   */\n  readonly outputPaths?: string[];\n  /**\n   * A property used to configure logging during lambda function execution.\n   *\n   * Note: The default Logging configuration is all. This configuration will enable logging on all logged data\n   * in the lambda handler. This includes:\n   *  - The event object that is received by the lambda handler\n   *  - The response received after making a API call\n   *  - The response object that the lambda handler will return\n   *  - SDK versioning information\n   *  - Caught and uncaught errors\n   *\n   * @default Logging.all()\n   */\n  readonly logging?: Logging;\n}\n\n/**\n * The IAM Policy that will be applied to the calls.\n */\nexport class RabbitMqCustomResourcePolicy {\n  /**\n   * Use this constant to configure access to any resource.\n   */\n  public static readonly ANY_RESOURCE = [\"*\"];\n\n  /**\n   * Explicit IAM Policy Statements.\n   *\n   * @param statements the statements to propagate to the SDK calls.\n   */\n  public static fromStatements(statements: PolicyStatement[]) {\n    return new RabbitMqCustomResourcePolicy(statements);\n  }\n\n  /**\n   * @param statements statements for explicit policy.\n   * @param resources resources for auto-generated from SDK calls.\n   */\n  private constructor(public readonly statements: PolicyStatement[]) {}\n}\n\n/**\n * Properties for RabbitMqCustomResource.\n *\n * Note that at least onCreate, onUpdate or onDelete must be specified.\n */\nexport interface RabbitMqCustomResourceProps {\n  /**\n   * The broker to send requests to.\n   */\n  readonly broker: IRabbitMqBroker;\n\n  /**\n   * The secret containing the broker login credentials.\n   */\n  readonly credentials: ISecret;\n\n  /**\n   * The RabbitMQ Management HTTP API call to make when the resource is created\n   *\n   * @default - the call when the resource is updated\n   */\n  readonly onCreate?: RabbitMqApiCall;\n\n  /**\n   * The RabbitMQ Management HTTP API call to make when the resource is updated\n   *\n   * @default - no call\n   */\n  readonly onUpdate?: RabbitMqApiCall;\n\n  /**\n   * The RabbitMQ Management HTTP API call to make when the resource is updated\n   *\n   * @default - no call\n   */\n  readonly onDelete?: RabbitMqApiCall;\n\n  /**\n   * The vpc to connect to.\n   */\n  readonly vpc?: IVpc;\n\n  /**\n   * The vpc subnets to connect to.\n   */\n  readonly vpcSubnets?: SubnetSelection;\n\n  /**\n   * The security groups to assign to the function.\n   */\n  readonly securityGroups?: SecurityGroup[];\n\n  /**\n   * LogGroup retention to use for the function\n   *\n   * @default RetentionDays.INFINITE\n   * @deprecated use logGroup instead\n   */\n  readonly logRetention?: RetentionDays;\n\n  /**\n   * The logGroup to use for the function\n   */\n  readonly logGroup?: LogGroup;\n\n  /**\n   * The timeout for the custom resource.\n   *\n   * @default Duration.minutes(1)\n   */\n  readonly timeout?: Duration;\n\n  /**\n   * The execution role for the function.\n   */\n  readonly role?: IRole;\n\n  /**\n   * The policies to attach to the function's role\n   */\n  readonly policy?: RabbitMqCustomResourcePolicy;\n}\n\n/**\n * @experimental\n *\n * Defines a custom resource that is materialized using specific RabbitMQ Management HTTP API calls.\n *\n * Use this to interact with the Amazon MQ for RabbitMQ broker. You can specify exactly which calls are invoked for the 'CREATE', 'UPDATE' and 'DELETE' life cycle events.\n */\nexport class RabbitMqCustomResource\n  extends Construct\n  implements IConnectable, IGrantable\n{\n  public readonly connections: Connections;\n\n  public readonly grantPrincipal: IPrincipal;\n\n  private readonly resource: CustomResource;\n\n  constructor(\n    scope: Construct,\n    id: string,\n    props: RabbitMqCustomResourceProps,\n  ) {\n    super(scope, id);\n\n    if (!props.onCreate && !props.onUpdate && !props.onDelete) {\n      throw new Error(\n        \"At least `onCreate`, `onUpdate` or `onDelete` must be specified.\",\n      );\n    }\n\n    if (props.onCreate && !props.onCreate.physicalResourceId) {\n      throw new Error(\n        \"'physicalResourceId' must be specified for 'onCreate' call.\",\n      );\n    }\n\n    if (\n      !props.onCreate &&\n      props.onUpdate &&\n      !props.onUpdate.physicalResourceId\n    ) {\n      throw new Error(\n        \"'physicalResourceId' must be specified for 'onUpdate' call when 'onCreate' is omitted.\",\n      );\n    }\n\n    let securityGroups = props.vpc\n      ? props.securityGroups || [\n          new SecurityGroup(this, \"ProviderSG\", { vpc: props.vpc }),\n        ]\n      : undefined;\n\n    const uuid = this.renderUniqueId(\n      props.broker,\n      props.credentials,\n      props.vpc,\n      props.vpcSubnets,\n      props.securityGroups,\n    );\n\n    const provider = new RabbitMqCustomResourceSingletonFunction(\n      this,\n      \"Provider\",\n      {\n        uuid,\n        vpc: props.vpc,\n        vpcSubnets: props.vpcSubnets,\n        securityGroups: securityGroups,\n        ...(props.logRetention ? { logRetention: props.logRetention } : {}),\n        logGroup: props.logGroup,\n        timeout: props.timeout || Duration.minutes(1),\n        initialPolicy: props.policy?.statements,\n      },\n    );\n\n    const onUpdate = props.onUpdate && this.formatSdkCall(props.onUpdate);\n    const onCreate =\n      (props.onCreate && this.formatSdkCall(props.onCreate)) || onUpdate;\n    const onDelete = props.onDelete && this.formatSdkCall(props.onDelete);\n\n    this.resource = new CustomResource(this, `Resource${uuid}`, {\n      resourceType: \"Custom::RMQAPI\",\n      serviceToken: provider.functionArn,\n      pascalCaseProperties: true,\n      properties: {\n        url: props.broker.endpoints.console.url,\n        credentials: props.credentials.secretArn,\n        create: onCreate,\n        update: onUpdate,\n        delete: onDelete,\n      },\n    });\n\n    this.connections = new Connections({\n      securityGroups,\n    });\n\n    props.credentials.grantRead(provider);\n\n    this.grantPrincipal = provider.grantPrincipal;\n  }\n\n  public getResponseField(key: string): string {\n    return this.resource.getAttString(key);\n  }\n\n  public getResponseFieldReference(key: string): Reference {\n    return this.resource.getAtt(key);\n  }\n\n  private formatSdkCall(sdkCall: RabbitMqApiCall) {\n    const { logging, ...call } = sdkCall;\n    const renderedLogging = (logging ?? Logging.all())._render(this);\n    return this.encodeJson({\n      ...call,\n      ...renderedLogging,\n    });\n  }\n\n  private encodeJson(obj: any) {\n    return Lazy.uncachedString({\n      produce: () => Stack.of(this).toJsonString(obj),\n    });\n  }\n\n  private renderUniqueId(\n    broker: IRabbitMqBroker,\n    creds: ISecret,\n    vpc?: IVpc,\n    subnets?: SubnetSelection,\n    securityGroups?: ISecurityGroup[],\n  ) {\n    let hashContent = \"\";\n    const resourceBroker = broker as IResource;\n    hashContent += Names.uniqueId(resourceBroker);\n    hashContent += Names.uniqueId(creds);\n    if (vpc) {\n      hashContent += Names.uniqueId(vpc);\n      if (subnets) {\n        hashContent += vpc\n          .selectSubnets(subnets)\n          .subnets.map((s) => Names.uniqueId(s))\n          .join(\"\");\n      }\n      if (securityGroups) {\n        hashContent += securityGroups.map((sg) => Names.uniqueId(sg)).join(\"\");\n      }\n    }\n\n    // INFO: run this through the CDK team as in the S3 Bucket Deployment implementation there is no hashing, just verbatim value addition\n    // see: https://github.com/aws/aws-cdk/blob/318eae6c9eca456e0c34ed21855dad9d2bfa2a0f/packages/aws-cdk-lib/aws-s3-deployment/lib/bucket-deployment.ts#L556\n\n    return createHash(\"sha256\")\n      .update(hashContent)\n      .digest(\"hex\")\n      .slice(0, HASH_LEN)\n      .toUpperCase();\n  }\n}\n"]}