@aws/pdk
Version:
All documentation is located at: https://aws.github.io/aws-pdk
110 lines • 14.2 kB
JavaScript
;
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":";;;AAAA;sCACsC;AACtC,0CAAsC;AACtC,6CAAoC;AACpC,qDAI+B;AAC/B,2CAAuC;AAavC;;GAEG;AACH,MAAa,oBAAqB,SAAQ,sBAAS;IAKjD,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAgC;QACxE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,OAAO,GAAG,GAAG,gBAAM,CAAC,cAAc,CAAC,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;aACrD,KAAK,CAAC,GAAG,CAAC;aACV,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QAChD,MAAM,SAAS,GAAG,GAAG,OAAO,QAAQ,CAAC;QAErC,iCAAiC;QACjC,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC,aAAa;YAC9B,CAAC,CAAC,IAAI,oBAAQ,CAAC,IAAI,EAAE,UAAU,EAAE;gBAC7B,IAAI,EAAE,SAAS;gBACf,SAAS,EAAE,KAAK,CAAC,aAAa,CAAC,UAAU;gBACzC,gBAAgB,EAAE,KAAK,CAAC,aAAa,CAAC,QAAQ;gBAC9C,KAAK,EAAE,UAAU;aAClB,CAAC;YACJ,CAAC,CAAC,SAAS,CAAC;QAEd,8EAA8E;QAC9E,kCAAkC;QAClC,MAAM,gBAAgB,GAAG,KAAK,CAAC,YAAY,EAAE,IAAI,CAC/C,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,UAAU,IAAI,CAAC,CAAC,CAAC,MAAM,CAClC,CAAC;QAEF,IAAI,gBAAgB,EAAE,CAAC;YACrB,MAAM,IAAI,KAAK,CACb,gGAAgG,CACjG,CAAC;QACJ,CAAC;QAED,MAAM,YAAY,GAAG,KAAK,CAAC,YAAY,IAAI;YACzC,EAAE,UAAU,EAAE,KAAK,EAAE,IAAI,EAAE,8BAA8B,EAAE;SAC5D,CAAC;QAEF,MAAM,KAAK,GAA6B;YACtC,yCAAyC;YACzC,GAAG,CAAC,IAAI,CAAC,KAAK;gBACZ,CAAC,CAAC;oBACE;wBACE,IAAI,EAAE,SAAS;wBACf,QAAQ,EAAE,CAAC;wBACX,gBAAgB,EAAE;4BAChB,UAAU,EAAE,SAAS;4BACrB,wBAAwB,EAAE,IAAI;4BAC9B,sBAAsB,EAAE,IAAI;yBAC7B;wBACD,MAAM,EAAE;4BACN,KAAK,EAAE,EAAE;yBACV;wBACD,SAAS,EAAE;4BACT,YAAY,EAAE;gCACZ,SAAS,EAAE;oCACT,uBAAuB,EAAE;wCACvB,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO;qCACxB;iCACF;6BACF;yBACF;qBACF;iBACF;gBACH,CAAC,CAAC,EAAE,CAAC;YACP,wBAAwB;YACxB,GAAG,YAAY,CAAC,GAAG,CACjB,CACE,EAAE,MAAM,EAAE,UAAU,EAAE,IAAI,EAAE,GAAG,MAAM,EAAE,EACvC,CAAC,EACuB,EAAE;gBAC1B,8DAA8D;gBAC9D,8CAA8C;gBAC9C,MAAM,gBAAgB,GAAG,CAAC,MAAM,IAAI,UAAU,CAAE,CAAC;gBAEjD,OAAO;oBACL,IAAI,EAAE,GAAG,gBAAgB,IAAI,IAAI,EAAE;oBACnC,QAAQ,EAAE,CAAC,GAAG,CAAC;oBACf,SAAS,EAAE;wBACT,yBAAyB,EAAE;4BACzB,GAAG,MAAM;4BACT,UAAU,EAAE,gBAAgB;4BAC5B,IAAI;yBACL;qBACF;oBACD,cAAc,EAAE,EAAE,IAAI,EAAE,EAAE,EAAE;oBAC5B,gBAAgB,EAAE;wBAChB,UAAU,EAAE,GAAG,OAAO,IAAI,gBAAgB,IAAI,IAAI,EAAE;wBACpD,wBAAwB,EAAE,IAAI;wBAC9B,sBAAsB,EAAE,IAAI;qBAC7B;iBACF,CAAC;YACJ,CAAC,CACF;SACF,CAAC;QAEF,IAAI,CAAC,MAAM,GAAG,IAAI,qBAAS,CAAC,IAAI,EAAE,WAAW,EAAE;YAC7C,IAAI,EAAE,OAAO;YACb,aAAa,EAAE;gBACb,4DAA4D;gBAC5D,KAAK,EAAE,EAAE;aACV;YACD,KAAK,EAAE,UAAU;YACjB,gBAAgB,EAAE;gBAChB,wBAAwB,EAAE,IAAI;gBAC9B,sBAAsB,EAAE,IAAI;gBAC5B,UAAU,EAAE,OAAO;aACpB;YACD,KAAK;SACN,CAAC,CAAC;QAEH,IAAI,CAAC,iBAAiB,GAAG,IAAI,gCAAoB,CAC/C,IAAI,EACJ,mBAAmB,EACnB;YACE,WAAW,EAAE,KAAK,CAAC,qBAAqB;YACxC,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO;SAC/B,CACF,CAAC;IACJ,CAAC;CACF;AAzHD,oDAyHC","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"]}