@aws-solutions-constructs/core
Version:
Core CDK Construct for patterns library
246 lines • 41 kB
JavaScript
/**
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License"). You may not use this file except in compliance
* with the License. A copy of the License is located at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* or in the 'license' file accompanying this file. This file is distributed on an 'AS IS' BASIS, WITHOUT WARRANTIES
* OR CONDITIONS OF ANY KIND, express or implied. See the License for the specific language governing permissions
* and limitations under the License.
*/
Object.defineProperty(exports, "__esModule", { value: true });
exports.createS3AccessLoggingBucket = createS3AccessLoggingBucket;
exports.createCloudFrontLoggingBucket = createCloudFrontLoggingBucket;
exports.createAlbLoggingBucket = createAlbLoggingBucket;
exports.buildS3Bucket = buildS3Bucket;
exports.addCfnNagS3BucketNotificationRulesToSuppress = addCfnNagS3BucketNotificationRulesToSuppress;
exports.CheckS3Props = CheckS3Props;
const s3 = require("aws-cdk-lib/aws-s3");
const s3_bucket_defaults_1 = require("./s3-bucket-defaults");
const utils_1 = require("./utils");
const aws_s3_1 = require("aws-cdk-lib/aws-s3");
const aws_cdk_lib_1 = require("aws-cdk-lib");
/**
* @internal This is an internal core function and should not be called directly by Solutions Constructs clients.
*/
function createS3AccessLoggingBucket(scope, bucketId, loggingBucketProps) {
// Introduce the default props since we can't be certain the caller used them and
// they are important best practices
const combinedBucketProps = (0, utils_1.consolidateProps)((0, s3_bucket_defaults_1.DefaultS3Props)(), loggingBucketProps);
// Create the Logging Bucket
// NOSONAR (typescript:S6281)
// Block Public Access is set by DefaultS3Props, but Sonarqube can't detect it
// It is verified by 's3 bucket with default props' in the unit tests
// NOSONAR (typescript:S6245)
// Encryption is turned on in the default properties that Sonarqube doesn't see
// Verified by unit test 's3 bucket with default props'
// NOSONAR (typescript:S6249)
// enforceSSL is turned on in the default properties that Sonarqube doesn't see
// Verified by unit test 's3 bucket with default props'
// NOSONAR (typescript:typescript:S6249)
// versioning is turned on in the default properties that Sonarqube doesn't see
// Verified by unit test 's3 bucket with default props'
const loggingBucket = new s3.Bucket(scope, bucketId, combinedBucketProps); // NOSONAR
(0, utils_1.addCfnSuppressRules)(loggingBucket, [
{
id: 'W35',
reason: "This S3 bucket is used as the access logging bucket for another bucket"
}
]);
return loggingBucket;
}
/**
* @internal This is an internal core function and should not be called directly by Solutions Constructs clients.
*/
function createCloudFrontLoggingBucket(scope, bucketId, props) {
let cloudFrontLogAccessLogBucket;
// Introduce the default props since we can't be certain the caller used them and
// they are important best practices
let combinedBucketProps = (0, utils_1.consolidateProps)((0, s3_bucket_defaults_1.DefaultS3Props)(), props.loggingBucketProps);
if (!props.loggingBucketProps.serverAccessLogsBucket) {
// Create bucket and add to props
const combinedS3LogBucketProps = (0, utils_1.consolidateProps)((0, s3_bucket_defaults_1.DefaultS3Props)(), props.s3AccessLogBucketProps);
if ((0, utils_1.CheckBooleanWithDefault)(props.enableS3AccessLogs, true)) {
cloudFrontLogAccessLogBucket = new s3.Bucket(scope, `${bucketId}AccessLog`, combinedS3LogBucketProps); // NOSONAR
combinedBucketProps = (0, utils_1.overrideProps)(combinedBucketProps, { serverAccessLogsBucket: cloudFrontLogAccessLogBucket });
(0, utils_1.addCfnSuppressRules)(cloudFrontLogAccessLogBucket, [
{
id: 'W35',
reason: "This S3 bucket is used as the access logging bucket for another bucket"
}
]);
}
}
// Create the Logging Bucket
// NOSONAR (typescript:S6281)
// Block Public Access is set by DefaultS3Props, but Sonarqube can't detect it
// It is verified by 's3 bucket with default props' in the unit tests
// NOSONAR (typescript:S6245)
// Encryption is turned on in the default properties that Sonarqube doesn't see
// Verified by unit test 's3 bucket with default props'
// NOSONAR (typescript:S6249)
// enforceSSL is turned on in the default properties that Sonarqube doesn't see
// Verified by unit test 's3 bucket with default props'
// NOSONAR (typescript:typescript:S6249)
// versioning is turned on in the default properties that Sonarqube doesn't see
// Verified by unit test 's3 bucket with default props'
const cloudfrontLogBucket = new s3.Bucket(scope, bucketId, combinedBucketProps); // NOSONAR
return {
logBucket: cloudfrontLogBucket,
s3AccessLogBucket: cloudFrontLogAccessLogBucket
};
}
/**
* @internal This is an internal core function and should not be called directly by Solutions Constructs clients.
*/
function createAlbLoggingBucket(scope, bucketId, loggingBucketProps) {
// Introduce the default props since we can't be certain the caller used them and
// they are important best practices
const combinedBucketProps = (0, utils_1.consolidateProps)((0, s3_bucket_defaults_1.DefaultS3Props)(), loggingBucketProps);
// Create the Logging Bucket
// NOSONAR (typescript:S6281)
// Block Public Access is set by DefaultS3Props, but Sonarqube can't detect it
// It is verified by 's3 bucket with default props' in the unit tests
// NOSONAR (typescript:S6245)
// Encryption is turned on in the default properties that Sonarqube doesn't see
// Verified by unit test 's3 bucket with default props'
// NOSONAR (typescript:S6249)
// enforceSSL is turned on in the default properties that Sonarqube doesn't see
// Verified by unit test 's3 bucket with default props'
// NOSONAR (typescript:typescript:S6249)
// versioning is turned on in the default properties that Sonarqube doesn't see
// Verified by unit test 's3 bucket with default props'
const loggingBucket = new s3.Bucket(scope, bucketId, combinedBucketProps); // NOSONAR
// Extract the CfnBucket from the loggingBucket
const loggingBucketResource = loggingBucket.node.findChild('Resource');
(0, utils_1.addCfnSuppressRules)(loggingBucketResource, [
{
id: 'W35',
reason: "This is a log bucket for an Application Load Balancer"
}
]);
return loggingBucket;
}
/**
* @internal This is an internal core function and should not be called directly by Solutions Constructs clients.
* @internal This functionality is exposed externally through aws-constructs-factories
*/
function buildS3Bucket(scope, props, bucketId) {
/** Default Life Cycle policy to transition older versions to Glacier after 90 days */
const lifecycleRules = [{
noncurrentVersionTransitions: [{
storageClass: aws_s3_1.StorageClass.GLACIER,
transitionAfter: aws_cdk_lib_1.Duration.days(90)
}]
}];
// Create the Application Bucket
let defaultBucketProps;
let loggingBucket;
const resolvedBucketId = bucketId ? bucketId + 'S3Bucket' : 'S3Bucket';
const loggingBucketId = bucketId ? bucketId + 'S3LoggingBucket' : 'S3LoggingBucket';
// If logging S3 access logs is enabled/undefined and an existing bucket object is not provided
if (props.logS3AccessLogs !== false && !(props.bucketProps?.serverAccessLogsBucket)) {
// Create the Logging Bucket
let loggingBucketProps = (0, s3_bucket_defaults_1.DefaultS3Props)();
if (props.loggingBucketProps) {
// User provided logging bucket props
loggingBucketProps = (0, utils_1.overrideProps)(loggingBucketProps, props.loggingBucketProps);
}
else if (props.bucketProps?.removalPolicy) {
// If the client explicitly specified a removal policy for the main bucket,
// then replicate that policy on the logging bucket
loggingBucketProps = (0, utils_1.overrideProps)(loggingBucketProps, { removalPolicy: props.bucketProps.removalPolicy });
}
loggingBucket = createS3AccessLoggingBucket(scope, loggingBucketId, loggingBucketProps);
}
else if (props.bucketProps?.serverAccessLogsBucket) {
loggingBucket = props.bucketProps?.serverAccessLogsBucket;
}
// Attach the Default Life Cycle policy ONLY IF the versioning is ENABLED
if (props.bucketProps?.versioned === undefined || props.bucketProps.versioned) {
defaultBucketProps = (0, s3_bucket_defaults_1.DefaultS3Props)(loggingBucket, lifecycleRules);
}
else {
defaultBucketProps = (0, s3_bucket_defaults_1.DefaultS3Props)(loggingBucket);
}
const combinedBucketProps = (0, utils_1.consolidateProps)(defaultBucketProps, props.bucketProps);
// NOSONAR (typescript:S6281) - Block Public Access is set by DefaultS3Props,
// but Sonarqube can't detect it
// It is verified by 's3 bucket with default props' in the unit tests
// NOSONAR (typescript:S6245)
// Encryption is turned on in the default properties that Sonarqube doesn't see
// Verified by unit test 's3 bucket with default props'
// NOSONAR (typescript:S6249)
// enforceSSL is turned on in the default properties that Sonarqube doesn't see
// Verified by unit test 's3 bucket with default props'
// NOSONAR (typescript:typescript:S6249)
// versioning is turned on in the default properties that Sonarqube doesn't see
// Verified by unit test 's3 bucket with default props'
const s3Bucket = new s3.Bucket(scope, resolvedBucketId, combinedBucketProps); // NOSONAR
return { bucket: s3Bucket, loggingBucket };
}
/**
* @internal This is an internal core function and should not be called directly by Solutions Constructs clients.
*/
function addCfnNagS3BucketNotificationRulesToSuppress(stackRoot, logicalId) {
const notificationsResourceHandler = stackRoot.node.tryFindChild(logicalId);
const notificationsResourceHandlerRoleRole = notificationsResourceHandler.node.findChild('Role');
const notificationsResourceHandlerRolePolicy = notificationsResourceHandlerRoleRole.node.findChild('DefaultPolicy');
// Extract the CfnFunction from the Function
const fnResource = notificationsResourceHandler.node.findChild('Resource');
(0, utils_1.addCfnSuppressRules)(fnResource, [
{
id: 'W58',
reason: `Lambda functions has the required permission to write CloudWatch Logs. It uses custom policy instead of arn:aws:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole with tighter permissions.`
},
{
id: 'W89',
reason: `This is not a rule for the general case, just for specific use cases/industries`
},
{
id: 'W92',
reason: `Impossible for us to define the correct concurrency for clients`
}
]);
// Extract the CfnPolicy from the iam.Policy
const policyResource = notificationsResourceHandlerRolePolicy.node.findChild('Resource');
(0, utils_1.addCfnSuppressRules)(policyResource, [
{
id: 'W12',
reason: `Bucket resource is '*' due to circular dependency with bucket and role creation at the same time`
}
]);
}
function CheckS3Props(propsObject) {
let errorMessages = '';
let errorFound = false;
if ((propsObject.existingBucketObj || propsObject.existingBucketInterface) && propsObject.bucketProps) {
errorMessages += 'Error - Either provide bucketProps or existingBucketObj, but not both.\n';
errorFound = true;
}
if (propsObject.existingLoggingBucketObj && propsObject.loggingBucketProps) {
errorMessages += 'Error - Either provide existingLoggingBucketObj or loggingBucketProps, but not both.\n';
errorFound = true;
}
if ((propsObject?.logS3AccessLogs === false) && (propsObject.loggingBucketProps || propsObject.existingLoggingBucketObj)) {
errorMessages += 'Error - If logS3AccessLogs is false, supplying loggingBucketProps or existingLoggingBucketObj is invalid.\n';
errorFound = true;
}
if (propsObject.existingBucketObj && (propsObject.loggingBucketProps || propsObject.logS3AccessLogs)) {
errorMessages += 'Error - If existingBucketObj is provided, supplying loggingBucketProps or logS3AccessLogs is an error.\n';
errorFound = true;
}
if (propsObject?.bucketProps?.encryption === s3.BucketEncryption.KMS_MANAGED) {
if (!propsObject.bucketProps.bucketKeyEnabled) {
(0, utils_1.printWarning)("When using SSE-KMS Bucket Encryption, set bucketKeyEnabled to true to lower costs");
(0, utils_1.printWarning)('https://docs.aws.amazon.com/AmazonS3/latest/userguide/bucket-key.html');
}
}
if (errorFound) {
throw new Error(errorMessages);
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"s3-bucket-helper.js","sourceRoot":"","sources":["s3-bucket-helper.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n *  Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n *\n *  Licensed under the Apache License, Version 2.0 (the \"License\"). You may not use this file except in compliance\n *  with the License. A copy of the License is located at\n *\n *      http://www.apache.org/licenses/LICENSE-2.0\n *\n *  or in the 'license' file accompanying this file. This file is distributed on an 'AS IS' BASIS, WITHOUT WARRANTIES\n *  OR CONDITIONS OF ANY KIND, express or implied. See the License for the specific language governing permissions\n *  and limitations under the License.\n */\n\n/*\n *  The functions found here in the core library are for internal use and can be changed\n *  or removed outside of a major release. We recommend against calling them directly from client code.\n */\n\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport * as cdk from 'aws-cdk-lib';\nimport { DefaultS3Props } from './s3-bucket-defaults';\nimport { overrideProps, addCfnSuppressRules, consolidateProps, CheckBooleanWithDefault, printWarning } from './utils';\nimport { StorageClass } from 'aws-cdk-lib/aws-s3';\nimport { Duration } from 'aws-cdk-lib';\n// Note: To ensure CDKv2 compatibility, keep the import statement for Construct separate\nimport { Construct } from 'constructs';\n\nexport interface BuildS3BucketProps {\n  /**\n   * User provided props to override the default props for the S3 Bucket.\n   *\n   * @default - Default props are used\n   */\n  readonly bucketProps?: s3.BucketProps;\n  /**\n   * User provided props to override the default props for the S3 Logging Bucket.\n   *\n   * @default - Default props are used\n   */\n  readonly loggingBucketProps?: s3.BucketProps;\n  /**\n   * Whether to turn on Access Logs for S3. Uses an S3 bucket with associated storage costs.\n   * Enabling Access Logging is a best practice.\n   *\n   * @default - true\n   */\n  readonly logS3AccessLogs?: boolean;\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function createS3AccessLoggingBucket(scope: Construct,\n  bucketId: string,\n  loggingBucketProps: s3.BucketProps): s3.Bucket {\n\n  // Introduce the default props since we can't be certain the caller used them and\n  // they are important best practices\n  const combinedBucketProps = consolidateProps(DefaultS3Props(), loggingBucketProps);\n\n  // Create the Logging Bucket\n  // NOSONAR  (typescript:S6281)\n  // Block Public Access is set by DefaultS3Props, but Sonarqube can't detect it\n  // It is verified by 's3 bucket with default props' in the unit tests\n  // NOSONAR (typescript:S6245)\n  // Encryption is turned on in the default properties that Sonarqube doesn't see\n  // Verified by unit test 's3 bucket with default props'\n  // NOSONAR (typescript:S6249)\n  // enforceSSL  is turned on in the default properties that Sonarqube doesn't see\n  // Verified by unit test 's3 bucket with default props'\n  // NOSONAR (typescript:typescript:S6249)\n  // versioning is turned on in the default properties that Sonarqube doesn't see\n  // Verified by unit test 's3 bucket with default props'\n  const loggingBucket: s3.Bucket = new s3.Bucket(scope, bucketId, combinedBucketProps); // NOSONAR\n\n  addCfnSuppressRules(loggingBucket, [\n    {\n      id: 'W35',\n      reason: \"This S3 bucket is used as the access logging bucket for another bucket\"\n    }\n  ]);\n\n  return loggingBucket;\n}\n\nexport interface CreateCloudFrontLoggingBucketRequest {\n  readonly loggingBucketProps: s3.BucketProps,\n  readonly s3AccessLogBucketProps?: s3.BucketProps,\n  readonly enableS3AccessLogs?: boolean\n}\n\nexport interface CreateCloudFrontLoggingBucketResponse {\n  readonly logBucket: s3.Bucket,\n  readonly s3AccessLogBucket?: s3.Bucket\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function createCloudFrontLoggingBucket(scope: Construct,\n  bucketId: string,\n  props: CreateCloudFrontLoggingBucketRequest): CreateCloudFrontLoggingBucketResponse {\n\n  let cloudFrontLogAccessLogBucket: s3.Bucket | undefined;\n\n  // Introduce the default props since we can't be certain the caller used them and\n  // they are important best practices\n  let combinedBucketProps = consolidateProps(DefaultS3Props(), props.loggingBucketProps);\n\n  if (!props.loggingBucketProps.serverAccessLogsBucket) {\n    // Create bucket and add to props\n    const combinedS3LogBucketProps = consolidateProps(DefaultS3Props(), props.s3AccessLogBucketProps);\n\n    if (CheckBooleanWithDefault(props.enableS3AccessLogs, true)) {\n      cloudFrontLogAccessLogBucket = new s3.Bucket(scope, `${bucketId}AccessLog`, combinedS3LogBucketProps); // NOSONAR\n      combinedBucketProps = overrideProps(combinedBucketProps, { serverAccessLogsBucket: cloudFrontLogAccessLogBucket });\n      addCfnSuppressRules(cloudFrontLogAccessLogBucket, [\n        {\n          id: 'W35',\n          reason: \"This S3 bucket is used as the access logging bucket for another bucket\"\n        }\n      ]);\n    }\n  }\n\n  // Create the Logging Bucket\n  // NOSONAR  (typescript:S6281)\n  // Block Public Access is set by DefaultS3Props, but Sonarqube can't detect it\n  // It is verified by 's3 bucket with default props' in the unit tests\n  // NOSONAR (typescript:S6245)\n  // Encryption is turned on in the default properties that Sonarqube doesn't see\n  // Verified by unit test 's3 bucket with default props'\n  // NOSONAR (typescript:S6249)\n  // enforceSSL  is turned on in the default properties that Sonarqube doesn't see\n  // Verified by unit test 's3 bucket with default props'\n  // NOSONAR (typescript:typescript:S6249)\n  // versioning is turned on in the default properties that Sonarqube doesn't see\n  // Verified by unit test 's3 bucket with default props'\n  const cloudfrontLogBucket: s3.Bucket = new s3.Bucket(scope, bucketId, combinedBucketProps); // NOSONAR\n\n  return {\n    logBucket: cloudfrontLogBucket,\n    s3AccessLogBucket: cloudFrontLogAccessLogBucket\n  };\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function createAlbLoggingBucket(scope: Construct,\n  bucketId: string,\n  loggingBucketProps: s3.BucketProps): s3.Bucket {\n\n  // Introduce the default props since we can't be certain the caller used them and\n  // they are important best practices\n  const combinedBucketProps = consolidateProps(DefaultS3Props(), loggingBucketProps);\n\n  // Create the Logging Bucket\n  // NOSONAR (typescript:S6281)\n  // Block Public Access is set by DefaultS3Props, but Sonarqube can't detect it\n  // It is verified by 's3 bucket with default props' in the unit tests\n  // NOSONAR (typescript:S6245)\n  // Encryption is turned on in the default properties that Sonarqube doesn't see\n  // Verified by unit test 's3 bucket with default props'\n  // NOSONAR (typescript:S6249)\n  // enforceSSL  is turned on in the default properties that Sonarqube doesn't see\n  // Verified by unit test 's3 bucket with default props'\n  // NOSONAR (typescript:typescript:S6249)\n  // versioning is turned on in the default properties that Sonarqube doesn't see\n  // Verified by unit test 's3 bucket with default props'\n  const loggingBucket: s3.Bucket = new s3.Bucket(scope, bucketId, combinedBucketProps); // NOSONAR\n\n  // Extract the CfnBucket from the loggingBucket\n  const loggingBucketResource = loggingBucket.node.findChild('Resource') as s3.CfnBucket;\n\n  addCfnSuppressRules(loggingBucketResource, [\n    {\n      id: 'W35',\n      reason: \"This is a log bucket for an Application Load Balancer\"\n    }\n  ]);\n\n  return loggingBucket;\n}\n\nexport interface BuildS3BucketResponse {\n  readonly bucket: s3.Bucket,\n  readonly loggingBucket?: s3.Bucket\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n * @internal This functionality is exposed externally through aws-constructs-factories\n */\nexport function buildS3Bucket(scope: Construct,\n  props: BuildS3BucketProps,\n  bucketId?: string): BuildS3BucketResponse {\n\n  /** Default Life Cycle policy to transition older versions to Glacier after 90 days */\n  const lifecycleRules: s3.LifecycleRule[] = [{\n    noncurrentVersionTransitions: [{\n      storageClass: StorageClass.GLACIER,\n      transitionAfter: Duration.days(90)\n    }]\n  }];\n\n  // Create the Application Bucket\n  let defaultBucketProps: s3.BucketProps;\n  let loggingBucket;\n  const resolvedBucketId = bucketId ? bucketId + 'S3Bucket' : 'S3Bucket';\n  const loggingBucketId = bucketId ? bucketId + 'S3LoggingBucket' : 'S3LoggingBucket';\n\n  // If logging S3 access logs is enabled/undefined and an existing bucket object is not provided\n  if (props.logS3AccessLogs !== false && !(props.bucketProps?.serverAccessLogsBucket)) {\n    // Create the Logging Bucket\n    let loggingBucketProps = DefaultS3Props();\n\n    if (props.loggingBucketProps) {\n      // User provided logging bucket props\n      loggingBucketProps = overrideProps(loggingBucketProps, props.loggingBucketProps);\n    } else if (props.bucketProps?.removalPolicy) {\n      // If the client explicitly specified a removal policy for the main bucket,\n      // then replicate that policy on the logging bucket\n      loggingBucketProps = overrideProps(loggingBucketProps, { removalPolicy: props.bucketProps.removalPolicy });\n    }\n\n    loggingBucket = createS3AccessLoggingBucket(scope, loggingBucketId, loggingBucketProps);\n  } else if (props.bucketProps?.serverAccessLogsBucket) {\n    loggingBucket = props.bucketProps?.serverAccessLogsBucket as s3.Bucket;\n  }\n\n  // Attach the Default Life Cycle policy ONLY IF the versioning is ENABLED\n  if (props.bucketProps?.versioned === undefined || props.bucketProps.versioned) {\n    defaultBucketProps = DefaultS3Props(loggingBucket, lifecycleRules);\n  } else {\n    defaultBucketProps = DefaultS3Props(loggingBucket);\n  }\n\n  const combinedBucketProps = consolidateProps(defaultBucketProps, props.bucketProps);\n\n  // NOSONAR (typescript:S6281) - Block Public Access is set by DefaultS3Props,\n  // but Sonarqube can't detect it\n  // It is verified by 's3 bucket with default props' in the unit tests\n  // NOSONAR (typescript:S6245)\n  // Encryption is turned on in the default properties that Sonarqube doesn't see\n  // Verified by unit test 's3 bucket with default props'\n  // NOSONAR (typescript:S6249)\n  // enforceSSL  is turned on in the default properties that Sonarqube doesn't see\n  // Verified by unit test 's3 bucket with default props'\n  // NOSONAR (typescript:typescript:S6249)\n  // versioning is turned on in the default properties that Sonarqube doesn't see\n  // Verified by unit test 's3 bucket with default props'\n  const s3Bucket: s3.Bucket = new s3.Bucket(scope, resolvedBucketId, combinedBucketProps ); // NOSONAR\n\n  return { bucket: s3Bucket, loggingBucket };\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function addCfnNagS3BucketNotificationRulesToSuppress(stackRoot: cdk.Stack, logicalId: string) {\n  const notificationsResourceHandler = stackRoot.node.tryFindChild(logicalId) as lambda.Function;\n  const notificationsResourceHandlerRoleRole = notificationsResourceHandler.node.findChild('Role') as iam.Role;\n  const notificationsResourceHandlerRolePolicy = notificationsResourceHandlerRoleRole.node.findChild('DefaultPolicy') as iam.Policy;\n\n  // Extract the CfnFunction from the Function\n  const fnResource = notificationsResourceHandler.node.findChild('Resource') as lambda.CfnFunction;\n  addCfnSuppressRules(fnResource, [\n    {\n      id: 'W58',\n      reason: `Lambda functions has the required permission to write CloudWatch Logs. It uses custom policy instead of arn:aws:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole with tighter permissions.`\n    },\n    {\n      id: 'W89',\n      reason: `This is not a rule for the general case, just for specific use cases/industries`\n    },\n    {\n      id: 'W92',\n      reason: `Impossible for us to define the correct concurrency for clients`\n    }\n  ]);\n\n  // Extract the CfnPolicy from the iam.Policy\n  const policyResource = notificationsResourceHandlerRolePolicy.node.findChild('Resource') as iam.CfnPolicy;\n  addCfnSuppressRules(policyResource, [\n    {\n      id: 'W12',\n      reason: `Bucket resource is '*' due to circular dependency with bucket and role creation at the same time`\n    }\n  ]);\n}\n\nexport interface S3Props {\n  readonly existingBucketObj?: s3.Bucket,\n  readonly existingBucketInterface?: s3.IBucket,\n  readonly bucketProps?: s3.BucketProps,\n  readonly existingLoggingBucketObj?: s3.IBucket;\n  readonly loggingBucketProps?: s3.BucketProps;\n  readonly logS3AccessLogs?: boolean;\n}\n\nexport function CheckS3Props(propsObject: S3Props | any) {\n  let errorMessages = '';\n  let errorFound = false;\n\n  if ((propsObject.existingBucketObj || propsObject.existingBucketInterface) && propsObject.bucketProps) {\n    errorMessages += 'Error - Either provide bucketProps or existingBucketObj, but not both.\\n';\n    errorFound = true;\n  }\n\n  if (propsObject.existingLoggingBucketObj && propsObject.loggingBucketProps) {\n    errorMessages += 'Error - Either provide existingLoggingBucketObj or loggingBucketProps, but not both.\\n';\n    errorFound = true;\n  }\n\n  if ((propsObject?.logS3AccessLogs === false) && (propsObject.loggingBucketProps || propsObject.existingLoggingBucketObj)) {\n    errorMessages += 'Error - If logS3AccessLogs is false, supplying loggingBucketProps or existingLoggingBucketObj is invalid.\\n';\n    errorFound = true;\n  }\n\n  if (propsObject.existingBucketObj && (propsObject.loggingBucketProps || propsObject.logS3AccessLogs)) {\n    errorMessages += 'Error - If existingBucketObj is provided, supplying loggingBucketProps or logS3AccessLogs is an error.\\n';\n    errorFound = true;\n  }\n\n  if (propsObject?.bucketProps?.encryption === s3.BucketEncryption.KMS_MANAGED) {\n    if (!propsObject.bucketProps.bucketKeyEnabled) {\n      printWarning(\"When using SSE-KMS Bucket Encryption, set bucketKeyEnabled to true to lower costs\");\n      printWarning('https://docs.aws.amazon.com/AmazonS3/latest/userguide/bucket-key.html');\n    }\n  }\n\n  if (errorFound) {\n    throw new Error(errorMessages);\n  }\n}\n"]}
;