UNPKG

@k9securityio/k9-cdk

Version:

Provision strong AWS security policies easily using the AWS CDK.

116 lines 18.9 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.READ_DATA, k9policy_1.AccessCapability.WRITE_DATA, k9policy_1.AccessCapability.DELETE_DATA); exports.SID_DENY_EVERYONE_ELSE = 'DenyEveryoneElse'; function partitionArray(arr, maxLength) { const result = []; for (let i = 0; i < arr.length; i += maxLength) { result.push(arr.slice(i, i + maxLength)); } return result; } /** * Generate a SQS resource policy from the provided props that can be attached to a queue. * * @param props specifying desired access * @return a PolicyDocument that can be attached to an SQS queue */ function makeResourcePolicy(props) { const policyFactory = new k9policy_1.K9PolicyFactory(); const policy = new iam.PolicyDocument(); const resourceArns = ['*']; 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 SQS resources' + ' so data 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('SQS', SUPPORTED_CAPABILITIES, Array.from(accessSpecsByCapability.values()), resourceArns); const max_actions_in_statement = 7; for (let allowStatement of allowStatements) { //SQS resource policy has a limit of 7 actions per statement (Really). //But you can have as many statements as you want up to the queue policy size limit. //So, if an allowStatement has more than 7 actions (like the administer-resource statement does), //then create additional statements and spread the original statement's permissions across them if (allowStatement.actions.length > max_actions_in_statement) { const partitionedActions = partitionArray(allowStatement.actions, max_actions_in_statement); partitionedActions.forEach((actions, index) => { const newStatement = allowStatement.copy({ sid: `${allowStatement.sid} ${index + 1}`, actions: actions, }); policy.addStatements(newStatement); }); } else { policy.addStatements(allowStatement); } } const denyEveryoneElseStatement = new aws_iam_1.PolicyStatement({ sid: exports.SID_DENY_EVERYONE_ELSE, effect: aws_iam_1.Effect.DENY, principals: policyFactory.makeDenyEveryoneElsePrincipals(), actions: ['sqs:*'], 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('SQS', SUPPORTED_CAPABILITIES, accessSpecsByCapability, resourceArns); if (denyUntrustedOrgsStatement) { policy.addStatements(denyUntrustedOrgsStatement); } policy.validateForResourcePolicy(); return policy; } /** * Grant access to a queue via resource policy using k9 IAccessSpec definitions. This function * is the preferred interface for granting access to a queue. * * The grant and make operations are split because SQS policies can only be managed via the * IQueue.addToResourcePolicy method but IQueue does not offer a way to read the policy. * So making the policy is done in a separate function so policy generation can be tested. * * @param props specifying the queue 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 queue = props.queue; const addToResourcePolicyResults = new Array(); for (let statement of k9Statements) { let addToResourcePolicyResult = queue.addToResourcePolicy(aws_iam_1.PolicyStatement.fromJson(statement)); addToResourcePolicyResults.push(addToResourcePolicyResult); } return addToResourcePolicyResults; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sqs.js","sourceRoot":"","sources":["../src/sqs.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 { IQueue } from 'aws-cdk-lib/aws-sqs';\nimport {\n  AccessCapability,\n  canPrincipalsManageResources,\n  getAccessCapabilityFromValue,\n  IAccessSpec,\n  K9PolicyFactory,\n} from './k9policy';\n\n\nexport interface K9SQSResourcePolicyProps {\n  readonly queue: IQueue;\n  readonly k9DesiredAccess: Array<IAccessSpec>;\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_EVERYONE_ELSE = 'DenyEveryoneElse';\n\nfunction partitionArray<T>(arr: T[], maxLength: number): T[][] {\n  const result: T[][] = [];\n  for (let i = 0; i < arr.length; i += maxLength) {\n    result.push(arr.slice(i, i + maxLength));\n  }\n  return result;\n}\n\n/**\n * Generate a SQS resource policy from the provided props that can be attached to a queue.\n *\n * @param props specifying desired access\n * @return a PolicyDocument that can be attached to an SQS queue\n */\nexport function makeResourcePolicy(props: K9SQSResourcePolicyProps): PolicyDocument {\n  const policyFactory = new K9PolicyFactory();\n  const policy = new iam.PolicyDocument();\n\n  const resourceArns = ['*'];\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 SQS resources' +\n            ' so data 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('SQS',\n    SUPPORTED_CAPABILITIES,\n    Array.from(accessSpecsByCapability.values()),\n    resourceArns);\n\n  const max_actions_in_statement = 7;\n  for (let allowStatement of allowStatements) {\n    //SQS resource policy has a limit of 7 actions per statement (Really).\n    //But you can have as many statements as you want up to the queue policy size limit.\n    //So, if an allowStatement has more than 7 actions (like the administer-resource statement does),\n    //then create additional statements and spread the original statement's permissions across them\n    if (allowStatement.actions.length > max_actions_in_statement) {\n      const partitionedActions = partitionArray(allowStatement.actions, max_actions_in_statement);\n      partitionedActions.forEach((actions, index) => {\n        const newStatement = allowStatement.copy({\n          sid: `${allowStatement.sid} ${index + 1}`,\n          actions: actions,\n        });\n        policy.addStatements(newStatement);\n      });\n    } else {\n      policy.addStatements(allowStatement);\n    }\n  }\n\n  const denyEveryoneElseStatement = new PolicyStatement({\n    sid: SID_DENY_EVERYONE_ELSE,\n    effect: Effect.DENY,\n    principals: policyFactory.makeDenyEveryoneElsePrincipals(),\n    actions: ['sqs:*'],\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(\n    denyEveryoneElseStatement,\n  );\n\n  const denyUntrustedOrgsStatement = policyFactory._makeDenyUntrustedOrgsStatement(\n    'SQS', 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 a queue via resource policy using k9 IAccessSpec definitions. This function\n * is the preferred interface for granting access to a queue.\n *\n * The grant and make operations are split because SQS policies can only be managed via the\n * IQueue.addToResourcePolicy method but IQueue does not offer a way to read the policy.\n * So making the policy is done in a separate function so policy generation can be tested.\n *\n * @param props specifying the queue and desired access\n *\n * @return the results for adding each statement\n */\nexport function grantAccessViaResourcePolicy(props: K9SQSResourcePolicyProps):\nAddToResourcePolicyResult[] {\n  const resourcePolicy = makeResourcePolicy(props);\n\n  resourcePolicy.validateForResourcePolicy();\n\n  const policyJson = resourcePolicy.toJSON();\n  const k9Statements = policyJson.Statement;\n  const queue = props.queue;\n  const addToResourcePolicyResults = new Array<AddToResourcePolicyResult>();\n\n  for (let statement of k9Statements) {\n    let addToResourcePolicyResult = queue.addToResourcePolicy(\n      PolicyStatement.fromJson(statement),\n    );\n    addToResourcePolicyResults.push(addToResourcePolicyResult);\n  }\n\n  return addToResourcePolicyResults;\n}\n"]}