UNPKG

@aws/pdk

Version:

All documentation is located at: https://aws.github.io/aws-pdk

239 lines 36.9 kB
"use strict"; /*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved. SPDX-License-Identifier: Apache-2.0 */ Object.defineProperty(exports, "__esModule", { value: true }); exports.FeatureBranches = void 0; const path = require("path"); const pdk_nag_1 = require("../pdk-nag"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_codebuild_1 = require("aws-cdk-lib/aws-codebuild"); const aws_events_targets_1 = require("aws-cdk-lib/aws-events-targets"); const aws_iam_1 = require("aws-cdk-lib/aws-iam"); const aws_lambda_1 = require("aws-cdk-lib/aws-lambda"); const cdk_nag_1 = require("cdk-nag"); const constructs_1 = require("constructs"); class FeatureBranches extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); const buildCommands = props.synthShellStepPartialProps?.commands && props.synthShellStepPartialProps.commands.length > 0 ? props.synthShellStepPartialProps.commands : ["npx nx run-many --target=build --all"]; const installCommands = props.synthShellStepPartialProps?.installCommands && props.synthShellStepPartialProps.installCommands.length > 0 ? props.synthShellStepPartialProps.installCommands : [ "npm install -g aws-cdk", "yarn install --frozen-lockfile || npx projen && yarn install --frozen-lockfile", ]; const cdkCommand = props.cdkCommand ?? "npx cdk"; const createFeatureBranchProject = new aws_codebuild_1.Project(this, "CreateFeatureBranchProject", { ...props.codeBuildDefaults, description: "Build project to deploy feature branch pipelines", source: aws_codebuild_1.Source.codeCommit({ repository: props.codeRepository }), environment: { buildImage: aws_codebuild_1.LinuxBuildImage.STANDARD_7_0, computeType: aws_codebuild_1.ComputeType.SMALL, ...props.codeBuildDefaults?.buildEnvironment, privileged: props.dockerEnabledForSynth, }, buildSpec: aws_codebuild_1.BuildSpec.fromObjectToYaml({ version: "0.2", phases: { install: { commands: installCommands, }, build: { commands: [ ...buildCommands, `cd ${props.cdkSrcDir}`, `${cdkCommand} synth`, `${cdkCommand} deploy --require-approval=never`, ], }, }, artifacts: { files: ["**/*"], }, }), }); if (props.codeBuildDefaults?.rolePolicy) { props.codeBuildDefaults.rolePolicy.forEach((policy) => { createFeatureBranchProject.addToRolePolicy(policy); }); } createFeatureBranchProject.addToRolePolicy(new aws_iam_1.PolicyStatement({ effect: aws_iam_1.Effect.ALLOW, actions: ["sts:AssumeRole"], resources: [`arn:*:iam::${aws_cdk_lib_1.Stack.of(this).account}:role/*`], conditions: { "ForAnyValue:StringEquals": { "iam:ResourceTag/aws-cdk:bootstrap-role": [ "image-publishing", "file-publishing", "deploy", ], }, }, })); const createFeatureBranchFunction = new aws_lambda_1.Function(this, "LambdaTriggerCreateBranch", { runtime: aws_lambda_1.Runtime.PYTHON_3_12, code: aws_lambda_1.Code.fromAsset(path.join(__dirname, "lambda/create_branch")), handler: "create_branch.handler", environment: { CODEBUILD_PROJECT: createFeatureBranchProject.projectName, MAIN_BRANCH: props.defaultBranchName, }, }); createFeatureBranchFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({ effect: aws_iam_1.Effect.ALLOW, actions: ["codebuild:StartBuild"], resources: [createFeatureBranchProject.projectArn], })); const destroyFeatureBranchFunction = new aws_lambda_1.Function(this, "LambdaTriggerDestroyBranch", { runtime: aws_lambda_1.Runtime.PYTHON_3_12, code: aws_lambda_1.Code.fromAsset(path.join(__dirname, "lambda/destroy_branch")), handler: "destroy_branch.handler", environment: { MAIN_BRANCH: props.defaultBranchName, REPO_NAME: props.codeRepository.repositoryName, }, }); destroyFeatureBranchFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({ effect: aws_iam_1.Effect.ALLOW, actions: ["cloudformation:DeleteStack"], resources: [ aws_cdk_lib_1.Stack.of(this).formatArn({ service: "cloudformation", resource: "stack", resourceName: "*/*", }), ], conditions: { "ForAllValues:StringEquals": { "aws:TagKeys": ["FeatureBranch", "RepoName"], }, }, })); destroyFeatureBranchFunction.addToRolePolicy(new aws_iam_1.PolicyStatement({ effect: aws_iam_1.Effect.ALLOW, actions: ["tag:GetResources"], resources: ["*"], })); props.codeRepository.onReferenceCreated("BranchCreateTrigger", { target: new aws_events_targets_1.LambdaFunction(createFeatureBranchFunction), description: "AWS CodeCommit reference created event.", eventPattern: { detail: { referenceType: ["branch"], referenceName: props.branchNamePrefixes.map((prefix) => ({ prefix })), }, }, }); props.codeRepository.onReferenceDeleted("BranchDestroyTrigger", { target: new aws_events_targets_1.LambdaFunction(destroyFeatureBranchFunction), description: "AWS CodeCommit reference deleted event.", eventPattern: { detail: { referenceType: ["branch"], referenceName: props.branchNamePrefixes.map((prefix) => { return { prefix }; }), }, }, }); const stack = aws_cdk_lib_1.Stack.of(this); ["AwsSolutions-IAM5", "AwsPrototyping-IAMNoWildcardPermissions"].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(createFeatureBranchProject.role, [ { id: RuleId, reason: "CodeBuild requires get, list, and pull access to the CodeCommit repository.", appliesTo: [ "Action::codecommit:Get*", "Action::codecommit:List*", ], }, { id: RuleId, reason: "CodeBuild requires access to create report groups that are dynamically determined.", appliesTo: [ { regex: `/^Resource::arn:${pdk_nag_1.PDKNag.getStackPartitionRegex(stack)}:codebuild:${pdk_nag_1.PDKNag.getStackRegionRegex(stack)}:${pdk_nag_1.PDKNag.getStackAccountRegex(stack)}:report-group/<[a-zA-Z0-9]*CreateFeatureBranchProject.*>-\\*$/g`, }, ], }, { id: RuleId, reason: "CodeBuild requires access to manage logs and streams whose names are dynamically determined.", appliesTo: [ { regex: `/^Resource::arn:${pdk_nag_1.PDKNag.getStackPartitionRegex(stack)}:logs:${pdk_nag_1.PDKNag.getStackRegionRegex(stack)}:${pdk_nag_1.PDKNag.getStackAccountRegex(stack)}:log-group:/aws/codebuild/<[a-zA-Z0-9]*CreateFeatureBranchProject.*>:\\*$/g`, }, ], }, { id: RuleId, reason: "CodeBuild requires access to assume a role from within the current account limited by a condition in order to deploy.", appliesTo: [ { regex: `/^Resource::arn:\\*:iam::${pdk_nag_1.PDKNag.getStackAccountRegex(stack)}:role/\\*$/g`, }, ], }, ], true); cdk_nag_1.NagSuppressions.addResourceSuppressions(destroyFeatureBranchFunction.role, [ { id: RuleId, reason: "The DestroyBranch Lambda requires access to delete any stacks with specific tags.", appliesTo: [ { regex: `/^Resource::arn:${pdk_nag_1.PDKNag.getStackPartitionRegex(stack)}:cloudformation:${pdk_nag_1.PDKNag.getStackRegionRegex(stack)}:${pdk_nag_1.PDKNag.getStackAccountRegex(stack)}:stack/\\*/\\*$/g`, }, ], }, { id: RuleId, reason: "The DestroyBranch Lambda requires access to look up CloudFormation stacks by tag. The Resource Group Tagging API must use 'Resource': '*'.", appliesTo: ["Resource::*"], }, ], true); }); ["AwsSolutions-IAM4", "AwsPrototyping-IAMNoManagedPolicies"].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(createFeatureBranchFunction, [ { id: RuleId, reason: "Lambda functions use the default AWS LambdaBasicExecutionRole managed role.", appliesTo: [ { regex: `/^Policy::arn:${pdk_nag_1.PDKNag.getStackPartitionRegex(stack)}:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole$/g`, }, ], }, ], true); cdk_nag_1.NagSuppressions.addResourceSuppressions(destroyFeatureBranchFunction, [ { id: RuleId, reason: "Lambda functions use the default AWS LambdaBasicExecutionRole managed role.", appliesTo: [ { regex: `/^Policy::arn:${pdk_nag_1.PDKNag.getStackPartitionRegex(stack)}:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole$/g`, }, ], }, ], true); }); [ "AwsSolutions-CB4", "AwsPrototyping-CodeBuildProjectKMSEncryptedArtifacts", ].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(createFeatureBranchProject, [ { id: RuleId, reason: "Encryption of Codebuild is not required.", }, ]); }); } } exports.FeatureBranches = FeatureBranches; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"feature-branches.js","sourceRoot":"","sources":["feature-branches.ts"],"names":[],"mappings":"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","sourcesContent":["/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.\nSPDX-License-Identifier: Apache-2.0 */\n\nimport * as path from \"path\";\nimport { PDKNag } from \"@aws/pdk-nag\";\nimport { Stack } from \"aws-cdk-lib\";\nimport {\n  BuildSpec,\n  ComputeType,\n  LinuxBuildImage,\n  Project,\n  Source,\n} from \"aws-cdk-lib/aws-codebuild\";\nimport { IRepository } from \"aws-cdk-lib/aws-codecommit\";\nimport { LambdaFunction } from \"aws-cdk-lib/aws-events-targets\";\nimport { Effect, PolicyStatement } from \"aws-cdk-lib/aws-iam\";\nimport { Code, Function, Runtime } from \"aws-cdk-lib/aws-lambda\";\nimport { CodePipelineProps, ShellStepProps } from \"aws-cdk-lib/pipelines\";\nimport { NagSuppressions } from \"cdk-nag\";\nimport { Construct } from \"constructs\";\n\nexport interface FeatureBranchesProps\n  extends Pick<\n    CodePipelineProps,\n    \"codeBuildDefaults\" | \"dockerEnabledForSynth\"\n  > {\n  /**\n   * Branch name prefixes\n   * Any branches created matching this list of prefixes will create a new pipeline and stack.\n   *\n   * @example\n   * // Creates a new pipeline and stack for any branch\n   * new PDKPipeline(this, 'PDKPipeline', {\n   *   repositoryName: 'my-repo',\n   *   branchNamePrefixes: [''],\n   * }\n   * @example\n   * // Creates a new pipeline and stack for any branch starting with 'feature/' or 'fix/'\n   * new PDKPipeline(this, 'PDKPipeline', {\n   *   repositoryName: 'my-repo',\n   *   branchNamePrefixes: ['feature/', 'fix/'],\n   * }\n   * @example\n   * // Disables feature branches (default)\n   * new PDKPipeline(this, 'PDKPipeline', {\n   *   repositoryName: 'my-repo',\n   *   branchNamePrefixes: [], // or simply exclude this line\n   * }\n   */\n  readonly branchNamePrefixes: string[];\n\n  /**\n   * The directory with `cdk.json` to run cdk synth from.\n   */\n  readonly cdkSrcDir: string;\n\n  /**\n   * The CodeCommit repository.\n   */\n  readonly codeRepository: IRepository;\n\n  /**\n   * Default branch.\n   */\n  readonly defaultBranchName: string;\n\n  /**\n   * PDKPipeline by default assumes a NX Monorepo structure for it's codebase and\n   * uses sane defaults for the install and run commands. To override these defaults\n   * and/or provide additional inputs, specify env settings, etc you can provide\n   * a partial ShellStepProps.\n   */\n  readonly synthShellStepPartialProps?: ShellStepProps;\n\n  /**\n   * CDK command. Override the command used to call cdk for synth and deploy.\n   *\n   * @default 'npx cdk'\n   */\n  readonly cdkCommand?: string;\n}\n\nexport class FeatureBranches extends Construct {\n  constructor(scope: Construct, id: string, props: FeatureBranchesProps) {\n    super(scope, id);\n\n    const buildCommands: string[] =\n      props.synthShellStepPartialProps?.commands &&\n      props.synthShellStepPartialProps.commands.length > 0\n        ? props.synthShellStepPartialProps.commands\n        : [\"npx nx run-many --target=build --all\"];\n\n    const installCommands: string[] =\n      props.synthShellStepPartialProps?.installCommands &&\n      props.synthShellStepPartialProps.installCommands.length > 0\n        ? props.synthShellStepPartialProps.installCommands\n        : [\n            \"npm install -g aws-cdk\",\n            \"yarn install --frozen-lockfile || npx projen && yarn install --frozen-lockfile\",\n          ];\n\n    const cdkCommand = props.cdkCommand ?? \"npx cdk\";\n\n    const createFeatureBranchProject = new Project(\n      this,\n      \"CreateFeatureBranchProject\",\n      {\n        ...props.codeBuildDefaults,\n        description: \"Build project to deploy feature branch pipelines\",\n        source: Source.codeCommit({ repository: props.codeRepository }),\n        environment: {\n          buildImage: LinuxBuildImage.STANDARD_7_0,\n          computeType: ComputeType.SMALL,\n          ...props.codeBuildDefaults?.buildEnvironment,\n          privileged: props.dockerEnabledForSynth,\n        },\n        buildSpec: BuildSpec.fromObjectToYaml({\n          version: \"0.2\",\n          phases: {\n            install: {\n              commands: installCommands,\n            },\n            build: {\n              commands: [\n                ...buildCommands,\n                `cd ${props.cdkSrcDir}`,\n                `${cdkCommand} synth`,\n                `${cdkCommand} deploy --require-approval=never`,\n              ],\n            },\n          },\n          artifacts: {\n            files: [\"**/*\"],\n          },\n        }),\n      }\n    );\n\n    if (props.codeBuildDefaults?.rolePolicy) {\n      props.codeBuildDefaults.rolePolicy.forEach((policy) => {\n        createFeatureBranchProject.addToRolePolicy(policy);\n      });\n    }\n\n    createFeatureBranchProject.addToRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: [\"sts:AssumeRole\"],\n        resources: [`arn:*:iam::${Stack.of(this).account}:role/*`],\n        conditions: {\n          \"ForAnyValue:StringEquals\": {\n            \"iam:ResourceTag/aws-cdk:bootstrap-role\": [\n              \"image-publishing\",\n              \"file-publishing\",\n              \"deploy\",\n            ],\n          },\n        },\n      })\n    );\n\n    const createFeatureBranchFunction = new Function(\n      this,\n      \"LambdaTriggerCreateBranch\",\n      {\n        runtime: Runtime.PYTHON_3_12,\n        code: Code.fromAsset(path.join(__dirname, \"lambda/create_branch\")),\n        handler: \"create_branch.handler\",\n        environment: {\n          CODEBUILD_PROJECT: createFeatureBranchProject.projectName,\n          MAIN_BRANCH: props.defaultBranchName,\n        },\n      }\n    );\n\n    createFeatureBranchFunction.addToRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: [\"codebuild:StartBuild\"],\n        resources: [createFeatureBranchProject.projectArn],\n      })\n    );\n\n    const destroyFeatureBranchFunction = new Function(\n      this,\n      \"LambdaTriggerDestroyBranch\",\n      {\n        runtime: Runtime.PYTHON_3_12,\n        code: Code.fromAsset(path.join(__dirname, \"lambda/destroy_branch\")),\n        handler: \"destroy_branch.handler\",\n        environment: {\n          MAIN_BRANCH: props.defaultBranchName,\n          REPO_NAME: props.codeRepository.repositoryName,\n        },\n      }\n    );\n    destroyFeatureBranchFunction.addToRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: [\"cloudformation:DeleteStack\"],\n        resources: [\n          Stack.of(this).formatArn({\n            service: \"cloudformation\",\n            resource: \"stack\",\n            resourceName: \"*/*\",\n          }),\n        ],\n        conditions: {\n          \"ForAllValues:StringEquals\": {\n            \"aws:TagKeys\": [\"FeatureBranch\", \"RepoName\"],\n          },\n        },\n      })\n    );\n    destroyFeatureBranchFunction.addToRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: [\"tag:GetResources\"],\n        resources: [\"*\"],\n      })\n    );\n\n    props.codeRepository.onReferenceCreated(\"BranchCreateTrigger\", {\n      target: new LambdaFunction(createFeatureBranchFunction),\n      description: \"AWS CodeCommit reference created event.\",\n      eventPattern: {\n        detail: {\n          referenceType: [\"branch\"],\n          referenceName: props.branchNamePrefixes.map((prefix) => ({ prefix })),\n        },\n      },\n    });\n\n    props.codeRepository.onReferenceDeleted(\"BranchDestroyTrigger\", {\n      target: new LambdaFunction(destroyFeatureBranchFunction),\n      description: \"AWS CodeCommit reference deleted event.\",\n      eventPattern: {\n        detail: {\n          referenceType: [\"branch\"],\n          referenceName: props.branchNamePrefixes.map((prefix) => {\n            return { prefix };\n          }),\n        },\n      },\n    });\n\n    const stack = Stack.of(this);\n\n    [\"AwsSolutions-IAM5\", \"AwsPrototyping-IAMNoWildcardPermissions\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          createFeatureBranchProject.role!,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"CodeBuild requires get, list, and pull access to the CodeCommit repository.\",\n              appliesTo: [\n                \"Action::codecommit:Get*\",\n                \"Action::codecommit:List*\",\n              ],\n            },\n            {\n              id: RuleId,\n              reason:\n                \"CodeBuild requires access to create report groups that are dynamically determined.\",\n              appliesTo: [\n                {\n                  regex: `/^Resource::arn:${PDKNag.getStackPartitionRegex(\n                    stack\n                  )}:codebuild:${PDKNag.getStackRegionRegex(\n                    stack\n                  )}:${PDKNag.getStackAccountRegex(\n                    stack\n                  )}:report-group/<[a-zA-Z0-9]*CreateFeatureBranchProject.*>-\\\\*$/g`,\n                },\n              ],\n            },\n            {\n              id: RuleId,\n              reason:\n                \"CodeBuild requires access to manage logs and streams whose names are dynamically determined.\",\n              appliesTo: [\n                {\n                  regex: `/^Resource::arn:${PDKNag.getStackPartitionRegex(\n                    stack\n                  )}:logs:${PDKNag.getStackRegionRegex(\n                    stack\n                  )}:${PDKNag.getStackAccountRegex(\n                    stack\n                  )}:log-group:/aws/codebuild/<[a-zA-Z0-9]*CreateFeatureBranchProject.*>:\\\\*$/g`,\n                },\n              ],\n            },\n            {\n              id: RuleId,\n              reason:\n                \"CodeBuild requires access to assume a role from within the current account limited by a condition in order to deploy.\",\n              appliesTo: [\n                {\n                  regex: `/^Resource::arn:\\\\*:iam::${PDKNag.getStackAccountRegex(\n                    stack\n                  )}:role/\\\\*$/g`,\n                },\n              ],\n            },\n          ],\n          true\n        );\n\n        NagSuppressions.addResourceSuppressions(\n          destroyFeatureBranchFunction.role!,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"The DestroyBranch Lambda requires access to delete any stacks with specific tags.\",\n              appliesTo: [\n                {\n                  regex: `/^Resource::arn:${PDKNag.getStackPartitionRegex(\n                    stack\n                  )}:cloudformation:${PDKNag.getStackRegionRegex(\n                    stack\n                  )}:${PDKNag.getStackAccountRegex(stack)}:stack/\\\\*/\\\\*$/g`,\n                },\n              ],\n            },\n            {\n              id: RuleId,\n              reason:\n                \"The DestroyBranch Lambda requires access to look up CloudFormation stacks by tag. The Resource Group Tagging API must use 'Resource': '*'.\",\n              appliesTo: [\"Resource::*\"],\n            },\n          ],\n          true\n        );\n      }\n    );\n\n    [\"AwsSolutions-IAM4\", \"AwsPrototyping-IAMNoManagedPolicies\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          createFeatureBranchFunction,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"Lambda functions use the default AWS LambdaBasicExecutionRole managed role.\",\n              appliesTo: [\n                {\n                  regex: `/^Policy::arn:${PDKNag.getStackPartitionRegex(\n                    stack\n                  )}:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole$/g`,\n                },\n              ],\n            },\n          ],\n          true\n        );\n        NagSuppressions.addResourceSuppressions(\n          destroyFeatureBranchFunction,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"Lambda functions use the default AWS LambdaBasicExecutionRole managed role.\",\n              appliesTo: [\n                {\n                  regex: `/^Policy::arn:${PDKNag.getStackPartitionRegex(\n                    stack\n                  )}:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole$/g`,\n                },\n              ],\n            },\n          ],\n          true\n        );\n      }\n    );\n\n    [\n      \"AwsSolutions-CB4\",\n      \"AwsPrototyping-CodeBuildProjectKMSEncryptedArtifacts\",\n    ].forEach((RuleId) => {\n      NagSuppressions.addResourceSuppressions(createFeatureBranchProject, [\n        {\n          id: RuleId,\n          reason: \"Encryption of Codebuild is not required.\",\n        },\n      ]);\n    });\n  }\n}\n"]}