UNPKG

@k9securityio/k9-cdk

Version:

Provision strong AWS security policies easily using the AWS CDK.

188 lines 30 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.grantAccessViaResourcePolicy = exports.CloudFrontOACReadAccessGenerator = exports.SID_ALLOW_PUBLIC_READ_ACCESS = exports.SID_DENY_UNENCRYPTED_STORAGE = exports.SID_DENY_UNEXPECTED_ENCRYPTION_METHOD = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const aws_iam_1 = require("aws-cdk-lib/aws-iam"); const s3 = require("aws-cdk-lib/aws-s3"); const aws_s3_1 = require("aws-cdk-lib/aws-s3"); const aws_iam_utils = require("./aws-iam-utils"); const k9policy_1 = require("./k9policy"); let SUPPORTED_CAPABILITIES = new Array(k9policy_1.AccessCapability.ADMINISTER_RESOURCE, k9policy_1.AccessCapability.READ_CONFIG, k9policy_1.AccessCapability.READ_DATA, k9policy_1.AccessCapability.WRITE_DATA, k9policy_1.AccessCapability.DELETE_DATA); exports.SID_DENY_UNEXPECTED_ENCRYPTION_METHOD = 'DenyUnexpectedEncryptionMethod'; exports.SID_DENY_UNENCRYPTED_STORAGE = 'DenyUnencryptedStorage'; exports.SID_ALLOW_PUBLIC_READ_ACCESS = 'AllowPublicReadAccess'; class CloudFrontOACReadAccessGenerator { constructor(bucket, distributionArn) { this.bucket = bucket; this.distributionArn = distributionArn; } makeAllowStatements() { return [new aws_iam_1.PolicyStatement({ sid: CloudFrontOACReadAccessGenerator.SID_ALLOW_CLOUDFRONT_OAC_READ_ACCESS, effect: aws_iam_1.Effect.ALLOW, principals: [new aws_iam_1.ServicePrincipal('cloudfront.amazonaws.com')], actions: ['s3:GetObject'], resources: [`${this.bucket.arnForObjects('*')}`], conditions: { StringEquals: { 'aws:SourceArn': this.distributionArn }, }, })]; } makeConditionsToExceptFromDenyEveryoneElse() { // return a (TypeScript) Record of the form: // {"Operator": { "keyInRequestContext": "value" } } return { StringNotEqualsIfExists: { 'aws:PrincipalServiceName': 'cloudfront.amazonaws.com' } }; } } exports.CloudFrontOACReadAccessGenerator = CloudFrontOACReadAccessGenerator; _a = JSII_RTTI_SYMBOL_1; CloudFrontOACReadAccessGenerator[_a] = { fqn: "@k9securityio/k9-cdk.s3.CloudFrontOACReadAccessGenerator", version: "2.1.3" }; CloudFrontOACReadAccessGenerator.SID_ALLOW_CLOUDFRONT_OAC_READ_ACCESS = 'AllowCloudFrontOACReadAccess'; /** * Grants least-privilege access to a bucket by generating a BucketPolicy from the access capabilities * described by `props`; the policy will be set on the Bucket specified in `props`. * * When a BucketPolicy already exists on the Bucket referenced in `props`: * * the BucketPolicy's existing Statements will pass through unmodified * * k9 will identify IAM principals there were allowed by the original policy and add those principals to * the `DenyEveryoneElse` Statement's exclusion list so that, e.g. autoDeleteObjects works as expected * * k9's Allow and Deny statements will be added to the policy * * @remarks * * k9 modifies the existing BucketPolicy in place instead of replacing or copying and modifying that * to preserve dependency references created by certain S3 CDK features such as `autoDeleteObjects`. * * @param scope The scope in which to define this construct. * @param id The scoped construct ID. * @param props describing the desired access capabilities for the bucket * * @return an array of AddToResourcePolicyResult */ function grantAccessViaResourcePolicy(scope, id, props) { const policyFactory = new k9policy_1.K9PolicyFactory(); // If the bucket already has a policy, use it. Maintaining the existing policy instance // is important because other CDK features like S3 autoDeleteObjects may have expressed dependencies // on that instance which must be maintained. if (!props.bucket.policy) { props.bucket.policy = new s3.BucketPolicy(scope, `${id}Policy`, { bucket: props.bucket }); } const policy = props.bucket.policy; const addToResourcePolicyResults = new Array(); let resourceArns = [ `${props.bucket.bucketArn}`, `${props.bucket.arnForObjects('*')}`, ]; // Capture the principals that were allowed prior to modifying policy // One could argue this can be done at the end because we're going to // narrow the DenyEveryoneElse to the unique set of allowed principals. // Record here for now to preserve ability to generate fine-grained DenyEveryoneElse-$capability statements. const origAllowedAWSPrincipals = aws_iam_utils.getAllowedPrincipalArns(policy.document); // Make Allow Statements const k9Statements = policyFactory.makeAllowStatements('S3', SUPPORTED_CAPABILITIES, props.k9DesiredAccess, resourceArns); if (props.publicReadAccess) { k9Statements.unshift(// very important statement; put at beginning. new aws_iam_1.PolicyStatement({ sid: exports.SID_ALLOW_PUBLIC_READ_ACCESS, effect: aws_iam_1.Effect.ALLOW, principals: [new aws_iam_1.AnyPrincipal()], actions: ['s3:GetObject'], resources: [`${props.bucket.arnForObjects('*')}`], })); } if (props.awsServiceAccessGenerators) { for (let serviceAccessSpec of props.awsServiceAccessGenerators) { let allowStatements = serviceAccessSpec.makeAllowStatements(); k9Statements.unshift(...allowStatements); } } // Make Deny Statement const denyEveryoneElseTest = policyFactory.wasLikeUsed(props.k9DesiredAccess) ? 'ArnNotLike' : 'ArnNotEquals'; let denyEveryoneElseStatement; if (props.publicReadAccess) { denyEveryoneElseStatement = new aws_iam_1.PolicyStatement({ sid: 'DenyEveryoneElse', effect: aws_iam_1.Effect.DENY, principals: policyFactory.makeDenyEveryoneElsePrincipals(), notActions: ['s3:GetObject'], resources: resourceArns, }); } else { denyEveryoneElseStatement = new aws_iam_1.PolicyStatement({ sid: 'DenyEveryoneElse', effect: aws_iam_1.Effect.DENY, principals: policyFactory.makeDenyEveryoneElsePrincipals(), actions: ['s3:*'], resources: resourceArns, }); } const allAllowedPrincipalArns = new Set(policyFactory.getAllowedPrincipalArns(props.k9DesiredAccess)); for (let origAWSPrincipal of origAllowedAWSPrincipals) { allAllowedPrincipalArns.add(origAWSPrincipal); } denyEveryoneElseStatement.addCondition(denyEveryoneElseTest, { 'aws:PrincipalArn': [...allAllowedPrincipalArns] }); if (props.awsServiceAccessGenerators) { for (let serviceAccessSpec of props.awsServiceAccessGenerators) { let conditionsToExceptFromDenyEveryoneElse = serviceAccessSpec.makeConditionsToExceptFromDenyEveryoneElse(); let conditionOps = Object.keys(conditionsToExceptFromDenyEveryoneElse); for (let conditionOp of conditionOps) { // note: when you call PolicyStatement#addCondition with the same conditionOp (e.g. StringEquals) // multiple times, addCondition will collect the values into an array. // c.f. https://github.com/aws/aws-cdk/blob/main/packages/aws-cdk-lib/aws-iam/lib/policy-statement.ts#L364 denyEveryoneElseStatement.addCondition(conditionOp, conditionsToExceptFromDenyEveryoneElse[conditionOp]); } } } // default encryption method to SSE-KMS, // allow override to SSE-S3 (AES256) let encryptionMethod = 'aws:kms'; if (props.encryption) { if (aws_s3_1.BucketEncryption.S3_MANAGED == props.encryption) { encryptionMethod = 'AES256'; } } k9Statements.push(new aws_iam_1.PolicyStatement({ sid: 'DenyInsecureCommunications', effect: aws_iam_1.Effect.DENY, principals: [new aws_iam_1.AnyPrincipal()], actions: ['s3:*'], resources: resourceArns, conditions: { Bool: { 'aws:SecureTransport': false }, }, })); if (props.enforceEncryptionAtRest ?? true) { k9Statements.push(new aws_iam_1.PolicyStatement({ sid: exports.SID_DENY_UNENCRYPTED_STORAGE, effect: aws_iam_1.Effect.DENY, principals: [new aws_iam_1.AnyPrincipal()], actions: ['s3:PutObject', 's3:ReplicateObject'], resources: resourceArns, conditions: { Null: { 's3:x-amz-server-side-encryption': true }, }, }), new aws_iam_1.PolicyStatement({ sid: exports.SID_DENY_UNEXPECTED_ENCRYPTION_METHOD, effect: aws_iam_1.Effect.DENY, principals: [new aws_iam_1.AnyPrincipal()], actions: ['s3:PutObject', 's3:ReplicateObject'], resources: resourceArns, conditions: { StringNotEquals: { 's3:x-amz-server-side-encryption': encryptionMethod }, }, })); } k9Statements.push(denyEveryoneElseStatement); for (let statement of k9Statements) { let addToResourcePolicyResult = props.bucket.addToResourcePolicy(statement); addToResourcePolicyResults.push(addToResourcePolicyResult); } policy.document.validateForResourcePolicy(); return addToResourcePolicyResults; } exports.grantAccessViaResourcePolicy = grantAccessViaResourcePolicy; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"s3.js","sourceRoot":"","sources":["../src/s3.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AddToResourcePolicyResult,\n  AnyPrincipal,\n  Conditions,\n  Effect,\n  PolicyStatement,\n  ServicePrincipal,\n} from 'aws-cdk-lib/aws-iam';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { IBucket, BucketEncryption } from 'aws-cdk-lib/aws-s3';\nimport { IConstruct } from 'constructs';\nimport * as aws_iam_utils from './aws-iam-utils';\nimport { AccessCapability, IAccessSpec, IAWSServiceAccessGenerator, K9PolicyFactory } from './k9policy';\n\n/**\n * Configure the k9 Security S3 Bucket policy generator with the K9BucketPolicyProps.\n */\nexport interface K9BucketPolicyProps extends s3.BucketPolicyProps {\n  /**\n   * An array of IAccessSpec defining the desired access.  The policy\n   * generator will combine and normalize overlapping access specs.\n   */\n  readonly k9DesiredAccess: Array<IAccessSpec>;\n\n  /**\n   * (Optionally) Provide the BucketEncryption object for the Bucket to\n   * allow the policy generator to customize the policy for the Bucket's\n   * configuration without handling, e.g. the encryption method options directly\n   */\n  readonly encryption?: BucketEncryption;\n\n  /**\n   * Enforce encryption at rest with policy conditions.  The policy will use\n   * the encryption method defined by the encryption property or default to `aws:kms`.\n   *\n   * @default true\n   */\n  readonly enforceEncryptionAtRest?: boolean;\n\n  /**\n   * Allow public read access to the bucket.\n   *\n   * @default false\n   */\n  readonly publicReadAccess?: boolean;\n\n  /**\n   * An (optional) array of IAWSServiceAccessGenerator instances which will generate statements to allow access to the\n   * bucket or bucket object(s) by an AWS service like CloudFront or Kinesis.\n   *\n   * @default undefined\n   */\n  readonly awsServiceAccessGenerators?: Array<IAWSServiceAccessGenerator>;\n}\n\nlet SUPPORTED_CAPABILITIES = new Array<AccessCapability>(\n  AccessCapability.ADMINISTER_RESOURCE,\n  AccessCapability.READ_CONFIG,\n  AccessCapability.READ_DATA,\n  AccessCapability.WRITE_DATA,\n  AccessCapability.DELETE_DATA,\n);\n\nexport const SID_DENY_UNEXPECTED_ENCRYPTION_METHOD = 'DenyUnexpectedEncryptionMethod';\nexport const SID_DENY_UNENCRYPTED_STORAGE = 'DenyUnencryptedStorage';\nexport const SID_ALLOW_PUBLIC_READ_ACCESS = 'AllowPublicReadAccess';\n\nexport class CloudFrontOACReadAccessGenerator implements IAWSServiceAccessGenerator {\n\n  static readonly SID_ALLOW_CLOUDFRONT_OAC_READ_ACCESS = 'AllowCloudFrontOACReadAccess';\n\n  readonly bucket: IBucket;\n  readonly distributionArn: string;\n\n  constructor(bucket: IBucket, distributionArn: string) {\n    this.bucket = bucket;\n    this.distributionArn = distributionArn;\n  }\n\n  makeAllowStatements(): Array<PolicyStatement> {\n    return [new PolicyStatement({\n      sid: CloudFrontOACReadAccessGenerator.SID_ALLOW_CLOUDFRONT_OAC_READ_ACCESS,\n      effect: Effect.ALLOW,\n      principals: [new ServicePrincipal('cloudfront.amazonaws.com')],\n      actions: ['s3:GetObject'],\n      resources: [`${this.bucket.arnForObjects('*')}`],\n      conditions: {\n        StringEquals: { 'aws:SourceArn': this.distributionArn },\n      },\n    })];\n  }\n\n  makeConditionsToExceptFromDenyEveryoneElse(): Conditions {\n    // return a (TypeScript) Record of the form:\n    //     {\"Operator\": { \"keyInRequestContext\": \"value\" } }\n    return { StringNotEqualsIfExists: { 'aws:PrincipalServiceName': 'cloudfront.amazonaws.com' } };\n  }\n}\n\n/**\n * Grants least-privilege access to a bucket by generating a BucketPolicy from the access capabilities\n * described by `props`; the policy will be set on the Bucket specified in `props`.\n *\n * When a BucketPolicy already exists on the Bucket referenced in `props`:\n *   * the BucketPolicy's existing Statements will pass through unmodified\n *   * k9 will identify IAM principals there were allowed by the original policy and add those principals to\n *   the `DenyEveryoneElse` Statement's exclusion list so that, e.g. autoDeleteObjects works as expected\n *   * k9's Allow and Deny statements will be added to the policy\n *\n * @remarks\n *\n * k9 modifies the existing BucketPolicy in place instead of replacing or copying and modifying that\n * to preserve dependency references created by certain S3 CDK features such as `autoDeleteObjects`.\n *\n * @param scope The scope in which to define this construct.\n * @param id The scoped construct ID.\n * @param props describing the desired access capabilities for the bucket\n *\n * @return an array of AddToResourcePolicyResult\n */\nexport function grantAccessViaResourcePolicy(scope: IConstruct, id: string, props: K9BucketPolicyProps): AddToResourcePolicyResult[] {\n  const policyFactory = new K9PolicyFactory();\n  // If the bucket already has a policy, use it.  Maintaining the existing policy instance\n  // is important because other CDK features like S3 autoDeleteObjects may have expressed dependencies\n  // on that instance which must be maintained.\n  if (!props.bucket.policy) {\n    props.bucket.policy = new s3.BucketPolicy(scope, `${id}Policy`, { bucket: props.bucket });\n  }\n  const policy = props.bucket.policy;\n\n  const addToResourcePolicyResults = new Array<AddToResourcePolicyResult>();\n  let resourceArns = [\n    `${props.bucket.bucketArn}`,\n    `${props.bucket.arnForObjects('*')}`,\n  ];\n\n  // Capture the principals that were allowed prior to modifying policy\n  // One could argue this can be done at the end because we're going to\n  // narrow the DenyEveryoneElse to the unique set of allowed principals.\n  // Record here for now to preserve ability to generate fine-grained DenyEveryoneElse-$capability statements.\n  const origAllowedAWSPrincipals = aws_iam_utils.getAllowedPrincipalArns(policy.document);\n\n  // Make Allow Statements\n  const k9Statements = policyFactory.makeAllowStatements('S3',\n    SUPPORTED_CAPABILITIES,\n    props.k9DesiredAccess,\n    resourceArns);\n\n  if (props.publicReadAccess) {\n    k9Statements.unshift( // very important statement; put at beginning.\n      new PolicyStatement({\n        sid: SID_ALLOW_PUBLIC_READ_ACCESS,\n        effect: Effect.ALLOW,\n        principals: [new AnyPrincipal()],\n        actions: ['s3:GetObject'],\n        resources: [`${props.bucket.arnForObjects('*')}`],\n      }),\n    );\n  }\n\n  if (props.awsServiceAccessGenerators) {\n    for (let serviceAccessSpec of props.awsServiceAccessGenerators) {\n      let allowStatements:Array<PolicyStatement> = serviceAccessSpec.makeAllowStatements();\n      k9Statements.unshift(...allowStatements);\n    }\n  }\n\n  // Make Deny Statement\n  const denyEveryoneElseTest = policyFactory.wasLikeUsed(props.k9DesiredAccess) ?\n    'ArnNotLike' :\n    'ArnNotEquals';\n  let denyEveryoneElseStatement: PolicyStatement;\n\n  if (props.publicReadAccess) {\n    denyEveryoneElseStatement = new PolicyStatement({\n      sid: 'DenyEveryoneElse',\n      effect: Effect.DENY,\n      principals: policyFactory.makeDenyEveryoneElsePrincipals(),\n      notActions: ['s3:GetObject'],\n      resources: resourceArns,\n    });\n  } else {\n    denyEveryoneElseStatement = new PolicyStatement({\n      sid: 'DenyEveryoneElse',\n      effect: Effect.DENY,\n      principals: policyFactory.makeDenyEveryoneElsePrincipals(),\n      actions: ['s3:*'],\n      resources: resourceArns,\n    });\n  }\n  const allAllowedPrincipalArns = new Set<string>(policyFactory.getAllowedPrincipalArns(props.k9DesiredAccess));\n  for (let origAWSPrincipal of origAllowedAWSPrincipals) {\n    allAllowedPrincipalArns.add(origAWSPrincipal);\n  }\n  denyEveryoneElseStatement.addCondition(denyEveryoneElseTest,\n    { 'aws:PrincipalArn': [...allAllowedPrincipalArns] });\n\n  if (props.awsServiceAccessGenerators) {\n    for (let serviceAccessSpec of props.awsServiceAccessGenerators) {\n      let conditionsToExceptFromDenyEveryoneElse = serviceAccessSpec.makeConditionsToExceptFromDenyEveryoneElse();\n      let conditionOps = Object.keys(conditionsToExceptFromDenyEveryoneElse) as Array<string>;\n      for (let conditionOp of conditionOps) {\n        // note: when you call PolicyStatement#addCondition with the same conditionOp (e.g. StringEquals)\n        // multiple times, addCondition will collect the values into an array.\n        // c.f. https://github.com/aws/aws-cdk/blob/main/packages/aws-cdk-lib/aws-iam/lib/policy-statement.ts#L364\n        denyEveryoneElseStatement.addCondition(conditionOp, conditionsToExceptFromDenyEveryoneElse[conditionOp]);\n      }\n    }\n  }\n\n  // default encryption method to SSE-KMS,\n  // allow override to SSE-S3 (AES256)\n  let encryptionMethod = 'aws:kms';\n  if (props.encryption) {\n    if (BucketEncryption.S3_MANAGED == props.encryption) {\n      encryptionMethod = 'AES256';\n    }\n  }\n  k9Statements.push(\n    new PolicyStatement({\n      sid: 'DenyInsecureCommunications',\n      effect: Effect.DENY,\n      principals: [new AnyPrincipal()],\n      actions: ['s3:*'],\n      resources: resourceArns,\n      conditions: {\n        Bool: { 'aws:SecureTransport': false },\n      },\n    }),\n  );\n\n  if (props.enforceEncryptionAtRest ?? true) {\n    k9Statements.push(\n      new PolicyStatement({\n        sid: SID_DENY_UNENCRYPTED_STORAGE,\n        effect: Effect.DENY,\n        principals: [new AnyPrincipal()],\n        actions: ['s3:PutObject', 's3:ReplicateObject'],\n        resources: resourceArns,\n        conditions: {\n          Null: { 's3:x-amz-server-side-encryption': true },\n        },\n      },\n      ),\n      new PolicyStatement({\n        sid: SID_DENY_UNEXPECTED_ENCRYPTION_METHOD,\n        effect: Effect.DENY,\n        principals: [new AnyPrincipal()],\n        actions: ['s3:PutObject', 's3:ReplicateObject'],\n        resources: resourceArns,\n        conditions: {\n          StringNotEquals: { 's3:x-amz-server-side-encryption': encryptionMethod },\n        },\n      },\n      ));\n  }\n\n  k9Statements.push(denyEveryoneElseStatement);\n\n  for (let statement of k9Statements) {\n    let addToResourcePolicyResult = props.bucket.addToResourcePolicy(statement);\n    addToResourcePolicyResults.push(addToResourcePolicyResult);\n  }\n\n  policy.document.validateForResourcePolicy();\n\n  return addToResourcePolicyResults;\n}\n"]}