UNPKG

@aws-solutions-constructs/core

Version:
430 lines 47 kB
"use strict"; /** * 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 }); const aws_cdk_lib_1 = require("aws-cdk-lib"); const s3 = require("aws-cdk-lib/aws-s3"); const s3n = require("aws-cdk-lib/aws-s3-notifications"); const defaults = require("../index"); const assertions_1 = require("aws-cdk-lib/assertions"); const aws_s3_1 = require("aws-cdk-lib/aws-s3"); const test_helper_1 = require("./test-helper"); test('check exception for Missing existingBucketObj from props for deploy = false', () => { const stack = new aws_cdk_lib_1.Stack(); try { defaults.buildS3Bucket(stack, {}); } catch (e) { expect(e).toBeInstanceOf(Error); } }); test('s3 bucket with bucketId', () => { const stack = new aws_cdk_lib_1.Stack(); const buildS3BucketResponse = defaults.buildS3Bucket(stack, {}, 'my'); expect(buildS3BucketResponse.bucket).toBeDefined(); expect(buildS3BucketResponse.loggingBucket).toBeDefined(); const template = assertions_1.Template.fromStack(stack); template.hasResourceProperties("AWS::S3::Bucket", { LoggingConfiguration: { DestinationBucketName: { Ref: "myS3LoggingBucketDE461344" } }, }); }); test('s3 bucket with bucketProps', () => { const stack = new aws_cdk_lib_1.Stack(); const buildS3BucketResponse = defaults.buildS3Bucket(stack, { bucketProps: { bucketName: 'mybucket' } }); expect(buildS3BucketResponse.bucket).toBeDefined(); expect(buildS3BucketResponse.loggingBucket).toBeDefined(); const template = assertions_1.Template.fromStack(stack); template.hasResourceProperties("AWS::S3::Bucket", { BucketName: "mybucket" }); }); test('s3 bucket with default props', () => { const stack = new aws_cdk_lib_1.Stack(); const buildS3BucketResponse = defaults.buildS3Bucket(stack, {}); expect(buildS3BucketResponse.bucket).toBeDefined(); expect(buildS3BucketResponse.loggingBucket).toBeDefined(); const template = assertions_1.Template.fromStack(stack); template.hasResourceProperties("AWS::S3::Bucket", { BucketEncryption: { ServerSideEncryptionConfiguration: [ { ServerSideEncryptionByDefault: { SSEAlgorithm: "AES256" } } ] }, PublicAccessBlockConfiguration: { BlockPublicAcls: true, BlockPublicPolicy: true, IgnorePublicAcls: true, RestrictPublicBuckets: true }, VersioningConfiguration: { Status: "Enabled" } }); }); test('s3 bucket with life cycle policy', () => { const stack = new aws_cdk_lib_1.Stack(); const buildS3BucketResponse = defaults.buildS3Bucket(stack, { bucketProps: { lifecycleRules: [{ expiration: aws_cdk_lib_1.Duration.days(365), transitions: [{ storageClass: aws_s3_1.StorageClass.INFREQUENT_ACCESS, transitionAfter: aws_cdk_lib_1.Duration.days(30) }, { storageClass: aws_s3_1.StorageClass.GLACIER, transitionAfter: aws_cdk_lib_1.Duration.days(90) }] }] } }); expect(buildS3BucketResponse.bucket).toBeDefined(); expect(buildS3BucketResponse.loggingBucket).toBeDefined(); const template = assertions_1.Template.fromStack(stack); template.hasResourceProperties("AWS::S3::Bucket", { LifecycleConfiguration: { Rules: [ { ExpirationInDays: 365, Status: "Enabled", Transitions: [ { StorageClass: "STANDARD_IA", TransitionInDays: 30 }, { StorageClass: "GLACIER", TransitionInDays: 90 } ] } ] } }); }); test('s3 bucket with access logging configured', () => { const stack = new aws_cdk_lib_1.Stack(); const mybucket = new aws_s3_1.Bucket(stack, 'mybucket', { serverAccessLogsBucket: new aws_s3_1.Bucket(stack, 'myaccesslogbucket', {}) }); const buildS3BucketResponse = defaults.buildS3Bucket(stack, { bucketProps: { serverAccessLogsBucket: mybucket } }); expect(buildS3BucketResponse.bucket).toBeDefined(); // This value should be populated, entered Issue 907 // expect(response.loggingBucket).toBeDefined(); const template = assertions_1.Template.fromStack(stack); template.hasResourceProperties("AWS::S3::Bucket", { LoggingConfiguration: { DestinationBucketName: { Ref: "mybucket160F8132" } }, }); }); test('Check S3 Bucket policy', () => { const stack = new aws_cdk_lib_1.Stack(); defaults.buildS3Bucket(stack, {}); const template = assertions_1.Template.fromStack(stack); template.hasResourceProperties("AWS::S3::BucketPolicy", { PolicyDocument: { Statement: [ { Action: "s3:*", Condition: { Bool: { "aws:SecureTransport": "false", }, }, Effect: "Deny", Principal: { AWS: "*" }, Resource: [ { "Fn::GetAtt": [ "S3Bucket07682993", "Arn" ] }, { "Fn::Join": [ "", [ { "Fn::GetAtt": [ "S3Bucket07682993", "Arn" ] }, "/*" ] ] } ] } ], Version: "2012-10-17", }, }); }); test('s3 bucket with LoggingBucket and versioning turned off', () => { const stack = new aws_cdk_lib_1.Stack(); const mybucket = new aws_s3_1.Bucket(stack, 'mybucket', { serverAccessLogsBucket: new aws_s3_1.Bucket(stack, 'myaccesslogbucket', {}) }); const buildS3BucketResponse = defaults.buildS3Bucket(stack, { bucketProps: { serverAccessLogsBucket: mybucket, serverAccessLogsPrefix: 'access-logs', versioned: false } }); expect(buildS3BucketResponse.bucket).toBeDefined(); // The line below fails, this appears to be a bug. Entered Issue 906 expect(buildS3BucketResponse.loggingBucket).toBeDefined(); const template = assertions_1.Template.fromStack(stack); template.hasResourceProperties("AWS::S3::Bucket", { BucketEncryption: { ServerSideEncryptionConfiguration: [ { ServerSideEncryptionByDefault: { SSEAlgorithm: "AES256" } } ] }, LoggingConfiguration: { DestinationBucketName: { Ref: "mybucket160F8132" }, LogFilePrefix: "access-logs" }, PublicAccessBlockConfiguration: { BlockPublicAcls: true, BlockPublicPolicy: true, IgnorePublicAcls: true, RestrictPublicBuckets: true } }); }); test('s3 bucket versioning turned off', () => { const stack = new aws_cdk_lib_1.Stack(); const buildS3BucketResponse = defaults.buildS3Bucket(stack, { bucketProps: { serverAccessLogsPrefix: 'access-logs', versioned: false } }); expect(buildS3BucketResponse.bucket).toBeDefined(); expect(buildS3BucketResponse.loggingBucket).toBeDefined(); const template = assertions_1.Template.fromStack(stack); template.hasResourceProperties("AWS::S3::Bucket", { BucketEncryption: { ServerSideEncryptionConfiguration: [ { ServerSideEncryptionByDefault: { SSEAlgorithm: "AES256" } } ] }, LoggingConfiguration: { DestinationBucketName: { Ref: "S3LoggingBucket800A2B27" }, LogFilePrefix: "access-logs" }, PublicAccessBlockConfiguration: { BlockPublicAcls: true, BlockPublicPolicy: true, IgnorePublicAcls: true, RestrictPublicBuckets: true } }); }); test('s3 bucket with LoggingBucket and auto delete objects', () => { const stack = new aws_cdk_lib_1.Stack(); const buildS3BucketResponse = defaults.buildS3Bucket(stack, { loggingBucketProps: { removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, autoDeleteObjects: true } }); expect(buildS3BucketResponse.bucket).toBeDefined(); expect(buildS3BucketResponse.loggingBucket).toBeDefined(); const template = assertions_1.Template.fromStack(stack); template.hasResourceProperties("AWS::S3::Bucket", { AccessControl: "LogDeliveryWrite" }); template.hasResourceProperties("Custom::S3AutoDeleteObjects", { ServiceToken: { "Fn::GetAtt": [ "CustomS3AutoDeleteObjectsCustomResourceProviderHandler9D90184F", "Arn" ] }, BucketName: { Ref: "S3LoggingBucket800A2B27" } }); }); test('s3 bucket versioning turned on', () => { const stack = new aws_cdk_lib_1.Stack(); const buildS3BucketResponse = defaults.buildS3Bucket(stack, { bucketProps: { serverAccessLogsPrefix: 'access-logs', } }); expect(buildS3BucketResponse.bucket).toBeDefined(); expect(buildS3BucketResponse.loggingBucket).toBeDefined(); const template = assertions_1.Template.fromStack(stack); template.hasResourceProperties("AWS::S3::Bucket", { BucketEncryption: { ServerSideEncryptionConfiguration: [ { ServerSideEncryptionByDefault: { SSEAlgorithm: "AES256" } } ] }, LifecycleConfiguration: { Rules: [ { NoncurrentVersionTransitions: [ { StorageClass: "GLACIER", TransitionInDays: 90 } ], Status: "Enabled" } ] }, LoggingConfiguration: { DestinationBucketName: { Ref: "S3LoggingBucket800A2B27" }, LogFilePrefix: "access-logs" }, PublicAccessBlockConfiguration: { BlockPublicAcls: true, BlockPublicPolicy: true, IgnorePublicAcls: true, RestrictPublicBuckets: true }, VersioningConfiguration: { Status: "Enabled" } }); }); test('Suppress cfn-nag warning for s3 bucket notification', () => { const stack = new aws_cdk_lib_1.Stack(); const buildS3BucketResponse = defaults.buildS3Bucket(stack, {}); const buildQueueResponse = defaults.buildQueue(stack, "S3BucketNotificationQueue", {}); buildS3BucketResponse.bucket.addEventNotification(s3.EventType.OBJECT_CREATED, new s3n.SqsDestination(buildQueueResponse.queue)); defaults.addCfnNagS3BucketNotificationRulesToSuppress(stack, "BucketNotificationsHandler050a0587b7544547bf325f094a3db834"); const template = assertions_1.Template.fromStack(stack); template.hasResource("AWS::Lambda::Function", { Metadata: { cfn_nag: { rules_to_suppress: [ { 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` } ] } } }); template.hasResource("AWS::IAM::Policy", { Metadata: { cfn_nag: { rules_to_suppress: [ { id: "W12", reason: "Bucket resource is '*' due to circular dependency with bucket and role creation at the same time" } ] } } }); }); test('test s3Bucket removalPolicy override', () => { const stack = new aws_cdk_lib_1.Stack(); defaults.buildS3Bucket(stack, { bucketProps: { removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, }, }, 'test-bucket'); assertions_1.Template.fromStack(stack).hasResource("AWS::S3::Bucket", { Type: 'AWS::S3::Bucket', Properties: { AccessControl: "LogDeliveryWrite" }, UpdateReplacePolicy: "Delete", DeletionPolicy: "Delete" }); }); test('s3 bucket with logging turned off', () => { const stack = new aws_cdk_lib_1.Stack(); const respbuildS3BucketResponsense = defaults.buildS3Bucket(stack, { logS3AccessLogs: false, }); expect(respbuildS3BucketResponsense.bucket).toBeDefined(); expect(respbuildS3BucketResponsense.loggingBucket).not.toBeDefined(); (0, test_helper_1.expectNonexistence)(stack, "AWS::S3::Bucket", { LoggingConfiguration: {} }); }); test('Confirm Log Bucket lifecycle rules persist', () => { const stack = new aws_cdk_lib_1.Stack(); defaults.buildS3Bucket(stack, { loggingBucketProps: { lifecycleRules: [ { expiration: aws_cdk_lib_1.Duration.days(365), } ], } }); const template = assertions_1.Template.fromStack(stack); template.hasResource("AWS::S3::Bucket", { Properties: { LifecycleConfiguration: {} } }); }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"s3-bucket-helper.test.js","sourceRoot":"","sources":["s3-bucket-helper.test.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\nimport { Duration, Stack, RemovalPolicy } from 'aws-cdk-lib';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport * as s3n from 'aws-cdk-lib/aws-s3-notifications';\nimport * as defaults from '../index';\nimport { Template } from 'aws-cdk-lib/assertions';\nimport { Bucket, StorageClass } from 'aws-cdk-lib/aws-s3';\nimport { expectNonexistence } from \"./test-helper\";\n\ntest('check exception for Missing existingBucketObj from props for deploy = false', () => {\n  const stack = new Stack();\n\n  try {\n    defaults.buildS3Bucket(stack, {});\n  } catch (e) {\n    expect(e).toBeInstanceOf(Error);\n  }\n});\n\ntest('s3 bucket with bucketId', () => {\n  const stack = new Stack();\n\n  const buildS3BucketResponse = defaults.buildS3Bucket(stack, {}, 'my');\n\n  expect(buildS3BucketResponse.bucket).toBeDefined();\n  expect(buildS3BucketResponse.loggingBucket).toBeDefined();\n\n  const template = Template.fromStack(stack);\n  template.hasResourceProperties(\"AWS::S3::Bucket\", {\n    LoggingConfiguration: {\n      DestinationBucketName: {\n        Ref: \"myS3LoggingBucketDE461344\"\n      }\n    },\n  });\n});\n\ntest('s3 bucket with bucketProps', () => {\n  const stack = new Stack();\n\n  const buildS3BucketResponse = defaults.buildS3Bucket(stack, {\n    bucketProps: {\n      bucketName: 'mybucket'\n    }\n  });\n\n  expect(buildS3BucketResponse.bucket).toBeDefined();\n  expect(buildS3BucketResponse.loggingBucket).toBeDefined();\n\n  const template = Template.fromStack(stack);\n  template.hasResourceProperties(\"AWS::S3::Bucket\", {\n    BucketName: \"mybucket\"\n  });\n});\n\ntest('s3 bucket with default props', () => {\n  const stack = new Stack();\n\n  const buildS3BucketResponse = defaults.buildS3Bucket(stack, {});\n\n  expect(buildS3BucketResponse.bucket).toBeDefined();\n  expect(buildS3BucketResponse.loggingBucket).toBeDefined();\n\n  const template = Template.fromStack(stack);\n  template.hasResourceProperties(\"AWS::S3::Bucket\", {\n    BucketEncryption: {\n      ServerSideEncryptionConfiguration: [\n        {\n          ServerSideEncryptionByDefault: {\n            SSEAlgorithm: \"AES256\"\n          }\n        }\n      ]\n    },\n    PublicAccessBlockConfiguration: {\n      BlockPublicAcls: true,\n      BlockPublicPolicy: true,\n      IgnorePublicAcls: true,\n      RestrictPublicBuckets: true\n    },\n    VersioningConfiguration: {\n      Status: \"Enabled\"\n    }\n  });\n});\n\ntest('s3 bucket with life cycle policy', () => {\n  const stack = new Stack();\n\n  const buildS3BucketResponse = defaults.buildS3Bucket(stack, {\n    bucketProps: {\n      lifecycleRules: [{\n        expiration: Duration.days(365),\n        transitions: [{\n          storageClass: StorageClass.INFREQUENT_ACCESS,\n          transitionAfter: Duration.days(30)\n        }, {\n          storageClass: StorageClass.GLACIER,\n          transitionAfter: Duration.days(90)\n        }]\n      }]\n    }\n  });\n\n  expect(buildS3BucketResponse.bucket).toBeDefined();\n  expect(buildS3BucketResponse.loggingBucket).toBeDefined();\n\n  const template = Template.fromStack(stack);\n  template.hasResourceProperties(\"AWS::S3::Bucket\", {\n    LifecycleConfiguration: {\n      Rules: [\n        {\n          ExpirationInDays: 365,\n          Status: \"Enabled\",\n          Transitions: [\n            {\n              StorageClass: \"STANDARD_IA\",\n              TransitionInDays: 30\n            },\n            {\n              StorageClass: \"GLACIER\",\n              TransitionInDays: 90\n            }\n          ]\n        }\n      ]\n    }\n  });\n});\n\ntest('s3 bucket with access logging configured', () => {\n  const stack = new Stack();\n  const mybucket = new Bucket(stack, 'mybucket', {\n    serverAccessLogsBucket: new Bucket(stack, 'myaccesslogbucket', {})\n  });\n\n  const buildS3BucketResponse = defaults.buildS3Bucket(stack, {\n    bucketProps: {\n      serverAccessLogsBucket: mybucket\n    }\n  });\n\n  expect(buildS3BucketResponse.bucket).toBeDefined();\n  // This value should be populated, entered Issue 907\n  // expect(response.loggingBucket).toBeDefined();\n\n  const template = Template.fromStack(stack);\n  template.hasResourceProperties(\"AWS::S3::Bucket\", {\n    LoggingConfiguration: {\n      DestinationBucketName: {\n        Ref: \"mybucket160F8132\"\n      }\n    },\n  });\n});\n\ntest('Check S3 Bucket policy', () => {\n  const stack = new Stack();\n  defaults.buildS3Bucket(stack, {});\n\n  const template = Template.fromStack(stack);\n  template.hasResourceProperties(\"AWS::S3::BucketPolicy\", {\n    PolicyDocument: {\n      Statement: [\n        {\n          Action: \"s3:*\",\n          Condition: {\n            Bool: {\n              \"aws:SecureTransport\": \"false\",\n            },\n          },\n          Effect: \"Deny\",\n          Principal: {\n            AWS: \"*\"\n          },\n          Resource: [\n            {\n              \"Fn::GetAtt\": [\n                \"S3Bucket07682993\",\n                \"Arn\"\n              ]\n            },\n            {\n              \"Fn::Join\": [\n                \"\",\n                [\n                  {\n                    \"Fn::GetAtt\": [\n                      \"S3Bucket07682993\",\n                      \"Arn\"\n                    ]\n                  },\n                  \"/*\"\n                ]\n              ]\n            }\n          ]\n        }\n      ],\n      Version: \"2012-10-17\",\n    },\n  });\n});\n\ntest('s3 bucket with LoggingBucket and versioning turned off', () => {\n  const stack = new Stack();\n  const mybucket = new Bucket(stack, 'mybucket', {\n    serverAccessLogsBucket: new Bucket(stack, 'myaccesslogbucket', {})\n  });\n\n  const buildS3BucketResponse = defaults.buildS3Bucket(stack, {\n    bucketProps: {\n      serverAccessLogsBucket: mybucket,\n      serverAccessLogsPrefix: 'access-logs',\n      versioned: false\n    }\n  });\n\n  expect(buildS3BucketResponse.bucket).toBeDefined();\n  // The line below fails, this appears to be a bug. Entered Issue 906\n  expect(buildS3BucketResponse.loggingBucket).toBeDefined();\n\n  const template = Template.fromStack(stack);\n  template.hasResourceProperties(\"AWS::S3::Bucket\", {\n    BucketEncryption: {\n      ServerSideEncryptionConfiguration: [\n        {\n          ServerSideEncryptionByDefault: {\n            SSEAlgorithm: \"AES256\"\n          }\n        }\n      ]\n    },\n    LoggingConfiguration: {\n      DestinationBucketName: {\n        Ref: \"mybucket160F8132\"\n      },\n      LogFilePrefix: \"access-logs\"\n    },\n    PublicAccessBlockConfiguration: {\n      BlockPublicAcls: true,\n      BlockPublicPolicy: true,\n      IgnorePublicAcls: true,\n      RestrictPublicBuckets: true\n    }\n  });\n});\n\ntest('s3 bucket versioning turned off', () => {\n  const stack = new Stack();\n\n  const buildS3BucketResponse = defaults.buildS3Bucket(stack, {\n    bucketProps: {\n      serverAccessLogsPrefix: 'access-logs',\n      versioned: false\n    }\n  });\n\n  expect(buildS3BucketResponse.bucket).toBeDefined();\n  expect(buildS3BucketResponse.loggingBucket).toBeDefined();\n\n  const template = Template.fromStack(stack);\n  template.hasResourceProperties(\"AWS::S3::Bucket\", {\n    BucketEncryption: {\n      ServerSideEncryptionConfiguration: [\n        {\n          ServerSideEncryptionByDefault: {\n            SSEAlgorithm: \"AES256\"\n          }\n        }\n      ]\n    },\n    LoggingConfiguration: {\n      DestinationBucketName: {\n        Ref: \"S3LoggingBucket800A2B27\"\n      },\n      LogFilePrefix: \"access-logs\"\n    },\n    PublicAccessBlockConfiguration: {\n      BlockPublicAcls: true,\n      BlockPublicPolicy: true,\n      IgnorePublicAcls: true,\n      RestrictPublicBuckets: true\n    }\n  });\n});\n\ntest('s3 bucket with LoggingBucket and auto delete objects', () => {\n  const stack = new Stack();\n\n  const buildS3BucketResponse = defaults.buildS3Bucket(stack, {\n    loggingBucketProps: {\n      removalPolicy: RemovalPolicy.DESTROY,\n      autoDeleteObjects: true\n    }\n  });\n\n  expect(buildS3BucketResponse.bucket).toBeDefined();\n  expect(buildS3BucketResponse.loggingBucket).toBeDefined();\n\n  const template = Template.fromStack(stack);\n  template.hasResourceProperties(\"AWS::S3::Bucket\", {\n    AccessControl: \"LogDeliveryWrite\"\n  });\n\n  template.hasResourceProperties(\"Custom::S3AutoDeleteObjects\", {\n    ServiceToken: {\n      \"Fn::GetAtt\": [\n        \"CustomS3AutoDeleteObjectsCustomResourceProviderHandler9D90184F\",\n        \"Arn\"\n      ]\n    },\n    BucketName: {\n      Ref: \"S3LoggingBucket800A2B27\"\n    }\n  });\n});\n\ntest('s3 bucket versioning turned on', () => {\n  const stack = new Stack();\n\n  const buildS3BucketResponse = defaults.buildS3Bucket(stack, {\n    bucketProps: {\n      serverAccessLogsPrefix: 'access-logs',\n    }\n  });\n\n  expect(buildS3BucketResponse.bucket).toBeDefined();\n  expect(buildS3BucketResponse.loggingBucket).toBeDefined();\n\n  const template = Template.fromStack(stack);\n\n  template.hasResourceProperties(\"AWS::S3::Bucket\", {\n    BucketEncryption: {\n      ServerSideEncryptionConfiguration: [\n        {\n          ServerSideEncryptionByDefault: {\n            SSEAlgorithm: \"AES256\"\n          }\n        }\n      ]\n    },\n    LifecycleConfiguration: {\n      Rules: [\n        {\n          NoncurrentVersionTransitions: [\n            {\n              StorageClass: \"GLACIER\",\n              TransitionInDays: 90\n            }\n          ],\n          Status: \"Enabled\"\n        }\n      ]\n    },\n    LoggingConfiguration: {\n      DestinationBucketName: {\n        Ref: \"S3LoggingBucket800A2B27\"\n      },\n      LogFilePrefix: \"access-logs\"\n    },\n    PublicAccessBlockConfiguration: {\n      BlockPublicAcls: true,\n      BlockPublicPolicy: true,\n      IgnorePublicAcls: true,\n      RestrictPublicBuckets: true\n    },\n    VersioningConfiguration: {\n      Status: \"Enabled\"\n    }\n  });\n});\n\ntest('Suppress cfn-nag warning for s3 bucket notification', () => {\n  const stack = new Stack();\n  const buildS3BucketResponse = defaults.buildS3Bucket(stack, {});\n  const buildQueueResponse = defaults.buildQueue(stack, \"S3BucketNotificationQueue\", {});\n  buildS3BucketResponse.bucket.addEventNotification(s3.EventType.OBJECT_CREATED, new s3n.SqsDestination(buildQueueResponse.queue));\n  defaults.addCfnNagS3BucketNotificationRulesToSuppress(stack, \"BucketNotificationsHandler050a0587b7544547bf325f094a3db834\");\n\n  const template = Template.fromStack(stack);\n  template.hasResource(\"AWS::Lambda::Function\", {\n    Metadata: {\n      cfn_nag: {\n        rules_to_suppress: [\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    }\n  });\n\n  template.hasResource(\"AWS::IAM::Policy\", {\n    Metadata: {\n      cfn_nag: {\n        rules_to_suppress: [\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    }\n  });\n});\n\ntest('test s3Bucket removalPolicy override', () => {\n  const stack = new Stack();\n\n  defaults.buildS3Bucket(stack, {\n    bucketProps: {\n      removalPolicy: RemovalPolicy.DESTROY,\n    },\n  }, 'test-bucket');\n\n  Template.fromStack(stack).hasResource(\"AWS::S3::Bucket\", {\n    Type: 'AWS::S3::Bucket',\n    Properties: {\n      AccessControl: \"LogDeliveryWrite\"\n    },\n    UpdateReplacePolicy: \"Delete\",\n    DeletionPolicy: \"Delete\"\n  });\n});\n\ntest('s3 bucket with logging turned off', () => {\n  const stack = new Stack();\n\n  const respbuildS3BucketResponsense = defaults.buildS3Bucket(stack, {\n    logS3AccessLogs: false,\n  });\n\n  expect(respbuildS3BucketResponsense.bucket).toBeDefined();\n  expect(respbuildS3BucketResponsense.loggingBucket).not.toBeDefined();\n\n  expectNonexistence(stack, \"AWS::S3::Bucket\", {\n    LoggingConfiguration: {\n    }\n  });\n});\n\ntest('Confirm Log Bucket lifecycle rules persist',  () => {\n  const stack = new Stack();\n\n  defaults.buildS3Bucket(stack, {\n    loggingBucketProps: {\n      lifecycleRules: [\n        {\n          expiration: Duration.days(365),\n        }],\n    }\n  });\n\n  const template = Template.fromStack(stack);\n  template.hasResource(\"AWS::S3::Bucket\", {\n    Properties: {\n      LifecycleConfiguration: {}\n    }\n  });\n});"]}