UNPKG

@aws/pdk

Version:

All documentation is located at: https://aws.github.io/aws-pdk

110 lines 14.2 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.OpenApiGatewayWebAcl = void 0; /*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved. SPDX-License-Identifier: Apache-2.0 */ const pdk_nag_1 = require("../../../pdk-nag"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_wafv2_1 = require("aws-cdk-lib/aws-wafv2"); const constructs_1 = require("constructs"); /** * Associate an AWS WAF v2 Web ACL with the given api */ class OpenApiGatewayWebAcl extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); const aclName = `${pdk_nag_1.PDKNag.getStackPrefix(aws_cdk_lib_1.Stack.of(this)) .split("/") .join("-")}${id}-${this.node.addr.slice(-8)}`; const ipSetName = `${aclName}-IPSet`; // Create the IP Set if requested this.ipSet = props.cidrAllowList ? new aws_wafv2_1.CfnIPSet(this, "ApiIPSet", { name: ipSetName, addresses: props.cidrAllowList.cidrRanges, ipAddressVersion: props.cidrAllowList.cidrType, scope: "REGIONAL", }) : undefined; // TODO: vendor property is deprecated, to be removed in the future iterations // and vendorName will be required const anyMissingVendor = props.managedRules?.some((q) => !q.vendorName && !q.vendor); if (anyMissingVendor) { throw new Error("The provided managed rules need to define either the vendor or vendorName (preferred) property"); } const managedRules = props.managedRules ?? [ { vendorName: "AWS", name: "AWSManagedRulesCommonRuleSet" }, ]; const rules = [ // Add a rule for the IP Set if specified ...(this.ipSet ? [ { name: ipSetName, priority: 1, visibilityConfig: { metricName: ipSetName, cloudWatchMetricsEnabled: true, sampledRequestsEnabled: true, }, action: { block: {}, }, statement: { notStatement: { statement: { ipSetReferenceStatement: { arn: this.ipSet.attrArn, }, }, }, }, }, ] : []), // Add the managed rules ...managedRules.map(({ vendor, vendorName, name, ...others }, i) => { // TODO: the usage of `vendor` it's for backward compatibility // it will be removed in the next PDK versions const vendorNameToUser = (vendor || vendorName); return { name: `${vendorNameToUser}-${name}`, priority: i + 2, statement: { managedRuleGroupStatement: { ...others, vendorName: vendorNameToUser, name, }, }, overrideAction: { none: {} }, visibilityConfig: { metricName: `${aclName}-${vendorNameToUser}-${name}`, cloudWatchMetricsEnabled: true, sampledRequestsEnabled: true, }, }; }), ]; this.webAcl = new aws_wafv2_1.CfnWebACL(this, "ApiWebACL", { name: aclName, defaultAction: { // Allow by default, and use rules to deny unwanted requests allow: {}, }, scope: "REGIONAL", visibilityConfig: { cloudWatchMetricsEnabled: true, sampledRequestsEnabled: true, metricName: aclName, }, rules, }); this.webAclAssociation = new aws_wafv2_1.CfnWebACLAssociation(this, "WebACLAssociation", { resourceArn: props.apiDeploymentStageArn, webAclArn: this.webAcl.attrArn, }); } } exports.OpenApiGatewayWebAcl = OpenApiGatewayWebAcl; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"open-api-gateway-web-acl.js","sourceRoot":"","sources":["open-api-gateway-web-acl.ts"],"names":[],"mappings":"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","sourcesContent":["/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.\nSPDX-License-Identifier: Apache-2.0 */\nimport { PDKNag } from \"@aws/pdk-nag\";\nimport { Stack } from \"aws-cdk-lib\";\nimport {\n  CfnIPSet,\n  CfnWebACL,\n  CfnWebACLAssociation,\n} from \"aws-cdk-lib/aws-wafv2\";\nimport { Construct } from \"constructs\";\nimport { TypeSafeApiWebAclOptions } from \"./types\";\n\n/**\n * Configuration for the Web ACL for the API\n */\nexport interface OpenApiGatewayWebAclProps extends TypeSafeApiWebAclOptions {\n  /**\n   * The arn of the deployment stage of the API with which the Web ACL will be associated\n   */\n  readonly apiDeploymentStageArn: string;\n}\n\n/**\n * Associate an AWS WAF v2 Web ACL with the given api\n */\nexport class OpenApiGatewayWebAcl extends Construct {\n  readonly webAcl?: CfnWebACL;\n  readonly ipSet?: CfnIPSet;\n  readonly webAclAssociation?: CfnWebACLAssociation;\n\n  constructor(scope: Construct, id: string, props: OpenApiGatewayWebAclProps) {\n    super(scope, id);\n\n    const aclName = `${PDKNag.getStackPrefix(Stack.of(this))\n      .split(\"/\")\n      .join(\"-\")}${id}-${this.node.addr.slice(-8)}`;\n    const ipSetName = `${aclName}-IPSet`;\n\n    // Create the IP Set if requested\n    this.ipSet = props.cidrAllowList\n      ? new CfnIPSet(this, \"ApiIPSet\", {\n          name: ipSetName,\n          addresses: props.cidrAllowList.cidrRanges,\n          ipAddressVersion: props.cidrAllowList.cidrType,\n          scope: \"REGIONAL\",\n        })\n      : undefined;\n\n    // TODO: vendor property is deprecated, to be removed in the future iterations\n    // and vendorName will be required\n    const anyMissingVendor = props.managedRules?.some(\n      (q) => !q.vendorName && !q.vendor\n    );\n\n    if (anyMissingVendor) {\n      throw new Error(\n        \"The provided managed rules need to define either the vendor or vendorName (preferred) property\"\n      );\n    }\n\n    const managedRules = props.managedRules ?? [\n      { vendorName: \"AWS\", name: \"AWSManagedRulesCommonRuleSet\" },\n    ];\n\n    const rules: CfnWebACL.RuleProperty[] = [\n      // Add a rule for the IP Set if specified\n      ...(this.ipSet\n        ? [\n            {\n              name: ipSetName,\n              priority: 1,\n              visibilityConfig: {\n                metricName: ipSetName,\n                cloudWatchMetricsEnabled: true,\n                sampledRequestsEnabled: true,\n              },\n              action: {\n                block: {},\n              },\n              statement: {\n                notStatement: {\n                  statement: {\n                    ipSetReferenceStatement: {\n                      arn: this.ipSet.attrArn,\n                    },\n                  },\n                },\n              },\n            },\n          ]\n        : []),\n      // Add the managed rules\n      ...managedRules.map(\n        (\n          { vendor, vendorName, name, ...others },\n          i\n        ): CfnWebACL.RuleProperty => {\n          // TODO: the usage of `vendor` it's for backward compatibility\n          // it will be removed in the next PDK versions\n          const vendorNameToUser = (vendor || vendorName)!;\n\n          return {\n            name: `${vendorNameToUser}-${name}`,\n            priority: i + 2,\n            statement: {\n              managedRuleGroupStatement: {\n                ...others,\n                vendorName: vendorNameToUser,\n                name,\n              },\n            },\n            overrideAction: { none: {} },\n            visibilityConfig: {\n              metricName: `${aclName}-${vendorNameToUser}-${name}`,\n              cloudWatchMetricsEnabled: true,\n              sampledRequestsEnabled: true,\n            },\n          };\n        }\n      ),\n    ];\n\n    this.webAcl = new CfnWebACL(this, \"ApiWebACL\", {\n      name: aclName,\n      defaultAction: {\n        // Allow by default, and use rules to deny unwanted requests\n        allow: {},\n      },\n      scope: \"REGIONAL\",\n      visibilityConfig: {\n        cloudWatchMetricsEnabled: true,\n        sampledRequestsEnabled: true,\n        metricName: aclName,\n      },\n      rules,\n    });\n\n    this.webAclAssociation = new CfnWebACLAssociation(\n      this,\n      \"WebACLAssociation\",\n      {\n        resourceArn: props.apiDeploymentStageArn,\n        webAclArn: this.webAcl.attrArn,\n      }\n    );\n  }\n}\n"]}