p6-cdk-s3-protector 0.0.1 → 0.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/.jsii CHANGED
@@ -3866,15 +3866,15 @@
3866
3866
  }
3867
3867
  }
3868
3868
  },
3869
- "description": "AWS CDK: S3 a real-time protector",
3869
+ "description": "AWS CDK: A Real-Time S3 Protector",
3870
3870
  "homepage": "https://github.com/p6m7g8/p6-cdk-s3-protector.git",
3871
3871
  "jsiiVersion": "5.5.4 (build 1378d94)",
3872
3872
  "keywords": [
3873
3873
  "aws",
3874
3874
  "cdk",
3875
3875
  "s3",
3876
- "compliance",
3877
- "security"
3876
+ "security",
3877
+ "compliance"
3878
3878
  ],
3879
3879
  "license": "Apache-2.0",
3880
3880
  "metadata": {
@@ -3887,7 +3887,7 @@
3887
3887
  },
3888
3888
  "name": "p6-cdk-s3-protector",
3889
3889
  "readme": {
3890
- "markdown": "AWS CDK: A real-time S3 protector\n\n# P6CDKS3Protector\n\n## LICENSE\n\n[![License](https://img.shields.io/badge/License-Apache%202.0-yellowgreen.svg)](https://opensource.org/licenses/Apache-2.0)\n\n## Other\n\n[![Gitpod ready-to-code](https://img.shields.io/badge/Gitpod-ready--to--code-blue?logo=gitpod)](https://gitpod.io/#https://github.com/p6m7g8/p6-cdk-s3-protector) ![Sonarcloud Status](https://sonarcloud.io/api/project_badges/measure?project=p6m7g8_p6-cdk-s3-protector&metric=alert_status) ![GitHub commit activity](https://img.shields.io/github/commit-activity/y/p6m7g8/p6-cdk-s3-protector) ![GitHub commit activity](https://img.shields.io/github/commit-activity/m/p6m7g8/p6-cdk-s3-protector)\n\n## Usage\n\n```ts\n...\nimport { P6CDKS3Protector } from 'p6-cdk-s3-protector';\nnew P6CDKS3Protector(this, 'p6S3CDKProtector')\n```\n\n## Architecture\n\n![./assets/diagram.png](./assets/diagram.png)\n\n## Author\n\nPhilip M. Gollucci <pgollucci@p6m7g8.com>\n"
3890
+ "markdown": "AWS CDK: A real-time S3 Protector\n\n# P6CDKS3Protector\n\n## LICENSE\n\n[![License](https://img.shields.io/badge/License-Apache%202.0-yellowgreen.svg)](https://opensource.org/licenses/Apache-2.0)\n\n## Other\n\n[![Gitpod ready-to-code](https://img.shields.io/badge/Gitpod-ready--to--code-blue?logo=gitpod)](https://gitpod.io/#https://github.com/p6m7g8/p6-cdk-s3-protector) ![Sonarcloud Status](https://sonarcloud.io/api/project_badges/measure?project=p6m7g8_p6-cdk-s3-protector&metric=alert_status) ![GitHub commit activity](https://img.shields.io/github/commit-activity/y/p6m7g8/p6-cdk-s3-protector) ![GitHub commit activity](https://img.shields.io/github/commit-activity/m/p6m7g8/p6-cdk-s3-protector)\n\n## Usage\n\n```ts\n...\nimport { P6CDKS3Protector } from 'p6-cdk-s3-protector'\n\nnew P6CDKS3Protector(this, 'p6CDKS3Protector')\n```\n\n## Architecture\n\n![./assets/diagram.png](./assets/diagram.png)\n\n## Author\n\nPhilip M. Gollucci <pgollucci@p6m7g8.com>\n"
3891
3891
  },
3892
3892
  "repository": {
3893
3893
  "type": "git",
@@ -3951,6 +3951,6 @@
3951
3951
  "symbolId": "src/p6cdks3protector:P6CDKS3Protector"
3952
3952
  }
3953
3953
  },
3954
- "version": "0.0.1",
3955
- "fingerprint": "BYjHglzRYKfcIPBrNKMiM1D/F1qvzAhy1hcINy6RM4s="
3954
+ "version": "0.0.2",
3955
+ "fingerprint": "c1Yo4ti+uQ4AcgmXBliBrMnHUst1V+qKzcqiQzKLOxM="
3956
3956
  }
package/README.md CHANGED
@@ -1,4 +1,4 @@
1
- AWS CDK: A real-time S3 protector
1
+ AWS CDK: A real-time S3 Protector
2
2
 
3
3
  # P6CDKS3Protector
4
4
 
@@ -14,8 +14,9 @@ AWS CDK: A real-time S3 protector
14
14
 
15
15
  ```ts
16
16
  ...
17
- import { P6CDKS3Protector } from 'p6-cdk-s3-protector';
18
- new P6CDKS3Protector(this, 'p6S3CDKProtector')
17
+ import { P6CDKS3Protector } from 'p6-cdk-s3-protector'
18
+
19
+ new P6CDKS3Protector(this, 'p6CDKS3Protector')
19
20
  ```
20
21
 
21
22
  ## Architecture
@@ -31,7 +31,7 @@ const lambdajs = __importStar(require("aws-cdk-lib/aws-lambda-nodejs"));
31
31
  const cr = __importStar(require("aws-cdk-lib/custom-resources"));
32
32
  const floyd = __importStar(require("cdk-iam-floyd"));
