UNPKG

cdk-nag

Version:

Check CDK v2 applications for best practices using a combination on available rule packs.

107 lines 13.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); /* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. SPDX-License-Identifier: Apache-2.0 */ const path_1 = require("path"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_iam_1 = require("aws-cdk-lib/aws-iam"); const nag_rules_1 = require("../../nag-rules"); /** * Lambda functions have least privileged access permissions. * @param node the CfnResource to check */ exports.default = Object.defineProperty((node) => { // Only check IAM roles if (!(node instanceof aws_iam_1.CfnRole)) { return nag_rules_1.NagRuleCompliance.NOT_APPLICABLE; } // Check if this is a Lambda role if (!isLambdaRole(node)) { return nag_rules_1.NagRuleCompliance.NOT_APPLICABLE; } // Check if the role has any policies with wildcard permissions const inlinePolicies = aws_cdk_lib_1.Stack.of(node).resolve(node.policies); if (inlinePolicies && inlinePolicies.length > 0) { for (const policy of inlinePolicies) { const resolvedPolicy = aws_cdk_lib_1.Stack.of(node).resolve(policy); const policyDocument = aws_cdk_lib_1.Stack.of(node).resolve(resolvedPolicy.policyDocument); if (policyDocument.Statement) { for (const statement of policyDocument.Statement) { if (statementContainsWildcard(statement)) { return nag_rules_1.NagRuleCompliance.NON_COMPLIANT; } } } } } // If we've checked all policies and found no wildcards, the role is compliant return nag_rules_1.NagRuleCompliance.COMPLIANT; }, 'name', { value: (0, path_1.parse)(__filename).name }); /** * Checks if a role is assumed by the Lambda service * @param node The CfnRole to check * @returns true if the role is assumed by Lambda service */ function isLambdaRole(node) { const assumeRolePolicyDocument = aws_cdk_lib_1.Stack.of(node).resolve(node.assumeRolePolicyDocument); if (!assumeRolePolicyDocument || !assumeRolePolicyDocument.Statement) { return false; } for (const statement of assumeRolePolicyDocument.Statement) { if (statement.Principal && statement.Principal.Service) { const service = Array.isArray(statement.Principal.Service) ? statement.Principal.Service : [statement.Principal.Service]; if (service.includes('lambda.amazonaws.com')) { return true; } } } return false; } /** * Checks if a policy statement contains wildcard permissions * @param statement The policy statement to check * @returns true if the statement contains wildcard permissions */ function statementContainsWildcard(statement) { // Only check Allow statements if (statement.Effect !== 'Allow') { return false; } // Check for wildcard in actions const actions = normalizeToArray(statement.Action); for (const action of actions) { if (typeof action === 'string') { // Check for full wildcard ('*') if (action === '*') { return true; } // Check for service level wildcard (e.g., 's3:*') // But allow service specific partial actions (e.g., 's3:Get*') if (action.endsWith(':*')) { return true; } } } // Check for full wildcard in resources // Only flag resources that are exactly '*' const resources = normalizeToArray(statement.Resource); for (const resource of resources) { if (typeof resource === 'string' && resource === '*') { return true; } } return false; } /** * Normalizes a value to an array * @param value The value to normalize * @returns An array containing the value(s) */ function normalizeToArray(value) { return Array.isArray(value) ? value : [value]; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"LambdaStarPermissions.js","sourceRoot":"","sources":["../../../src/rules/lambda/LambdaStarPermissions.ts"],"names":[],"mappings":"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","sourcesContent":["/*\nCopyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\nSPDX-License-Identifier: Apache-2.0\n*/\nimport { parse } from 'path';\nimport { CfnResource, Stack } from 'aws-cdk-lib';\nimport { CfnRole } from 'aws-cdk-lib/aws-iam';\nimport { NagRuleCompliance } from '../../nag-rules';\n\ninterface IAMPolicyDocument {\n  Statement?: IAMPolicyStatement[];\n}\n\ninterface IAMPolicyStatement {\n  Action: string | string[];\n  Effect: 'Allow' | 'Deny';\n  Resource: unknown;\n}\n\n/**\n * Lambda functions have least privileged access permissions.\n * @param node the CfnResource to check\n */\nexport default Object.defineProperty(\n  (node: CfnResource): NagRuleCompliance => {\n    // Only check IAM roles\n    if (!(node instanceof CfnRole)) {\n      return NagRuleCompliance.NOT_APPLICABLE;\n    }\n\n    // Check if this is a Lambda role\n    if (!isLambdaRole(node)) {\n      return NagRuleCompliance.NOT_APPLICABLE;\n    }\n\n    // Check if the role has any policies with wildcard permissions\n    const inlinePolicies = Stack.of(node).resolve(node.policies);\n\n    if (inlinePolicies && inlinePolicies.length > 0) {\n      for (const policy of inlinePolicies) {\n        const resolvedPolicy = Stack.of(node).resolve(policy);\n        const policyDocument: IAMPolicyDocument = Stack.of(node).resolve(\n          resolvedPolicy.policyDocument\n        );\n\n        if (policyDocument.Statement) {\n          for (const statement of policyDocument.Statement) {\n            if (statementContainsWildcard(statement)) {\n              return NagRuleCompliance.NON_COMPLIANT;\n            }\n          }\n        }\n      }\n    }\n\n    // If we've checked all policies and found no wildcards, the role is compliant\n    return NagRuleCompliance.COMPLIANT;\n  },\n  'name',\n  { value: parse(__filename).name }\n);\n\n/**\n * Checks if a role is assumed by the Lambda service\n * @param node The CfnRole to check\n * @returns true if the role is assumed by Lambda service\n */\nfunction isLambdaRole(node: CfnRole): boolean {\n  const assumeRolePolicyDocument = Stack.of(node).resolve(\n    node.assumeRolePolicyDocument\n  );\n\n  if (!assumeRolePolicyDocument || !assumeRolePolicyDocument.Statement) {\n    return false;\n  }\n\n  for (const statement of assumeRolePolicyDocument.Statement) {\n    if (statement.Principal && statement.Principal.Service) {\n      const service = Array.isArray(statement.Principal.Service)\n        ? statement.Principal.Service\n        : [statement.Principal.Service];\n\n      if (service.includes('lambda.amazonaws.com')) {\n        return true;\n      }\n    }\n  }\n\n  return false;\n}\n\n/**\n * Checks if a policy statement contains wildcard permissions\n * @param statement The policy statement to check\n * @returns true if the statement contains wildcard permissions\n */\nfunction statementContainsWildcard(statement: IAMPolicyStatement): boolean {\n  // Only check Allow statements\n  if (statement.Effect !== 'Allow') {\n    return false;\n  }\n\n  // Check for wildcard in actions\n  const actions = normalizeToArray(statement.Action);\n  for (const action of actions) {\n    if (typeof action === 'string') {\n      // Check for full wildcard ('*')\n      if (action === '*') {\n        return true;\n      }\n\n      // Check for service level wildcard (e.g., 's3:*')\n      // But allow service specific partial actions (e.g., 's3:Get*')\n      if (action.endsWith(':*')) {\n        return true;\n      }\n    }\n  }\n\n  // Check for full wildcard in resources\n  // Only flag resources that are exactly '*'\n  const resources = normalizeToArray(statement.Resource);\n  for (const resource of resources) {\n    if (typeof resource === 'string' && resource === '*') {\n      return true;\n    }\n  }\n\n  return false;\n}\n\n/**\n * Normalizes a value to an array\n * @param value The value to normalize\n * @returns An array containing the value(s)\n */\nfunction normalizeToArray<T>(value: T | T[]): T[] {\n  return Array.isArray(value) ? value : [value];\n}\n"]}