@aws/pdk
Version:
All documentation is located at: https://aws.github.io/aws-pdk
195 lines • 30.2 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.CloudfrontWebAcl = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.
SPDX-License-Identifier: Apache-2.0 */
const path = require("path");
const pdk_nag_1 = require("../pdk-nag");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
const custom_resources_1 = require("aws-cdk-lib/custom-resources");
const cdk_nag_1 = require("cdk-nag");
const constructs_1 = require("constructs");
/**
* This construct creates a WAFv2 Web ACL for cloudfront in the us-east-1 region (required for cloudfront) no matter the
* region of the parent cdk stack.
*/
class CloudfrontWebAcl extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
const stack = aws_cdk_lib_1.Stack.of(this);
const aclName = `${stack.stackName}-${id}-${this.node.addr.slice(-4)}`;
const onEventHandler = this.createOnEventHandler(stack, aclName);
const customResource = this.createAclCustomResource(stack, aclName, onEventHandler, props);
this.webAclId = customResource.getAttString("WebAclId");
this.webAclArn = customResource.getAttString("WebAclArn");
}
/**
* Creates an event handler for managing an ACL in us-east-1.
*
* @param stack containing Stack instance.
* @param aclName name of the ACL to manage.
* @private
*/
createOnEventHandler(stack, aclName) {
// NB without manually defining a name, the cdk generated name for the Provider function can become too long and
// deployments fail. This is because the Provider's name references the onEvent handler name and appends "-Provider"
// rather than being generated by cdk and truncated appropriately
const onEventHandlerName = `${pdk_nag_1.PDKNag.getStackPrefix(stack)
.split("/")
.join("-")}AclEvent-${this.node.addr.slice(-6)}`;
const onEventHandlerRole = new aws_iam_1.Role(this, "OnEventHandlerRole", {
assumedBy: new aws_iam_1.ServicePrincipal("lambda.amazonaws.com"),
inlinePolicies: {
logs: new aws_iam_1.PolicyDocument({
statements: [
new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.ALLOW,
actions: [
"logs:CreateLogGroup",
"logs:CreateLogStream",
"logs:PutLogEvents",
],
resources: [
`arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/${onEventHandlerName}`,
`arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/${onEventHandlerName}:*`,
],
}),
],
}),
wafv2: new aws_iam_1.PolicyDocument({
statements: [
new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.ALLOW,
actions: [
"wafv2:CreateWebACL",
"wafv2:DeleteWebACL",
"wafv2:UpdateWebACL",
"wafv2:GetWebACL",
],
resources: [
`arn:aws:wafv2:us-east-1:${stack.account}:global/ipset/${aclName}-IPSet/*`,
`arn:aws:wafv2:us-east-1:${stack.account}:global/webacl/${aclName}/*`,
`arn:aws:wafv2:us-east-1:${stack.account}:global/managedruleset/*/*`,
],
}),
new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.ALLOW,
actions: [
"wafv2:CreateIPSet",
"wafv2:DeleteIPSet",
"wafv2:UpdateIPSet",
"wafv2:GetIPSet",
],
resources: [
`arn:aws:wafv2:us-east-1:${stack.account}:global/ipset/${aclName}-IPSet/*`,
],
}),
],
}),
},
});
const onEventHandler = new aws_lambda_1.Function(this, "CloudfrontWebAclOnEventHandler", {
code: aws_lambda_1.Code.fromAsset(path.join(__dirname, "./webacl_event_handler")),
role: onEventHandlerRole,
functionName: onEventHandlerName,
handler: "index.onEvent",
runtime: aws_lambda_1.Runtime.NODEJS_18_X,
timeout: aws_cdk_lib_1.Duration.seconds(300),
});
["AwsSolutions-IAM5", "AwsPrototyping-IAMNoWildcardPermissions"].forEach((RuleId) => {
cdk_nag_1.NagSuppressions.addResourceSuppressions(onEventHandlerRole, [
{
id: RuleId,
reason: "WafV2 resources have been scoped down to the ACL/IPSet level, however * is still needed as resource id's are created just in time.",
appliesTo: [
{
regex: `/^Resource::arn:aws:wafv2:us-east-1:${pdk_nag_1.PDKNag.getStackAccountRegex(stack)}:global/(.*)$/g`,
},
],
},
{
id: RuleId,
reason: "Cloudwatch resources have been scoped down to the LogGroup level, however * is still needed as stream names are created just in time.",
appliesTo: [
{
regex: `/^Resource::arn:aws:logs:${pdk_nag_1.PDKNag.getStackRegionRegex(stack)}:${pdk_nag_1.PDKNag.getStackAccountRegex(stack)}:log-group:/aws/lambda/${onEventHandlerName}:\*/g`,
},
],
},
], true);
});
return onEventHandler;
}
/**
* Creates a Custom resource to manage the deployment of the ACL.
*
* @param stack containing Stack instance.
* @param aclName name of the ACL to manage.
* @param onEventHandler event handler to use for deployment.
* @param props user provided properties for configuring the ACL.
* @private
*/
createAclCustomResource(stack, aclName, onEventHandler, props) {
const providerFunctionName = `${onEventHandler.functionName}-Provider`;
const providerRole = new aws_iam_1.Role(this, "CloudfrontWebAclProviderRole", {
assumedBy: new aws_iam_1.ServicePrincipal("lambda.amazonaws.com"),
inlinePolicies: {
logs: new aws_iam_1.PolicyDocument({
statements: [
new aws_iam_1.PolicyStatement({
effect: aws_iam_1.Effect.ALLOW,
actions: [
"logs:CreateLogGroup",
"logs:CreateLogStream",
"logs:PutLogEvents",
],
resources: [
`arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/${providerFunctionName}`,
`arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/${providerFunctionName}:*`,
],
}),
],
}),
},
});
const provider = new custom_resources_1.Provider(this, "CloudfrontAclProvider", {
onEventHandler,
role: providerRole,
providerFunctionName,
});
["AwsSolutions-IAM5", "AwsPrototyping-IAMNoWildcardPermissions"].forEach((RuleId) => {
cdk_nag_1.NagSuppressions.addResourceSuppressions(providerRole, [
{
id: RuleId,
reason: "Cloudwatch resources have been scoped down to the LogGroup level, however * is still needed as stream names are created just in time.",
},
], true);
});
["AwsSolutions-L1", "AwsPrototyping-LambdaLatestVersion"].forEach((RuleId) => {
cdk_nag_1.NagSuppressions.addResourceSuppressions(provider, [
{
id: RuleId,
reason: "Latest runtime cannot be configured. CDK will need to upgrade the Provider construct accordingly.",
},
], true);
});
return new aws_cdk_lib_1.CustomResource(this, "CFAclCustomResource", {
serviceToken: provider.serviceToken,
properties: {
ID: aclName,
MANAGED_RULES: props?.managedRules ?? [
{ vendor: "AWS", name: "AWSManagedRulesCommonRuleSet" },
],
CIDR_ALLOW_LIST: props?.cidrAllowList,
},
});
}
}
exports.CloudfrontWebAcl = CloudfrontWebAcl;
_a = JSII_RTTI_SYMBOL_1;
CloudfrontWebAcl[_a] = { fqn: "@aws/pdk.static_website.CloudfrontWebAcl", version: "0.26.14" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudfront-web-acl.js","sourceRoot":"","sources":["cloudfront-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 * as path from \"path\";\nimport { PDKNag } from \"@aws/pdk-nag\";\nimport { CustomResource, Duration, Stack } from \"aws-cdk-lib\";\nimport {\n  Effect,\n  PolicyDocument,\n  PolicyStatement,\n  Role,\n  ServicePrincipal,\n} from \"aws-cdk-lib/aws-iam\";\nimport { Code, Function, Runtime } from \"aws-cdk-lib/aws-lambda\";\nimport { Provider } from \"aws-cdk-lib/custom-resources\";\nimport { NagSuppressions } from \"cdk-nag\";\nimport { Construct } from \"constructs\";\n\n/**\n * Represents a WAF V2 managed rule.\n */\nexport interface ManagedRule {\n  /**\n   * The name of the managed rule group vendor. You use this, along with the rule group name, to identify the rule group.\n   */\n  readonly vendor: string;\n\n  /**\n   * The name of the managed rule group. You use this, along with the vendor name, to identify the rule group.\n   */\n  readonly name: string;\n}\n\n/**\n * Type of Cidr.\n */\nexport type CidrType = \"IPV4\" | \"IPV6\";\n\n/**\n * Representation of a CIDR range.\n */\nexport interface CidrAllowList {\n  /**\n   * Type of CIDR range.\n   */\n  readonly cidrType: CidrType;\n\n  /**\n   * Specify an IPv4 address by using CIDR notation. For example:\n   * To configure AWS WAF to allow, block, or count requests that originated from the IP address 192.0.2.44, specify 192.0.2.44/32 .\n   * To configure AWS WAF to allow, block, or count requests that originated from IP addresses from 192.0.2.0 to 192.0.2.255, specify 192.0.2.0/24 .\n   *\n   * For more information about CIDR notation, see the Wikipedia entry Classless Inter-Domain Routing .\n   *\n   * Specify an IPv6 address by using CIDR notation. For example:\n   * To configure AWS WAF to allow, block, or count requests that originated from the IP address 1111:0000:0000:0000:0000:0000:0000:0111, specify 1111:0000:0000:0000:0000:0000:0000:0111/128 .\n   * To configure AWS WAF to allow, block, or count requests that originated from IP addresses 1111:0000:0000:0000:0000:0000:0000:0000 to 1111:0000:0000:0000:ffff:ffff:ffff:ffff, specify 1111:0000:0000:0000:0000:0000:0000:0000/64 .\n   */\n  readonly cidrRanges: string[];\n}\n\n/**\n * Properties to configure the web acl.\n */\nexport interface CloudFrontWebAclProps {\n  /**\n   * List of managed rules to apply to the web acl.\n   *\n   * @default - [{ vendor: \"AWS\", name: \"AWSManagedRulesCommonRuleSet\" }]\n   */\n  readonly managedRules?: ManagedRule[];\n\n  /**\n   * List of cidr ranges to allow.\n   *\n   * @default - undefined\n   */\n  readonly cidrAllowList?: CidrAllowList;\n\n  /**\n   * Set to true to prevent creation of a web acl for the static website\n   * @default false\n   */\n  readonly disable?: boolean;\n}\n\n/**\n * This construct creates a WAFv2 Web ACL for cloudfront in the us-east-1 region (required for cloudfront) no matter the\n * region of the parent cdk stack.\n */\nexport class CloudfrontWebAcl extends Construct {\n  public readonly webAclId: string;\n  public readonly webAclArn: string;\n\n  constructor(scope: Construct, id: string, props?: CloudFrontWebAclProps) {\n    super(scope, id);\n\n    const stack = Stack.of(this);\n    const aclName = `${stack.stackName}-${id}-${this.node.addr.slice(-4)}`;\n    const onEventHandler = this.createOnEventHandler(stack, aclName);\n    const customResource = this.createAclCustomResource(\n      stack,\n      aclName,\n      onEventHandler,\n      props\n    );\n\n    this.webAclId = customResource.getAttString(\"WebAclId\");\n    this.webAclArn = customResource.getAttString(\"WebAclArn\");\n  }\n\n  /**\n   * Creates an event handler for managing an ACL in us-east-1.\n   *\n   * @param stack containing Stack instance.\n   * @param aclName name of the ACL to manage.\n   * @private\n   */\n  private createOnEventHandler(stack: Stack, aclName: string): Function {\n    // NB without manually defining a name, the cdk generated name for the Provider function can become too long and\n    // deployments fail. This is because the Provider's name references the onEvent handler name and appends \"-Provider\"\n    // rather than being generated by cdk and truncated appropriately\n    const onEventHandlerName = `${PDKNag.getStackPrefix(stack)\n      .split(\"/\")\n      .join(\"-\")}AclEvent-${this.node.addr.slice(-6)}`;\n    const onEventHandlerRole = new Role(this, \"OnEventHandlerRole\", {\n      assumedBy: new ServicePrincipal(\"lambda.amazonaws.com\"),\n      inlinePolicies: {\n        logs: new PolicyDocument({\n          statements: [\n            new PolicyStatement({\n              effect: Effect.ALLOW,\n              actions: [\n                \"logs:CreateLogGroup\",\n                \"logs:CreateLogStream\",\n                \"logs:PutLogEvents\",\n              ],\n              resources: [\n                `arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/${onEventHandlerName}`,\n                `arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/${onEventHandlerName}:*`,\n              ],\n            }),\n          ],\n        }),\n        wafv2: new PolicyDocument({\n          statements: [\n            new PolicyStatement({\n              effect: Effect.ALLOW,\n              actions: [\n                \"wafv2:CreateWebACL\",\n                \"wafv2:DeleteWebACL\",\n                \"wafv2:UpdateWebACL\",\n                \"wafv2:GetWebACL\",\n              ],\n              resources: [\n                `arn:aws:wafv2:us-east-1:${stack.account}:global/ipset/${aclName}-IPSet/*`,\n                `arn:aws:wafv2:us-east-1:${stack.account}:global/webacl/${aclName}/*`,\n                `arn:aws:wafv2:us-east-1:${stack.account}:global/managedruleset/*/*`,\n              ],\n            }),\n            new PolicyStatement({\n              effect: Effect.ALLOW,\n              actions: [\n                \"wafv2:CreateIPSet\",\n                \"wafv2:DeleteIPSet\",\n                \"wafv2:UpdateIPSet\",\n                \"wafv2:GetIPSet\",\n              ],\n              resources: [\n                `arn:aws:wafv2:us-east-1:${stack.account}:global/ipset/${aclName}-IPSet/*`,\n              ],\n            }),\n          ],\n        }),\n      },\n    });\n\n    const onEventHandler = new Function(\n      this,\n      \"CloudfrontWebAclOnEventHandler\",\n      {\n        code: Code.fromAsset(path.join(__dirname, \"./webacl_event_handler\")),\n        role: onEventHandlerRole,\n        functionName: onEventHandlerName,\n        handler: \"index.onEvent\",\n        runtime: Runtime.NODEJS_18_X,\n        timeout: Duration.seconds(300),\n      }\n    );\n\n    [\"AwsSolutions-IAM5\", \"AwsPrototyping-IAMNoWildcardPermissions\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          onEventHandlerRole,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"WafV2 resources have been scoped down to the ACL/IPSet level, however * is still needed as resource id's are created just in time.\",\n              appliesTo: [\n                {\n                  regex: `/^Resource::arn:aws:wafv2:us-east-1:${PDKNag.getStackAccountRegex(\n                    stack\n                  )}:global/(.*)$/g`,\n                },\n              ],\n            },\n            {\n              id: RuleId,\n              reason:\n                \"Cloudwatch resources have been scoped down to the LogGroup level, however * is still needed as stream names are created just in time.\",\n              appliesTo: [\n                {\n                  regex: `/^Resource::arn:aws:logs:${PDKNag.getStackRegionRegex(\n                    stack\n                  )}:${PDKNag.getStackAccountRegex(\n                    stack\n                  )}:log-group:/aws/lambda/${onEventHandlerName}:\\*/g`,\n                },\n              ],\n            },\n          ],\n          true\n        );\n      }\n    );\n\n    return onEventHandler;\n  }\n\n  /**\n   * Creates a Custom resource to manage the deployment of the ACL.\n   *\n   * @param stack containing Stack instance.\n   * @param aclName name of the ACL to manage.\n   * @param onEventHandler event handler to use for deployment.\n   * @param props user provided properties for configuring the ACL.\n   * @private\n   */\n  private createAclCustomResource(\n    stack: Stack,\n    aclName: string,\n    onEventHandler: Function,\n    props?: CloudFrontWebAclProps\n  ): CustomResource {\n    const providerFunctionName = `${onEventHandler.functionName}-Provider`;\n    const providerRole = new Role(this, \"CloudfrontWebAclProviderRole\", {\n      assumedBy: new ServicePrincipal(\"lambda.amazonaws.com\"),\n      inlinePolicies: {\n        logs: new PolicyDocument({\n          statements: [\n            new PolicyStatement({\n              effect: Effect.ALLOW,\n              actions: [\n                \"logs:CreateLogGroup\",\n                \"logs:CreateLogStream\",\n                \"logs:PutLogEvents\",\n              ],\n              resources: [\n                `arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/${providerFunctionName}`,\n                `arn:aws:logs:${stack.region}:${stack.account}:log-group:/aws/lambda/${providerFunctionName}:*`,\n              ],\n            }),\n          ],\n        }),\n      },\n    });\n    const provider = new Provider(this, \"CloudfrontAclProvider\", {\n      onEventHandler,\n      role: providerRole,\n      providerFunctionName,\n    });\n\n    [\"AwsSolutions-IAM5\", \"AwsPrototyping-IAMNoWildcardPermissions\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          providerRole,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"Cloudwatch resources have been scoped down to the LogGroup level, however * is still needed as stream names are created just in time.\",\n            },\n          ],\n          true\n        );\n      }\n    );\n\n    [\"AwsSolutions-L1\", \"AwsPrototyping-LambdaLatestVersion\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          provider,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"Latest runtime cannot be configured. CDK will need to upgrade the Provider construct accordingly.\",\n            },\n          ],\n          true\n        );\n      }\n    );\n\n    return new CustomResource(this, \"CFAclCustomResource\", {\n      serviceToken: provider.serviceToken,\n      properties: {\n        ID: aclName,\n        MANAGED_RULES: props?.managedRules ?? [\n          { vendor: \"AWS\", name: \"AWSManagedRulesCommonRuleSet\" },\n        ],\n        CIDR_ALLOW_LIST: props?.cidrAllowList,\n      },\n    });\n  }\n}\n"]}