UNPKG

@k9securityio/k9-cdk

Version:

Provision strong AWS security policies easily using the AWS CDK.

92 lines 15.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.SID_DENY_EVERYONE_ELSE = void 0; exports.makeResourcePolicy = makeResourcePolicy; exports.grantAccessViaResourcePolicy = grantAccessViaResourcePolicy; const aws_iam_1 = require("aws-cdk-lib/aws-iam"); const iam = require("aws-cdk-lib/aws-iam"); const k9policy_1 = require("./k9policy"); let SUPPORTED_CAPABILITIES = new Array(k9policy_1.AccessCapability.ADMINISTER_RESOURCE, k9policy_1.AccessCapability.READ_CONFIG, k9policy_1.AccessCapability.WRITE_DATA); exports.SID_DENY_EVERYONE_ELSE = 'DenyEveryoneElse'; /** * Generate an EventBridge Bus resource policy from the provided props. * * @param props specifying desired access * @return a PolicyDocument that can be attached to an EventBridge event bus */ function makeResourcePolicy(props) { const policyFactory = new k9policy_1.K9PolicyFactory(); const policy = new iam.PolicyDocument(); const resourceArns = ['*']; (0, k9policy_1.validateAccessSpecs)(props.k9DesiredAccess); let accessSpecsByCapabilityRecs = policyFactory.mergeDesiredAccessSpecsByCapability(SUPPORTED_CAPABILITIES, props.k9DesiredAccess); let accessSpecsByCapability = new Map(); for (let [capabilityStr, accessSpec] of Object.entries(accessSpecsByCapabilityRecs)) { accessSpecsByCapability.set((0, k9policy_1.getAccessCapabilityFromValue)(capabilityStr), accessSpec); } if (!(0, k9policy_1.canPrincipalsManageResources)(accessSpecsByCapability)) { throw Error('At least one principal must be able to administer and read-config for EventBridge resources' + ' so the bus remains accessible; found:\n' + `administer-resource: '${accessSpecsByCapability.get(k9policy_1.AccessCapability.ADMINISTER_RESOURCE)?.allowPrincipalArns}'\n` + `read-config: '${accessSpecsByCapability.get(k9policy_1.AccessCapability.READ_CONFIG)?.allowPrincipalArns}'`); } const allowStatements = policyFactory.makeAllowStatements('EventBridge', SUPPORTED_CAPABILITIES, Array.from(accessSpecsByCapability.values()), resourceArns, true); policy.addStatements(...allowStatements); // DenyEveryoneElse — conditional on access pattern: // When wildcard + org constraint is used, skip DenyEveryoneElse because // putting "*" in the deny exception would exempt everyone. // The org constraint on the Allow side already limits access. if (!(0, k9policy_1.hasWildcardPrincipal)(props.k9DesiredAccess)) { const denyEveryoneElseStatement = new aws_iam_1.PolicyStatement({ sid: exports.SID_DENY_EVERYONE_ELSE, effect: aws_iam_1.Effect.DENY, principals: policyFactory.makeDenyEveryoneElsePrincipals(), actions: ['events:*'], resources: resourceArns, }); denyEveryoneElseStatement.addCondition('Bool', { 'aws:PrincipalIsAWSService': ['false'], }); const denyEveryoneElseTest = policyFactory.wasLikeUsed(props.k9DesiredAccess) ? 'ArnNotLike' : 'ArnNotEquals'; const allAllowedPrincipalArns = policyFactory.getAllowedPrincipalArns(props.k9DesiredAccess); const accountRootPrincipal = new aws_iam_1.AccountRootPrincipal(); denyEveryoneElseStatement.addCondition(denyEveryoneElseTest, { 'aws:PrincipalArn': [ // Place Root Principal arn in stable, prominent position; // will render as an object Fn::Join'ing Partition & AccountId accountRootPrincipal.arn, ...allAllowedPrincipalArns, ], }); policy.addStatements(denyEveryoneElseStatement); } const denyUntrustedOrgsStatement = policyFactory._makeDenyUntrustedOrgsStatement('EventBridge', SUPPORTED_CAPABILITIES, accessSpecsByCapability, resourceArns); if (denyUntrustedOrgsStatement) { policy.addStatements(denyUntrustedOrgsStatement); } policy.validateForResourcePolicy(); return policy; } /** * Grant access to an event bus via resource policy using k9 IAccessSpec definitions. * * @param props specifying the event bus and desired access * * @return the results for adding each statement */ function grantAccessViaResourcePolicy(props) { const resourcePolicy = makeResourcePolicy(props); resourcePolicy.validateForResourcePolicy(); const policyJson = resourcePolicy.toJSON(); const k9Statements = policyJson.Statement; const bus = props.bus; const addToResourcePolicyResults = new Array(); for (let statement of k9Statements) { let addToResourcePolicyResult = bus.addToResourcePolicy(aws_iam_1.PolicyStatement.fromJson(statement)); addToResourcePolicyResults.push(addToResourcePolicyResult); } return addToResourcePolicyResults; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"events.js","sourceRoot":"","sources":["../src/events.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AccountRootPrincipal,\n  AddToResourcePolicyResult,\n  Effect,\n  PolicyDocument,\n  PolicyStatement,\n} from 'aws-cdk-lib/aws-iam';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport { EventBus } from 'aws-cdk-lib/aws-events';\nimport {\n  AccessCapability,\n  canPrincipalsManageResources,\n  getAccessCapabilityFromValue,\n  hasWildcardPrincipal,\n  IAccessSpec,\n  K9PolicyFactory,\n  validateAccessSpecs,\n} from './k9policy';\n\n\nexport interface K9EventBusResourcePolicyProps {\n  readonly bus: EventBus;\n  readonly k9DesiredAccess: Array<IAccessSpec>;\n}\n\nlet SUPPORTED_CAPABILITIES = new Array<AccessCapability>(\n  AccessCapability.ADMINISTER_RESOURCE,\n  AccessCapability.READ_CONFIG,\n  AccessCapability.WRITE_DATA,\n);\n\nexport const SID_DENY_EVERYONE_ELSE = 'DenyEveryoneElse';\n\n/**\n * Generate an EventBridge Bus resource policy from the provided props.\n *\n * @param props specifying desired access\n * @return a PolicyDocument that can be attached to an EventBridge event bus\n */\nexport function makeResourcePolicy(props: K9EventBusResourcePolicyProps): PolicyDocument {\n  const policyFactory = new K9PolicyFactory();\n  const policy = new iam.PolicyDocument();\n\n  const resourceArns = ['*'];\n\n  validateAccessSpecs(props.k9DesiredAccess);\n\n  let accessSpecsByCapabilityRecs = policyFactory.mergeDesiredAccessSpecsByCapability(SUPPORTED_CAPABILITIES, props.k9DesiredAccess);\n  let accessSpecsByCapability: Map<AccessCapability, IAccessSpec> = new Map();\n\n  for (let [capabilityStr, accessSpec] of Object.entries(accessSpecsByCapabilityRecs)) {\n    accessSpecsByCapability.set(getAccessCapabilityFromValue(capabilityStr), accessSpec);\n  }\n\n  if (!canPrincipalsManageResources(accessSpecsByCapability)) {\n    throw Error('At least one principal must be able to administer and read-config for EventBridge resources' +\n            ' so the bus remains accessible; found:\\n' +\n            `administer-resource: '${accessSpecsByCapability.get(AccessCapability.ADMINISTER_RESOURCE)?.allowPrincipalArns}'\\n` +\n            `read-config: '${accessSpecsByCapability.get(AccessCapability.READ_CONFIG)?.allowPrincipalArns}'`,\n    );\n  }\n\n  const allowStatements = policyFactory.makeAllowStatements('EventBridge',\n    SUPPORTED_CAPABILITIES,\n    Array.from(accessSpecsByCapability.values()),\n    resourceArns,\n    true);\n  policy.addStatements(...allowStatements);\n\n  // DenyEveryoneElse — conditional on access pattern:\n  // When wildcard + org constraint is used, skip DenyEveryoneElse because\n  // putting \"*\" in the deny exception would exempt everyone.\n  // The org constraint on the Allow side already limits access.\n  if (!hasWildcardPrincipal(props.k9DesiredAccess)) {\n    const denyEveryoneElseStatement = new PolicyStatement({\n      sid: SID_DENY_EVERYONE_ELSE,\n      effect: Effect.DENY,\n      principals: policyFactory.makeDenyEveryoneElsePrincipals(),\n      actions: ['events:*'],\n      resources: resourceArns,\n    });\n    denyEveryoneElseStatement.addCondition('Bool', {\n      'aws:PrincipalIsAWSService': ['false'],\n    });\n    const denyEveryoneElseTest = policyFactory.wasLikeUsed(props.k9DesiredAccess) ?\n      'ArnNotLike' :\n      'ArnNotEquals';\n    const allAllowedPrincipalArns = policyFactory.getAllowedPrincipalArns(props.k9DesiredAccess);\n    const accountRootPrincipal = new AccountRootPrincipal();\n    denyEveryoneElseStatement.addCondition(denyEveryoneElseTest, {\n      'aws:PrincipalArn': [\n        // Place Root Principal arn in stable, prominent position;\n        // will render as an object Fn::Join'ing Partition & AccountId\n        accountRootPrincipal.arn,\n        ...allAllowedPrincipalArns,\n      ],\n    });\n\n    policy.addStatements(denyEveryoneElseStatement);\n  }\n\n  const denyUntrustedOrgsStatement = policyFactory._makeDenyUntrustedOrgsStatement(\n    'EventBridge', SUPPORTED_CAPABILITIES, accessSpecsByCapability, resourceArns);\n  if (denyUntrustedOrgsStatement) {\n    policy.addStatements(denyUntrustedOrgsStatement);\n  }\n\n  policy.validateForResourcePolicy();\n\n  return policy;\n}\n\n/**\n * Grant access to an event bus via resource policy using k9 IAccessSpec definitions.\n *\n * @param props specifying the event bus and desired access\n *\n * @return the results for adding each statement\n */\nexport function grantAccessViaResourcePolicy(props: K9EventBusResourcePolicyProps):\nAddToResourcePolicyResult[] {\n  const resourcePolicy = makeResourcePolicy(props);\n\n  resourcePolicy.validateForResourcePolicy();\n\n  const policyJson = resourcePolicy.toJSON();\n  const k9Statements = policyJson.Statement;\n  const bus = props.bus;\n  const addToResourcePolicyResults = new Array<AddToResourcePolicyResult>();\n\n  for (let statement of k9Statements) {\n    let addToResourcePolicyResult = bus.addToResourcePolicy(\n      PolicyStatement.fromJson(statement),\n    );\n    addToResourcePolicyResults.push(addToResourcePolicyResult);\n  }\n\n  return addToResourcePolicyResults;\n}\n"]}