p6-cdk-s3-protector 0.0.1 → 0.0.3
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 +9 -9
- package/README.md +4 -3
- package/lib/p6cdks3protector.js +1 -1
- package/lib/p6cdks3protector.p6CDKS3Protector.js +23 -23
- package/package.json +12 -12
- package/src/p6cdks3protector.p6CDKS3Protector.ts +22 -22
package/.jsii
CHANGED
|
@@ -8,9 +8,9 @@
|
|
|
8
8
|
]
|
|
9
9
|
},
|
|
10
10
|
"bundled": {
|
|
11
|
-
"@aws-sdk/client-s3": "^3.
|
|
12
|
-
"@aws-sdk/client-s3-control": "^3.
|
|
13
|
-
"@aws-sdk/client-sts": "^3.
|
|
11
|
+
"@aws-sdk/client-s3": "^3.682.0",
|
|
12
|
+
"@aws-sdk/client-s3-control": "^3.682.0",
|
|
13
|
+
"@aws-sdk/client-sts": "^3.682.0",
|
|
14
14
|
"@types/aws-lambda": "^8.10.145",
|
|
15
15
|
"aws-sdk": "^2.1691.0",
|
|
16
16
|
"cdk-iam-floyd": "^0.658.0",
|
|
@@ -3866,15 +3866,15 @@
|
|
|
3866
3866
|
}
|
|
3867
3867
|
}
|
|
3868
3868
|
},
|
|
3869
|
-
"description": "AWS CDK:
|
|
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
|
-
"
|
|
3877
|
-
"
|
|
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
|
|
3890
|
+
"markdown": "AWS CDK: A real-time S3 Protector\n\n# P6CDKS3Protector\n\n## LICENSE\n\n[](https://opensource.org/licenses/Apache-2.0)\n\n## Other\n\n[](https://gitpod.io/#https://github.com/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\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.
|
|
3955
|
-
"fingerprint": "
|
|
3954
|
+
"version": "0.0.3",
|
|
3955
|
+
"fingerprint": "i8c0ygBbVjf5YtUApVd9jWQiP70l233TMozw1i6tdDE="
|
|
3956
3956
|
}
|
package/README.md
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
AWS CDK: A real-time S3
|
|
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
|
-
|
|
17
|
+
import { P6CDKS3Protector } from 'p6-cdk-s3-protector'
|
|
18
|
+
|
|
19
|
+
new P6CDKS3Protector(this, 'p6CDKS3Protector')
|
|
19
20
|
```
|
|
20
21
|
|
|
21
22
|
## Architecture
|
package/lib/p6cdks3protector.js
CHANGED
|
@@ -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.
|
|
34
|
+
static [JSII_RTTI_SYMBOL_1] = { fqn: "p6-cdk-s3-protector.P6CDKS3Protector", version: "0.0.3" };
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
121
|
-
if (
|
|
120
|
+
async function smileS3PublicBucketAcl(event) {
|
|
121
|
+
if (smileShortCircuitShould(event)) {
|
|
122
122
|
return true;
|
|
123
123
|
}
|
|
124
|
-
if (
|
|
124
|
+
if (smileLoopPrevent(event)) {
|
|
125
125
|
return true;
|
|
126
126
|
}
|
|
127
|
-
const bucketAcl = await
|
|
127
|
+
const bucketAcl = await smileS3BucketAclGet(event);
|
|
128
128
|
if (!bucketAcl) {
|
|
129
129
|
return false;
|
|
130
130
|
}
|
|
131
|
-
const [uriList, preserveLogDelivery] =
|
|
132
|
-
if (
|
|
133
|
-
await
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
217
|
+
await smileS3PublicBucketAcl(event);
|
|
218
218
|
}
|
|
219
219
|
else if (eventName === 'PutObjectAcl') {
|
|
220
|
-
await
|
|
220
|
+
await smileS3PublicBucketObjectAcl(event);
|
|
221
221
|
}
|
|
222
222
|
else if (eventName === 'PutBucketPublicAccessBlock') {
|
|
223
|
-
await
|
|
223
|
+
await smileS3PublicBucketAccessBlock(event);
|
|
224
224
|
}
|
|
225
225
|
else if (eventName === 'PutAccountPublicAccessBlock') {
|
|
226
|
-
await
|
|
226
|
+
await smileS3PublicAccessBlock(event);
|
|
227
227
|
}
|
|
228
228
|
return true;
|
|
229
229
|
}
|
|
230
230
|
async function handler(event, _context) {
|
|
231
|
-
await
|
|
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:
|
|
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"
|
|
@@ -52,9 +52,9 @@
|
|
|
52
52
|
"@swc-node/register": "^1.10.9",
|
|
53
53
|
"@types/aws-lambda": "^8.10.145",
|
|
54
54
|
"@types/jest": "^29.5.14",
|
|
55
|
-
"@types/node": "22.8.
|
|
56
|
-
"@typescript-eslint/eslint-plugin": "^8.12.
|
|
57
|
-
"@typescript-eslint/parser": "^8.12.
|
|
55
|
+
"@types/node": "22.8.4",
|
|
56
|
+
"@typescript-eslint/eslint-plugin": "^8.12.2",
|
|
57
|
+
"@typescript-eslint/parser": "^8.12.2",
|
|
58
58
|
"aws-cdk": "^2.164.1",
|
|
59
59
|
"aws-cdk-lib": "2.164.1",
|
|
60
60
|
"cdk-dia": "^0.11.0",
|
|
@@ -68,7 +68,7 @@
|
|
|
68
68
|
"jsii-diff": "^1.104.0",
|
|
69
69
|
"jsii-docgen": "^10.5.5",
|
|
70
70
|
"jsii-pacmak": "^1.104.0",
|
|
71
|
-
"publib": "^0.2.
|
|
71
|
+
"publib": "^0.2.907",
|
|
72
72
|
"ts-jest": "^29.2.5",
|
|
73
73
|
"ts-node": "^10.9.2",
|
|
74
74
|
"typescript": "~5.6.3"
|
|
@@ -78,9 +78,9 @@
|
|
|
78
78
|
"constructs": "^10.4.2"
|
|
79
79
|
},
|
|
80
80
|
"dependencies": {
|
|
81
|
-
"@aws-sdk/client-s3": "^3.
|
|
82
|
-
"@aws-sdk/client-s3-control": "^3.
|
|
83
|
-
"@aws-sdk/client-sts": "^3.
|
|
81
|
+
"@aws-sdk/client-s3": "^3.682.0",
|
|
82
|
+
"@aws-sdk/client-s3-control": "^3.682.0",
|
|
83
|
+
"@aws-sdk/client-sts": "^3.682.0",
|
|
84
84
|
"@types/aws-lambda": "^8.10.145",
|
|
85
85
|
"aws-cdk-lib": "2.164.1",
|
|
86
86
|
"aws-sdk": "^2.1691.0",
|
|
@@ -103,13 +103,13 @@
|
|
|
103
103
|
"aws",
|
|
104
104
|
"cdk",
|
|
105
105
|
"s3",
|
|
106
|
-
"
|
|
107
|
-
"
|
|
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.
|
|
112
|
+
"version": "0.0.3",
|
|
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.
|
|
140
|
+
"packageManager": "pnpm@9.12.3+sha512.cce0f9de9c5a7c95bef944169cc5dfe8741abfb145078c0d508b868056848a87c81e626246cb60967cbd7fd29a6c062ef73ff840d96b3c86c40ac92cf4a813ee"
|
|
141
141
|
}
|
|
@@ -33,7 +33,7 @@ interface S3Event {
|
|
|
33
33
|
}
|
|
34
34
|
}
|
|
35
35
|
|
|
36
|
-
function
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
143
|
-
if (
|
|
142
|
+
async function smileS3PublicBucketAcl(event: S3Event): Promise<boolean> {
|
|
143
|
+
if (smileShortCircuitShould(event)) {
|
|
144
144
|
return true
|
|
145
145
|
}
|
|
146
146
|
|
|
147
|
-
if (
|
|
147
|
+
if (smileLoopPrevent(event)) {
|
|
148
148
|
return true
|
|
149
149
|
}
|
|
150
150
|
|
|
151
|
-
const bucketAcl = await
|
|
151
|
+
const bucketAcl = await smileS3BucketAclGet(event)
|
|
152
152
|
if (!bucketAcl) {
|
|
153
153
|
return false
|
|
154
154
|
}
|
|
155
155
|
|
|
156
|
-
const [uriList, preserveLogDelivery] =
|
|
156
|
+
const [uriList, preserveLogDelivery] = smileLogDeliveryPreserve(bucketAcl)
|
|
157
157
|
|
|
158
|
-
if (
|
|
159
|
-
await
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
263
|
+
await smileS3PublicBucketAcl(event)
|
|
264
264
|
}
|
|
265
265
|
else if (eventName === 'PutObjectAcl') {
|
|
266
|
-
await
|
|
266
|
+
await smileS3PublicBucketObjectAcl(event)
|
|
267
267
|
}
|
|
268
268
|
else if (eventName === 'PutBucketPublicAccessBlock') {
|
|
269
|
-
await
|
|
269
|
+
await smileS3PublicBucketAccessBlock(event)
|
|
270
270
|
}
|
|
271
271
|
else if (eventName === 'PutAccountPublicAccessBlock') {
|
|
272
|
-
await
|
|
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
|
|
279
|
+
await smileS3PublicFusebox(event)
|
|
280
280
|
return true
|
|
281
281
|
}
|
|
282
282
|
|