33
33
  class P6CDKS3Protector extends cdk.Resource {
34
- static [JSII_RTTI_SYMBOL_1] = { fqn: "p6-cdk-s3-protector.P6CDKS3Protector", version: "0.0.1" };
34
+ static [JSII_RTTI_SYMBOL_1] = { fqn: "p6-cdk-s3-protector.P6CDKS3Protector", version: "0.0.2" };
35
35
  constructor(scope, id) {
36
36
  super(scope, id);
37
37
  const policy = new floyd.Statement.S3().allow().toPutObject().toPutObjectAcl();
@@ -23,7 +23,7 @@ const logger = winston_1.default.createLogger({
23
23
  const s3Client = new client_s3_1.S3({});
24
24
  const s3ControlClient = new client_s3_control_1.S3ControlClient({});
25
25
  const stsClient = new client_sts_1.STSClient({});
26
- function p6ShortCircuitShould(event) {
26
+ function smileShortCircuitShould(event) {
27
27
  if (event.detail.requestParameters['x-amz-acl']) {
28
28
  logger.info('ACL is currently %s', event.detail.requestParameters['x-amz-acl'][0]);
29
29
  if (event.detail.requestParameters['x-amz-acl'][0] === 'private') {
@@ -33,14 +33,14 @@ function p6ShortCircuitShould(event) {
33
33
  }
34
34
  return false;
35
35
  }
36
- function p6LoopPrevent(event) {
36
+ function smileLoopPrevent(event) {
37
37
  if (event.detail.errorCode || event.detail.errorMessage) {
38
38
  logger.info('Previous API call resulted in an error. Ending');
39
39
  return true;
40
40
  }
41
41
  return false;
42
42
  }
43
- async function p6S3BucketAclGet(event) {
43
+ async function smileS3BucketAclGet(event) {
44
44
  try {
45
45
  const bucketName = event.detail.requestParameters.bucketName;
46
46
  logger.info('Describing the current ACL: s3://%s', bucketName);
@@ -53,7 +53,7 @@ async function p6S3BucketAclGet(event) {
53
53
  return false;
54
54
  }
55
55
  }
56
- function p6LogDeliveryPreserve(bucketAcl) {
56
+ function smileLogDeliveryPreserve(bucketAcl) {
57
57
  let uriList = '';
58
58
  const preserveLogDelivery = [];
59
59
  for (const grant of bucketAcl.Grants || []) {
@@ -67,7 +67,7 @@ function p6LogDeliveryPreserve(bucketAcl) {
67
67
  }
68
68
  return [uriList, preserveLogDelivery];
69
69
  }
70
- function p6S3BucketAclViolation(uriList) {
70
+ function smileS3BucketAclViolation(uriList) {
71
71
  if (uriList.includes('AllUsers') || uriList.includes('AuthenticatedUsers')) {
72
72
  logger.info('Violation found. Grant ACL greater than Private');
73
73
  return true;
@@ -75,7 +75,7 @@ function p6S3BucketAclViolation(uriList) {
75
75
  logger.info('ACL is correctly already private');
76
76
  return false;
77
77
  }
78
- async function p6S3BucketAclCorrect(bucketAcl, preserveLogDelivery) {
78
+ async function smileS3BucketAclCorrect(bucketAcl, preserveLogDelivery) {
79
79
  logger.info('Attempting Automatic Resolution');
80
80
  try {
81
81
  if (preserveLogDelivery) {
@@ -117,20 +117,20 @@ async function p6S3BucketAclCorrect(bucketAcl, preserveLogDelivery) {
117
117
  logger.info('Error was: %s', err);
118
118
  }
119
119
  }
120
- async function p6S3PublicBucketAcl(event) {
121
- if (p6ShortCircuitShould(event)) {
120
+ async function smileS3PublicBucketAcl(event) {
121
+ if (smileShortCircuitShould(event)) {
122
122
  return true;
123
123
  }
124
- if (p6LoopPrevent(event)) {
124
+ if (smileLoopPrevent(event)) {
125
125
  return true;
126
126
  }
127
- const bucketAcl = await p6S3BucketAclGet(event);
127
+ const bucketAcl = await smileS3BucketAclGet(event);
128
128
  if (!bucketAcl) {
129
129
  return false;
130
130
  }
131
- const [uriList, preserveLogDelivery] = p6LogDeliveryPreserve(bucketAcl);
132
- if (p6S3BucketAclViolation(uriList)) {
133
- await p6S3BucketAclCorrect(bucketAcl, preserveLogDelivery);
131
+ const [uriList, preserveLogDelivery] = smileLogDeliveryPreserve(bucketAcl);
132
+ if (smileS3BucketAclViolation(uriList)) {
133
+ await smileS3BucketAclCorrect(bucketAcl, preserveLogDelivery);
134
134
  return true;
135
135
  }
136
136
  return false;
@@ -154,14 +154,14 @@ async function awsMakePrivate(bucket, key) {
154
154
  logger.info('Making s3://%s/%s private', bucket, key);
155
155
  await s3Client.putObjectAcl({ Bucket: bucket, Key: key, ACL: 'private' });
156
156
  }
157
- async function p6S3PublicBucketObjectAcl(event) {
157
+ async function smileS3PublicBucketObjectAcl(event) {
158
158
  const key = event.detail.requestParameters.key;
159
159
  const bucket = event.detail.requestParameters.bucketName;
160
160
  if (!(await awsIsPrivate(bucket, key))) {
161
161
  await awsMakePrivate(bucket, key);
162
162
  }
163
163
  }
164
- async function p6S3PublicBucketAccessBlock(event) {
164
+ async function smileS3PublicBucketAccessBlock(event) {
165
165
  const pbc = event.detail.requestParameters.PublicAccessBlockConfiguration;
166
166
  logger.info(JSON.stringify(pbc));
167
167
  if (!pbc.RestrictPublicBuckets || !pbc.BlockPublicPolicy || !pbc.BlockPublicAcls || !pbc.IgnorePublicAcls) {
@@ -179,7 +179,7 @@ async function p6S3PublicBucketAccessBlock(event) {
179
179
  logger.info(JSON.stringify(response));
180
180
  }
181
181
  }
182
- async function p6S3PublicAccessBlock(event) {
182
+ async function smileS3PublicAccessBlock(event) {
183
183
  const pbc = event.detail.requestParameters.PublicAccessBlockConfiguration;
184
184
  logger.info(JSON.stringify(pbc));
185
185
  if (!pbc.RestrictPublicBuckets || !pbc.BlockPublicPolicy || !pbc.BlockPublicAcls || !pbc.IgnorePublicAcls) {
@@ -198,7 +198,7 @@ async function p6S3PublicAccessBlock(event) {
198
198
  logger.info(JSON.stringify(response));
199
199
  }
200
200
  }
201
- async function p6S3PublicFusebox(event) {
201
+ async function smileS3PublicFusebox(event) {
202
202
  if (!event.detail || !event.detail.eventName) {
203
203
  return false;
204
204
  }
@@ -214,21 +214,21 @@ async function p6S3PublicFusebox(event) {
214
214
  logger.info('eventName: %s', eventName);
215
215
  }
216
216
  if (eventName === 'PutBucketAcl') {
217
- await p6S3PublicBucketAcl(event);
217
+ await smileS3PublicBucketAcl(event);
218
218
  }
219
219
  else if (eventName === 'PutObjectAcl') {
220
- await p6S3PublicBucketObjectAcl(event);
220
+ await smileS3PublicBucketObjectAcl(event);
221
221
  }
222
222
  else if (eventName === 'PutBucketPublicAccessBlock') {
223
- await p6S3PublicBucketAccessBlock(event);
223
+ await smileS3PublicBucketAccessBlock(event);
224
224
  }
225
225
  else if (eventName === 'PutAccountPublicAccessBlock') {
226
- await p6S3PublicAccessBlock(event);
226
+ await smileS3PublicAccessBlock(event);
227
227
  }
228
228
  return true;
229
229
  }
230
230
  async function handler(event, _context) {
231
- await p6S3PublicFusebox(event);
231
+ await smileS3PublicFusebox(event);
232
232
  return true;
233
233
  }
234
234
  async function main() {
@@ -240,4 +240,4 @@ async function main() {
240
240
  if (require.main === module) {
241
241
  main();
242
242
  }
243
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"p6cdks3protector.p6CDKS3Protector.js","sourceRoot":"","sources":["../src/p6cdks3protector.p6CDKS3Protector.ts"],"names":[],"mappings":"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","sourcesContent":["import type { GetBucketAclOutput, Grant } from '@aws-sdk/client-s3'\nimport type { Context } from 'aws-lambda'\nimport type { Logger } from 'winston'\nimport fs from 'node:fs'\nimport path from 'node:path'\nimport { S3 } from '@aws-sdk/client-s3'\nimport { PutPublicAccessBlockCommand, S3ControlClient } from '@aws-sdk/client-s3-control'\nimport { GetCallerIdentityCommand, STSClient } from '@aws-sdk/client-sts'\nimport winston from 'winston'\n\n// Configure the logger\nconst logger: Logger = winston.createLogger({\n  level: 'info',\n  format: winston.format.json(),\n  defaultMeta: { service: 'user-service' },\n  transports: [\n    new winston.transports.Console(),\n  ],\n})\n\nconst s3Client = new S3({})\nconst s3ControlClient = new S3ControlClient({})\nconst stsClient = new STSClient({})\n\ninterface S3Event {\n  detail: {\n    requestParameters: {\n      [key: string]: any\n    }\n    eventName: string\n    errorCode?: string\n    errorMessage?: string\n  }\n}\n\nfunction p6ShortCircuitShould(event: S3Event): boolean {\n  if (event.detail.requestParameters['x-amz-acl']) {\n    logger.info('ACL is currently %s', event.detail.requestParameters['x-amz-acl'][0])\n    if (event.detail.requestParameters['x-amz-acl'][0] === 'private') {\n      logger.info('ACL is already private.  Ending.')\n      return true\n    }\n  }\n  return false\n}\n\nfunction p6LoopPrevent(event: S3Event): boolean {\n  if (event.detail.errorCode || event.detail.errorMessage) {\n    logger.info('Previous API call resulted in an error. Ending')\n    return true\n  }\n  return false\n}\n\nasync function p6S3BucketAclGet(event: S3Event): Promise<GetBucketAclOutput | false> {\n  try {\n    const bucketName = event.detail.requestParameters.bucketName\n    logger.info('Describing the current ACL: s3://%s', bucketName)\n    const bucketAcl = await s3Client.getBucketAcl({ Bucket: bucketName })\n    logger.info(JSON.stringify(bucketAcl))\n    return bucketAcl\n  }\n  catch (err) {\n    logger.error('Error was: {%s} Manual followup recommended', err)\n    return false\n  }\n}\n\nfunction p6LogDeliveryPreserve(bucketAcl: GetBucketAclOutput): [string, Grant[]] {\n  let uriList = ''\n  const preserveLogDelivery: Grant[] = []\n\n  for (const grant of bucketAcl.Grants || []) {\n    if (grant.Grantee?.URI) {\n      logger.info('Found Grant: %s', JSON.stringify(grant))\n      uriList += grant.Grantee.URI\n      if (grant.Grantee.URI.includes('LogDelivery')) {\n        preserveLogDelivery.push(grant)\n      }\n    }\n  }\n\n  return [uriList, preserveLogDelivery]\n}\n\nfunction p6S3BucketAclViolation(uriList: string): boolean {\n  if (uriList.includes('AllUsers') || uriList.includes('AuthenticatedUsers')) {\n    logger.info('Violation found.  Grant ACL greater than Private')\n    return true\n  }\n  logger.info('ACL is correctly already private')\n  return false\n}\n\nasync function p6S3BucketAclCorrect(bucketAcl: GetBucketAclOutput, preserveLogDelivery: Array<Grant> | false): Promise<void> {\n  logger.info('Attempting Automatic Resolution')\n  try {\n    if (preserveLogDelivery) {\n      logger.info('ACL resetting ACL to LogDelivery')\n      logger.info('Preserve was: %s', JSON.stringify(preserveLogDelivery))\n\n      const aclString = {\n        Grants: preserveLogDelivery,\n        Owner: bucketAcl.Owner,\n      }\n\n      const response = await s3Client.putBucketAcl({\n        Bucket: bucketAcl?.Owner?.ID,\n        AccessControlPolicy: aclString,\n      })\n\n      logger.info(JSON.stringify(response))\n      if (response.$metadata.httpStatusCode === 200) {\n        logger.info('Reverted to only contain LogDelivery')\n      }\n      else {\n        logger.error('PutBucketACL failed. Manual followup')\n      }\n    }\n    else {\n      logger.info('ACL resetting ACL to Private')\n      const response = await s3Client.putBucketAcl({\n        Bucket: bucketAcl?.Owner?.ID,\n        ACL: 'private',\n      })\n\n      logger.info(JSON.stringify(response))\n      if (response.$metadata.httpStatusCode === 200) {\n        logger.info('Bucket ACL has been changed to Private')\n      }\n      else {\n        logger.error('PutBucketACL failed. Manual followup')\n      }\n    }\n  }\n  catch (err) {\n    logger.info('Unable to resolve violation automatically')\n    logger.info('Error was: %s', err)\n  }\n}\n\nasync function p6S3PublicBucketAcl(event: S3Event): Promise<boolean> {\n  if (p6ShortCircuitShould(event)) {\n    return true\n  }\n\n  if (p6LoopPrevent(event)) {\n    return true\n  }\n\n  const bucketAcl = await p6S3BucketAclGet(event)\n  if (!bucketAcl) {\n    return false\n  }\n\n  const [uriList, preserveLogDelivery] = p6LogDeliveryPreserve(bucketAcl)\n\n  if (p6S3BucketAclViolation(uriList)) {\n    await p6S3BucketAclCorrect(bucketAcl, preserveLogDelivery)\n    return true\n  }\n\n  return false\n}\n\nasync function awsIsPrivate(bucket: string, key: string): Promise<boolean> {\n  logger.info('Describing the ACL: s3://%s/%s', bucket, key)\n  const acl = await s3Client.getObjectAcl({ Bucket: bucket, Key: key })\n\n  if (acl.Grants!.length > 1) {\n    logger.info('Greater than one Grant')\n    return false\n  }\n\n  const ownerId = acl.Owner?.ID\n  const granteeId = acl.Grants![0].Grantee?.ID\n  if (ownerId !== granteeId) {\n    logger.info('owner:[%s], grantee[%s] do not match', ownerId, granteeId)\n    return false\n  }\n\n  return true\n}\n\nasync function awsMakePrivate(bucket: string, key: string): Promise<void> {\n  logger.info('Making s3://%s/%s private', bucket, key)\n  await s3Client.putObjectAcl({ Bucket: bucket, Key: key, ACL: 'private' })\n}\n\nasync function p6S3PublicBucketObjectAcl(event: S3Event): Promise<void> {\n  const key = event.detail.requestParameters.key\n  const bucket = event.detail.requestParameters.bucketName\n\n  if (!(await awsIsPrivate(bucket, key))) {\n    await awsMakePrivate(bucket, key)\n  }\n}\n\nasync function p6S3PublicBucketAccessBlock(event: S3Event): Promise<void> {\n  const pbc = event.detail.requestParameters.PublicAccessBlockConfiguration\n  logger.info(JSON.stringify(pbc))\n\n  if (!pbc.RestrictPublicBuckets || !pbc.BlockPublicPolicy || !pbc.BlockPublicAcls || !pbc.IgnorePublicAcls) {\n    const bucket = event.detail.requestParameters.bucketName\n    logger.info('s3://%s now not private, fixing...', bucket)\n\n    const response = await s3Client.putPublicAccessBlock({\n      Bucket: bucket,\n      PublicAccessBlockConfiguration: {\n        BlockPublicAcls: true,\n        IgnorePublicAcls: true,\n        BlockPublicPolicy: true,\n        RestrictPublicBuckets: true,\n      },\n    })\n\n    logger.info(JSON.stringify(response))\n  }\n}\n\nasync function p6S3PublicAccessBlock(event: S3Event): Promise<void> {\n  const pbc = event.detail.requestParameters.PublicAccessBlockConfiguration\n  logger.info(JSON.stringify(pbc))\n\n  if (!pbc.RestrictPublicBuckets || !pbc.BlockPublicPolicy || !pbc.BlockPublicAcls || !pbc.IgnorePublicAcls) {\n    const command = new GetCallerIdentityCommand({})\n    const account = await stsClient.send(command)\n    logger.info('%s', account)\n\n    const response = await s3ControlClient.send(new PutPublicAccessBlockCommand({\n      PublicAccessBlockConfiguration: {\n        BlockPublicAcls: true,\n        IgnorePublicAcls: true,\n        BlockPublicPolicy: true,\n        RestrictPublicBuckets: true,\n      },\n      AccountId: account.Account,\n    }))\n\n    logger.info(JSON.stringify(response))\n  }\n}\n\nasync function p6S3PublicFusebox(event: S3Event): Promise<boolean> {\n  if (!event.detail || !event.detail.eventName) {\n    return false\n  }\n\n  const events = [\n    'PutBucketAcl',\n    'PutObjectAcl',\n    'PutBucketPublicAccessBlock',\n    'PutAccountPublicAccessBlock',\n  ]\n\n  const eventName = event.detail.eventName\n  if (events.includes(eventName)) {\n    logger.info('======================================================================================')\n    logger.info('eventName: %s', eventName)\n  }\n\n  if (eventName === 'PutBucketAcl') {\n    await p6S3PublicBucketAcl(event)\n  }\n  else if (eventName === 'PutObjectAcl') {\n    await p6S3PublicBucketObjectAcl(event)\n  }\n  else if (eventName === 'PutBucketPublicAccessBlock') {\n    await p6S3PublicBucketAccessBlock(event)\n  }\n  else if (eventName === 'PutAccountPublicAccessBlock') {\n    await p6S3PublicAccessBlock(event)\n  }\n\n  return true\n}\n\nexport async function handler(event: S3Event, _context?: Context): Promise<boolean> {\n  await p6S3PublicFusebox(event)\n  return true\n}\n\nexport async function main(): Promise<void> {\n  logger.debug('Reading fixtures/putBucketAcl.json')\n  const data = JSON.parse(fs.readFileSync(path.resolve('fixtures/putBucketAcl.json'), 'utf8'))\n\n  logger.debug('handler()')\n  await handler(data)\n}\n\nif (require.main === module) {\n  main()\n}\n"]}
243
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"p6cdks3protector.p6CDKS3Protector.js","sourceRoot":"","sources":["../src/p6cdks3protector.p6CDKS3Protector.ts"],"names":[],"mappings":"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","sourcesContent":["import type { GetBucketAclOutput, Grant } from '@aws-sdk/client-s3'\nimport type { Context } from 'aws-lambda'\nimport type { Logger } from 'winston'\nimport fs from 'node:fs'\nimport path from 'node:path'\nimport { S3 } from '@aws-sdk/client-s3'\nimport { PutPublicAccessBlockCommand, S3ControlClient } from '@aws-sdk/client-s3-control'\nimport { GetCallerIdentityCommand, STSClient } from '@aws-sdk/client-sts'\nimport winston from 'winston'\n\n// Configure the logger\nconst logger: Logger = winston.createLogger({\n  level: 'info',\n  format: winston.format.json(),\n  defaultMeta: { service: 'user-service' },\n  transports: [\n    new winston.transports.Console(),\n  ],\n})\n\nconst s3Client = new S3({})\nconst s3ControlClient = new S3ControlClient({})\nconst stsClient = new STSClient({})\n\ninterface S3Event {\n  detail: {\n    requestParameters: {\n      [key: string]: any\n    }\n    eventName: string\n    errorCode?: string\n    errorMessage?: string\n  }\n}\n\nfunction smileShortCircuitShould(event: S3Event): boolean {\n  if (event.detail.requestParameters['x-amz-acl']) {\n    logger.info('ACL is currently %s', event.detail.requestParameters['x-amz-acl'][0])\n    if (event.detail.requestParameters['x-amz-acl'][0] === 'private') {\n      logger.info('ACL is already private.  Ending.')\n      return true\n    }\n  }\n  return false\n}\n\nfunction smileLoopPrevent(event: S3Event): boolean {\n  if (event.detail.errorCode || event.detail.errorMessage) {\n    logger.info('Previous API call resulted in an error. Ending')\n    return true\n  }\n  return false\n}\n\nasync function smileS3BucketAclGet(event: S3Event): Promise<GetBucketAclOutput | false> {\n  try {\n    const bucketName = event.detail.requestParameters.bucketName\n    logger.info('Describing the current ACL: s3://%s', bucketName)\n    const bucketAcl = await s3Client.getBucketAcl({ Bucket: bucketName })\n    logger.info(JSON.stringify(bucketAcl))\n    return bucketAcl\n  }\n  catch (err) {\n    logger.error('Error was: {%s} Manual followup recommended', err)\n    return false\n  }\n}\n\nfunction smileLogDeliveryPreserve(bucketAcl: GetBucketAclOutput): [string, Grant[]] {\n  let uriList = ''\n  const preserveLogDelivery: Grant[] = []\n\n  for (const grant of bucketAcl.Grants || []) {\n    if (grant.Grantee?.URI) {\n      logger.info('Found Grant: %s', JSON.stringify(grant))\n      uriList += grant.Grantee.URI\n      if (grant.Grantee.URI.includes('LogDelivery')) {\n        preserveLogDelivery.push(grant)\n      }\n    }\n  }\n\n  return [uriList, preserveLogDelivery]\n}\n\nfunction smileS3BucketAclViolation(uriList: string): boolean {\n  if (uriList.includes('AllUsers') || uriList.includes('AuthenticatedUsers')) {\n    logger.info('Violation found.  Grant ACL greater than Private')\n    return true\n  }\n  logger.info('ACL is correctly already private')\n  return false\n}\n\nasync function smileS3BucketAclCorrect(bucketAcl: GetBucketAclOutput, preserveLogDelivery: Array<Grant> | false): Promise<void> {\n  logger.info('Attempting Automatic Resolution')\n  try {\n    if (preserveLogDelivery) {\n      logger.info('ACL resetting ACL to LogDelivery')\n      logger.info('Preserve was: %s', JSON.stringify(preserveLogDelivery))\n\n      const aclString = {\n        Grants: preserveLogDelivery,\n        Owner: bucketAcl.Owner,\n      }\n\n      const response = await s3Client.putBucketAcl({\n        Bucket: bucketAcl?.Owner?.ID,\n        AccessControlPolicy: aclString,\n      })\n\n      logger.info(JSON.stringify(response))\n      if (response.$metadata.httpStatusCode === 200) {\n        logger.info('Reverted to only contain LogDelivery')\n      }\n      else {\n        logger.error('PutBucketACL failed. Manual followup')\n      }\n    }\n    else {\n      logger.info('ACL resetting ACL to Private')\n      const response = await s3Client.putBucketAcl({\n        Bucket: bucketAcl?.Owner?.ID,\n        ACL: 'private',\n      })\n\n      logger.info(JSON.stringify(response))\n      if (response.$metadata.httpStatusCode === 200) {\n        logger.info('Bucket ACL has been changed to Private')\n      }\n      else {\n        logger.error('PutBucketACL failed. Manual followup')\n      }\n    }\n  }\n  catch (err) {\n    logger.info('Unable to resolve violation automatically')\n    logger.info('Error was: %s', err)\n  }\n}\n\nasync function smileS3PublicBucketAcl(event: S3Event): Promise<boolean> {\n  if (smileShortCircuitShould(event)) {\n    return true\n  }\n\n  if (smileLoopPrevent(event)) {\n    return true\n  }\n\n  const bucketAcl = await smileS3BucketAclGet(event)\n  if (!bucketAcl) {\n    return false\n  }\n\n  const [uriList, preserveLogDelivery] = smileLogDeliveryPreserve(bucketAcl)\n\n  if (smileS3BucketAclViolation(uriList)) {\n    await smileS3BucketAclCorrect(bucketAcl, preserveLogDelivery)\n    return true\n  }\n\n  return false\n}\n\nasync function awsIsPrivate(bucket: string, key: string): Promise<boolean> {\n  logger.info('Describing the ACL: s3://%s/%s', bucket, key)\n  const acl = await s3Client.getObjectAcl({ Bucket: bucket, Key: key })\n\n  if (acl.Grants!.length > 1) {\n    logger.info('Greater than one Grant')\n    return false\n  }\n\n  const ownerId = acl.Owner?.ID\n  const granteeId = acl.Grants![0].Grantee?.ID\n  if (ownerId !== granteeId) {\n    logger.info('owner:[%s], grantee[%s] do not match', ownerId, granteeId)\n    return false\n  }\n\n  return true\n}\n\nasync function awsMakePrivate(bucket: string, key: string): Promise<void> {\n  logger.info('Making s3://%s/%s private', bucket, key)\n  await s3Client.putObjectAcl({ Bucket: bucket, Key: key, ACL: 'private' })\n}\n\nasync function smileS3PublicBucketObjectAcl(event: S3Event): Promise<void> {\n  const key = event.detail.requestParameters.key\n  const bucket = event.detail.requestParameters.bucketName\n\n  if (!(await awsIsPrivate(bucket, key))) {\n    await awsMakePrivate(bucket, key)\n  }\n}\n\nasync function smileS3PublicBucketAccessBlock(event: S3Event): Promise<void> {\n  const pbc = event.detail.requestParameters.PublicAccessBlockConfiguration\n  logger.info(JSON.stringify(pbc))\n\n  if (!pbc.RestrictPublicBuckets || !pbc.BlockPublicPolicy || !pbc.BlockPublicAcls || !pbc.IgnorePublicAcls) {\n    const bucket = event.detail.requestParameters.bucketName\n    logger.info('s3://%s now not private, fixing...', bucket)\n\n    const response = await s3Client.putPublicAccessBlock({\n      Bucket: bucket,\n      PublicAccessBlockConfiguration: {\n        BlockPublicAcls: true,\n        IgnorePublicAcls: true,\n        BlockPublicPolicy: true,\n        RestrictPublicBuckets: true,\n      },\n    })\n\n    logger.info(JSON.stringify(response))\n  }\n}\n\nasync function smileS3PublicAccessBlock(event: S3Event): Promise<void> {\n  const pbc = event.detail.requestParameters.PublicAccessBlockConfiguration\n  logger.info(JSON.stringify(pbc))\n\n  if (!pbc.RestrictPublicBuckets || !pbc.BlockPublicPolicy || !pbc.BlockPublicAcls || !pbc.IgnorePublicAcls) {\n    const command = new GetCallerIdentityCommand({})\n    const account = await stsClient.send(command)\n    logger.info('%s', account)\n\n    const response = await s3ControlClient.send(new PutPublicAccessBlockCommand({\n      PublicAccessBlockConfiguration: {\n        BlockPublicAcls: true,\n        IgnorePublicAcls: true,\n        BlockPublicPolicy: true,\n        RestrictPublicBuckets: true,\n      },\n      AccountId: account.Account,\n    }))\n\n    logger.info(JSON.stringify(response))\n  }\n}\n\nasync function smileS3PublicFusebox(event: S3Event): Promise<boolean> {\n  if (!event.detail || !event.detail.eventName) {\n    return false\n  }\n\n  const events = [\n    'PutBucketAcl',\n    'PutObjectAcl',\n    'PutBucketPublicAccessBlock',\n    'PutAccountPublicAccessBlock',\n  ]\n\n  const eventName = event.detail.eventName\n  if (events.includes(eventName)) {\n    logger.info('======================================================================================')\n    logger.info('eventName: %s', eventName)\n  }\n\n  if (eventName === 'PutBucketAcl') {\n    await smileS3PublicBucketAcl(event)\n  }\n  else if (eventName === 'PutObjectAcl') {\n    await smileS3PublicBucketObjectAcl(event)\n  }\n  else if (eventName === 'PutBucketPublicAccessBlock') {\n    await smileS3PublicBucketAccessBlock(event)\n  }\n  else if (eventName === 'PutAccountPublicAccessBlock') {\n    await smileS3PublicAccessBlock(event)\n  }\n\n  return true\n}\n\nexport async function handler(event: S3Event, _context?: Context): Promise<boolean> {\n  await smileS3PublicFusebox(event)\n  return true\n}\n\nexport async function main(): Promise<void> {\n  logger.debug('Reading fixtures/putBucketAcl.json')\n  const data = JSON.parse(fs.readFileSync(path.resolve('fixtures/putBucketAcl.json'), 'utf8'))\n\n  logger.debug('handler()')\n  await handler(data)\n}\n\nif (require.main === module) {\n  main()\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "p6-cdk-s3-protector",
3
- "description": "AWS CDK: S3 a real-time protector",
3
+ "description": "AWS CDK: A Real-Time S3 Protector",
4
4
  "repository": {
5
5
  "type": "git",
6
6
  "url": "https://github.com/p6m7g8/p6-cdk-s3-protector.git"
@@ -103,13 +103,13 @@
103
103
  "aws",
104
104
  "cdk",
105
105
  "s3",
106
- "compliance",
107
- "security"
106
+ "security",
107
+ "compliance"
108
108
  ],
109
109
  "main": "lib/index.js",
110
110
  "types": "src/index.d.ts",
111
111
  "license": "Apache-2.0",
112
- "version": "0.0.1",
112
+ "version": "0.0.2",
113
113
  "jsii": {
114
114
  "outdir": "dist",
115
115
  "tsc": {
@@ -137,5 +137,5 @@
137
137
  }
138
138
  }
139
139
  },
140
- "packageManager": "pnpm@9.12.2+sha512.22721b3a11f81661ae1ec68ce1a7b879425a1ca5b991c975b074ac220b187ce56c708fe5db69f4c962c989452eee76c82877f4ee80f474cebd61ee13461b6228"
140
+ "packageManager": "pnpm@9.12.3+sha512.cce0f9de9c5a7c95bef944169cc5dfe8741abfb145078c0d508b868056848a87c81e626246cb60967cbd7fd29a6c062ef73ff840d96b3c86c40ac92cf4a813ee"
141
141
  }
@@ -33,7 +33,7 @@ interface S3Event {
33
33
  }
34
34
  }
35
35
 
36
- function p6ShortCircuitShould(event: S3Event): boolean {
36
+ function smileShortCircuitShould(event: S3Event): boolean {
37
37
  if (event.detail.requestParameters['x-amz-acl']) {
38
38
  logger.info('ACL is currently %s', event.detail.requestParameters['x-amz-acl'][0])
39
39
  if (event.detail.requestParameters['x-amz-acl'][0] === 'private') {
@@ -44,7 +44,7 @@ function p6ShortCircuitShould(event: S3Event): boolean {
44
44
  return false
45
45
  }
46
46
 
47
- function p6LoopPrevent(event: S3Event): boolean {
47
+ function smileLoopPrevent(event: S3Event): boolean {
48
48
  if (event.detail.errorCode || event.detail.errorMessage) {
49
49
  logger.info('Previous API call resulted in an error. Ending')
50
50
  return true
@@ -52,7 +52,7 @@ function p6LoopPrevent(event: S3Event): boolean {
52
52
  return false
53
53
  }
54
54
 
55
- async function p6S3BucketAclGet(event: S3Event): Promise<GetBucketAclOutput | false> {
55
+ async function smileS3BucketAclGet(event: S3Event): Promise<GetBucketAclOutput | false> {
56
56
  try {
57
57
  const bucketName = event.detail.requestParameters.bucketName
58
58
  logger.info('Describing the current ACL: s3://%s', bucketName)
@@ -66,7 +66,7 @@ async function p6S3BucketAclGet(event: S3Event): Promise<GetBucketAclOutput | fa
66
66
  }
67
67
  }
68
68
 
69
- function p6LogDeliveryPreserve(bucketAcl: GetBucketAclOutput): [string, Grant[]] {
69
+ function smileLogDeliveryPreserve(bucketAcl: GetBucketAclOutput): [string, Grant[]] {
70
70
  let uriList = ''
71
71
  const preserveLogDelivery: Grant[] = []
72
72
 
@@ -83,7 +83,7 @@ function p6LogDeliveryPreserve(bucketAcl: GetBucketAclOutput): [string, Grant[]]
83
83
  return [uriList, preserveLogDelivery]
84
84
  }
85
85
 
86
- function p6S3BucketAclViolation(uriList: string): boolean {
86
+ function smileS3BucketAclViolation(uriList: string): boolean {
87
87
  if (uriList.includes('AllUsers') || uriList.includes('AuthenticatedUsers')) {
88
88
  logger.info('Violation found. Grant ACL greater than Private')
89
89
  return true
@@ -92,7 +92,7 @@ function p6S3BucketAclViolation(uriList: string): boolean {
92
92
  return false
93
93
  }
94
94
 
95
- async function p6S3BucketAclCorrect(bucketAcl: GetBucketAclOutput, preserveLogDelivery: Array<Grant> | false): Promise<void> {
95
+ async function smileS3BucketAclCorrect(bucketAcl: GetBucketAclOutput, preserveLogDelivery: Array<Grant> | false): Promise<void> {
96
96
  logger.info('Attempting Automatic Resolution')
97
97
  try {
98
98
  if (preserveLogDelivery) {
@@ -139,24 +139,24 @@ async function p6S3BucketAclCorrect(bucketAcl: GetBucketAclOutput, preserveLogDe
139
139
  }
140
140
  }
141
141
 
142
- async function p6S3PublicBucketAcl(event: S3Event): Promise<boolean> {
143
- if (p6ShortCircuitShould(event)) {
142
+ async function smileS3PublicBucketAcl(event: S3Event): Promise<boolean> {
143
+ if (smileShortCircuitShould(event)) {
144
144
  return true
145
145
  }
146
146
 
147
- if (p6LoopPrevent(event)) {
147
+ if (smileLoopPrevent(event)) {
148
148
  return true
149
149
  }
150
150
 
151
- const bucketAcl = await p6S3BucketAclGet(event)
151
+ const bucketAcl = await smileS3BucketAclGet(event)
152
152
  if (!bucketAcl) {
153
153
  return false
154
154
  }
155
155
 
156
- const [uriList, preserveLogDelivery] = p6LogDeliveryPreserve(bucketAcl)
156
+ const [uriList, preserveLogDelivery] = smileLogDeliveryPreserve(bucketAcl)
157
157
 
158
- if (p6S3BucketAclViolation(uriList)) {
159
- await p6S3BucketAclCorrect(bucketAcl, preserveLogDelivery)
158
+ if (smileS3BucketAclViolation(uriList)) {
159
+ await smileS3BucketAclCorrect(bucketAcl, preserveLogDelivery)
160
160
  return true
161
161
  }
162
162
 
@@ -187,7 +187,7 @@ async function awsMakePrivate(bucket: string, key: string): Promise<void> {
187
187
  await s3Client.putObjectAcl({ Bucket: bucket, Key: key, ACL: 'private' })
188
188
  }
189
189
 
190
- async function p6S3PublicBucketObjectAcl(event: S3Event): Promise<void> {
190
+ async function smileS3PublicBucketObjectAcl(event: S3Event): Promise<void> {
191
191
  const key = event.detail.requestParameters.key
192
192
  const bucket = event.detail.requestParameters.bucketName
193
193
 
@@ -196,7 +196,7 @@ async function p6S3PublicBucketObjectAcl(event: S3Event): Promise<void> {
196
196
  }
197
197
  }
198
198
 
199
- async function p6S3PublicBucketAccessBlock(event: S3Event): Promise<void> {
199
+ async function smileS3PublicBucketAccessBlock(event: S3Event): Promise<void> {
200
200
  const pbc = event.detail.requestParameters.PublicAccessBlockConfiguration
201
201
  logger.info(JSON.stringify(pbc))
202
202
 
@@ -218,7 +218,7 @@ async function p6S3PublicBucketAccessBlock(event: S3Event): Promise<void> {
218
218
  }
219
219
  }
220
220
 
221
- async function p6S3PublicAccessBlock(event: S3Event): Promise<void> {
221
+ async function smileS3PublicAccessBlock(event: S3Event): Promise<void> {
222
222
  const pbc = event.detail.requestParameters.PublicAccessBlockConfiguration
223
223
  logger.info(JSON.stringify(pbc))
224
224
 
@@ -241,7 +241,7 @@ async function p6S3PublicAccessBlock(event: S3Event): Promise<void> {
241
241
  }
242
242
  }
243
243
 
244
- async function p6S3PublicFusebox(event: S3Event): Promise<boolean> {
244
+ async function smileS3PublicFusebox(event: S3Event): Promise<boolean> {
245
245
  if (!event.detail || !event.detail.eventName) {
246
246
  return false
247
247
  }
@@ -260,23 +260,23 @@ async function p6S3PublicFusebox(event: S3Event): Promise<boolean> {
260
260
  }
261
261
 
262
262
  if (eventName === 'PutBucketAcl') {
263
- await p6S3PublicBucketAcl(event)
263
+ await smileS3PublicBucketAcl(event)
264
264
  }
265
265
  else if (eventName === 'PutObjectAcl') {
266
- await p6S3PublicBucketObjectAcl(event)
266
+ await smileS3PublicBucketObjectAcl(event)
267
267
  }
268
268
  else if (eventName === 'PutBucketPublicAccessBlock') {
269
- await p6S3PublicBucketAccessBlock(event)
269
+ await smileS3PublicBucketAccessBlock(event)
270
270
  }
271
271
  else if (eventName === 'PutAccountPublicAccessBlock') {
272
- await p6S3PublicAccessBlock(event)
272
+ await smileS3PublicAccessBlock(event)
273
273
  }
274
274
 
275
275
  return true
276
276
  }
277
277
 
278
278
  export async function handler(event: S3Event, _context?: Context): Promise<boolean> {
279
- await p6S3PublicFusebox(event)
279
+ await smileS3PublicFusebox(event)
280
280
  return true
281
281
  }
282
282