UNPKG

@aws/pdk

Version:

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

299 lines 49 kB
"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __exportStar = (this && this.__exportStar) || function(m, exports) { for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p); }; var _a, _b; Object.defineProperty(exports, "__esModule", { value: true }); exports.PDKPipelineWithCodeConnection = exports.PDKPipeline = exports.DEFAULT_BRANCH_NAME = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); /*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved. SPDX-License-Identifier: Apache-2.0 */ const path = require("path"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_codecommit_1 = require("aws-cdk-lib/aws-codecommit"); const aws_codepipeline_1 = require("aws-cdk-lib/aws-codepipeline"); const aws_kms_1 = require("aws-cdk-lib/aws-kms"); const aws_s3_1 = require("aws-cdk-lib/aws-s3"); const pipelines_1 = require("aws-cdk-lib/pipelines"); const cdk_nag_1 = require("cdk-nag"); const constructs_1 = require("constructs"); const sonar_code_scanner_1 = require("./code_scanner/sonar-code-scanner"); const feature_branches_1 = require("./feature-branches"); __exportStar(require("./code_scanner/sonar-code-scanner"), exports); exports.DEFAULT_BRANCH_NAME = "mainline"; /** * An extension to CodePipeline which configures sane defaults for a NX Monorepo * codebase. In addition to this, it also creates a CodeCommit repository with * automated PR builds and approvals. */ class BasePDKPipeline extends constructs_1.Construct { /** * A helper function to normalize the branch name with only alphanumeric characters and hypens ('-'). * @param branchName The name of the branch to normalize. * @returns The normalized branch name. */ static normalizeBranchName(branchName) { return branchName.replace(/[^a-zA-Z0-9-]/g, "-"); } /** * A helper function to determine if the current branch is the default branch. * * If there is no BRANCH environment variable, then assume this is the default * branch. Otherwise, check that BRANCH matches the default branch name. * * The default branch name is determined in the following priority: * * 1. defaultBranchName property * 2. defaultBranchName context * 3. PDKPipeline.defaultBranchName constant * * @param props? { * defaultBranchName? Specify the default branch name without context. * node? The current app to fetch defaultBranchName from context. * } * @returns True if the current branch is the default branch. */ static isDefaultBranch(props = { defaultBranchName: undefined, node: undefined, }) { if (!process.env.BRANCH) { return true; } const defaultBranchName = props.defaultBranchName || (props.node && props.node.tryGetContext("defaultBranchName")) || BasePDKPipeline.defaultBranchName; return defaultBranchName === process.env.BRANCH; } /** * A helper function to create a branch prefix. The prefix is empty on the default branch. * @param props? { * defaultBranchName? Specify the default branch name without context. * node? The current app to fetch defaultBranchName from context. * } * @returns The branch prefix. */ static getBranchPrefix(props = { defaultBranchName: undefined, node: undefined, }) { return BasePDKPipeline.isDefaultBranch(props) ? "" : BasePDKPipeline.normalizeBranchName(process.env.BRANCH) + "-"; } constructor(scope, id, props) { super(scope, id); this.node.setContext("@aws-cdk/aws-s3:serverAccessLogsUseBucketPolicy", true); let source; const branch = process.env.BRANCH || props.defaultBranchName || exports.DEFAULT_BRANCH_NAME; if (props.useCodeCommit) { let codeRepository; const repositoryName = props.repositoryName || ""; if (BasePDKPipeline.isDefaultBranch({ node: this.node, defaultBranchName: props.defaultBranchName, })) { // In the default branch, create a CodeCommit repository codeRepository = new aws_codecommit_1.Repository(this, "CodeRepository", { repositoryName, }); codeRepository.applyRemovalPolicy(props.codeCommitRemovalPolicy ?? aws_cdk_lib_1.RemovalPolicy.RETAIN); } else { // In a non-default branch, use an existing CodeCommit repository codeRepository = aws_codecommit_1.Repository.fromRepositoryName(scope, "CodeRepository", repositoryName); } source = pipelines_1.CodePipelineSource.codeCommit(codeRepository, branch); this.codeRepository = codeRepository; } else { const repositoryOwnerAndName = props.repositoryOwnerAndName || ""; const codeConnectionArn = props.codeConnectionArn || ""; source = pipelines_1.CodePipelineSource.connection(repositoryOwnerAndName, props.defaultBranchName || exports.DEFAULT_BRANCH_NAME, { connectionArn: codeConnectionArn, }); } const accessLogsBucket = new aws_s3_1.Bucket(this, "AccessLogsBucket", { versioned: false, enforceSSL: true, autoDeleteObjects: true, removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, encryption: aws_s3_1.BucketEncryption.S3_MANAGED, objectOwnership: aws_s3_1.ObjectOwnership.OBJECT_WRITER, publicReadAccess: false, blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL, }); const artifactBucket = new aws_s3_1.Bucket(this, "ArtifactsBucket", { enforceSSL: true, autoDeleteObjects: true, removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, encryption: props.crossAccountKeys ? aws_s3_1.BucketEncryption.KMS : aws_s3_1.BucketEncryption.S3_MANAGED, encryptionKey: props.crossAccountKeys ? new aws_kms_1.Key(this, "ArtifactKey", { enableKeyRotation: true, removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, }) : undefined, objectOwnership: aws_s3_1.ObjectOwnership.BUCKET_OWNER_ENFORCED, publicReadAccess: false, blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL, serverAccessLogsPrefix: "access-logs", serverAccessLogsBucket: accessLogsBucket, }); const codePipeline = new aws_codepipeline_1.Pipeline(this, "CodePipeline", { enableKeyRotation: props.crossAccountKeys, restartExecutionOnUpdate: true, crossAccountKeys: props.crossAccountKeys, artifactBucket, pipelineType: aws_codepipeline_1.PipelineType.V1, }); const { input, primaryOutputDirectory, commands, ...synthShellStepPartialProps } = props.synthShellStepPartialProps || {}; const synthShellStep = new pipelines_1.ShellStep("Synth", { input: source, env: props.branchNamePrefixes && props.branchNamePrefixes.length > 0 ? { BRANCH: branch, } : undefined, installCommands: ["npm install -g aws-cdk pnpm", "npx projen install"], commands: commands && commands.length > 0 ? commands : ["npx projen build"], primaryOutputDirectory: props.primarySynthDirectory, ...(synthShellStepPartialProps || {}), }); synthShellStep.addOutputDirectory("."); const codePipelineProps = { codePipeline, ...props, crossAccountKeys: undefined, synth: synthShellStep, }; this.codePipeline = new pipelines_1.CodePipeline(this, id, codePipelineProps); this.sonarCodeScannerConfig = props.sonarCodeScannerConfig ? { cdkOutDir: props.primarySynthDirectory, ...props.sonarCodeScannerConfig, } : undefined; this.branchNamePrefixes = props.branchNamePrefixes; this.defaultBranchName = props.defaultBranchName; this.repositoryName = (props.useCodeCommit ? props.repositoryName : props.repositoryOwnerAndName) || ""; if (this.codeRepository && props.branchNamePrefixes) { if (PDKPipeline.isDefaultBranch({ node: this.node, defaultBranchName: props.defaultBranchName, })) { new feature_branches_1.FeatureBranches(this, "FeatureBranchPipelines", { codeRepository: this.codeRepository, cdkSrcDir: props.cdkSrcDir || path.dirname(props.primarySynthDirectory), synthShellStepPartialProps: props.synthShellStepPartialProps, cdkCommand: props.cdkCommand, branchNamePrefixes: props.branchNamePrefixes, defaultBranchName: props.defaultBranchName || exports.DEFAULT_BRANCH_NAME, codeBuildDefaults: props.codeBuildDefaults, dockerEnabledForSynth: props.dockerEnabledForSynth, }); } else { aws_cdk_lib_1.Tags.of(aws_cdk_lib_1.Stack.of(this)).add("FeatureBranch", branch); aws_cdk_lib_1.Tags.of(aws_cdk_lib_1.Stack.of(this)).add("RepoName", this.repositoryName); } } if (props.useCodeCommit && this.codeRepository) { new aws_cdk_lib_1.CfnOutput(this, "CodeRepositoryGRCUrl", { value: this.codeRepository.repositoryCloneUrlGrc, }); } } /** * @inheritDoc */ addStage(stage, options) { if (this.branchNamePrefixes && !PDKPipeline.isDefaultBranch({ node: stage.node, defaultBranchName: this.defaultBranchName, })) { aws_cdk_lib_1.Tags.of(stage).add("FeatureBranch", process.env.BRANCH); aws_cdk_lib_1.Tags.of(stage).add("RepoName", this.repositoryName); } // Add any root Aspects to the stage level as currently this doesn't happen automatically aws_cdk_lib_1.Aspects.of(stage.node.root).all.forEach((aspect) => aws_cdk_lib_1.Aspects.of(stage).add(aspect)); return this.codePipeline.addStage(stage, options); } buildPipeline() { this.codePipeline.buildPipeline(); this.sonarCodeScannerConfig && new sonar_code_scanner_1.SonarCodeScanner(this, "SonarCodeScanner", { artifactBucketArn: this.codePipeline.pipeline.artifactBucket.bucketArn, artifactBucketKeyArn: this.codePipeline.pipeline.artifactBucket.encryptionKey?.keyArn, synthBuildArn: this.codePipeline.synthProject.projectArn, ...this.sonarCodeScannerConfig, }); this.suppressCDKViolations(); } suppressCDKViolations() { this.suppressRules(["AwsSolutions-IAM5", "AwsPrototyping-IAMNoWildcardPermissions"], "Wildcards are needed for dynamically created resources."); this.suppressRules([ "AwsSolutions-CB4", "AwsPrototyping-CodeBuildProjectKMSEncryptedArtifacts", ], "Encryption of Codebuild is not required."); this.suppressRules(["AwsSolutions-S1", "AwsPrototyping-S3BucketLoggingEnabled"], "Access Log buckets should not have s3 bucket logging"); } suppressRules(rules, reason) { cdk_nag_1.NagSuppressions.addResourceSuppressions(this, rules.map((r) => ({ id: r, reason, })), true); } } BasePDKPipeline.ALL_BRANCHES = [""]; BasePDKPipeline.defaultBranchName = exports.DEFAULT_BRANCH_NAME; /** * An extension to CodePipeline which configures same defaults for a NX Monorepo * codebase. In addition to this, it also creates a CodeCommit repository with * automated PR builds and approvals. */ class PDKPipeline extends BasePDKPipeline { constructor(scope, id, props) { super(scope, id, { ...props, useCodeCommit: true, codeConnectionArn: undefined, repositoryOwnerAndName: undefined, }); } } exports.PDKPipeline = PDKPipeline; _a = JSII_RTTI_SYMBOL_1; PDKPipeline[_a] = { fqn: "@aws/pdk.pipeline.PDKPipeline", version: "0.26.14" }; /** * An extension to CodePipeline which configures same defaults for a NX Monorepo and using a AWS CodeConnections as source. */ class PDKPipelineWithCodeConnection extends BasePDKPipeline { constructor(scope, id, props) { super(scope, id, { ...props, useCodeCommit: false, }); } } exports.PDKPipelineWithCodeConnection = PDKPipelineWithCodeConnection; _b = JSII_RTTI_SYMBOL_1; PDKPipelineWithCodeConnection[_b] = { fqn: "@aws/pdk.pipeline.PDKPipelineWithCodeConnection", version: "0.26.14" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pdk-pipeline.js","sourceRoot":"","sources":["pdk-pipeline.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 */\nimport * as path from \"path\";\nimport {\n  Aspects,\n  CfnOutput,\n  RemovalPolicy,\n  Stack,\n  Stage,\n  Tags,\n} from \"aws-cdk-lib\";\nimport { IRepository, Repository } from \"aws-cdk-lib/aws-codecommit\";\nimport { Pipeline, PipelineType } from \"aws-cdk-lib/aws-codepipeline\";\nimport { Key } from \"aws-cdk-lib/aws-kms\";\nimport {\n  BlockPublicAccess,\n  Bucket,\n  BucketEncryption,\n  ObjectOwnership,\n} from \"aws-cdk-lib/aws-s3\";\nimport {\n  AddStageOpts,\n  CodePipeline,\n  CodePipelineProps,\n  CodePipelineSource,\n  ShellStep,\n  ShellStepProps,\n  StageDeployment,\n} from \"aws-cdk-lib/pipelines\";\nimport { NagSuppressions } from \"cdk-nag\";\nimport { Construct, Node } from \"constructs\";\nimport {\n  SonarCodeScanner,\n  SonarCodeScannerConfig,\n} from \"./code_scanner/sonar-code-scanner\";\nimport { CodePipelineProps as _CodePipelineProps } from \"./codepipeline-props\";\nimport { FeatureBranches } from \"./feature-branches\";\n\nexport * from \"./code_scanner/sonar-code-scanner\";\n\nexport const DEFAULT_BRANCH_NAME = \"mainline\";\n\ninterface SharePDKPipelineProps extends _CodePipelineProps {\n  /**\n   * Output directory for cdk synthesized artifacts i.e: packages/infra/cdk.out.\n   */\n  readonly primarySynthDirectory: 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   * Branch to trigger the pipeline execution.\n   *\n   * @default mainline\n   */\n  readonly defaultBranchName?: string;\n\n  /**\n   * Configuration for enabling Sonarqube code scanning on a successful synth.\n   *\n   * @default undefined\n   */\n  readonly sonarCodeScannerConfig?: SonarCodeScannerConfig;\n\n  /**\n   * The directory with `cdk.json` to run cdk synth from. Set this if you enabled\n   * feature branches and `cdk.json` is not located in the parent directory of\n   * `primarySynthDirectory`.\n   *\n   * @default The parent directory of `primarySynthDirectory`\n   */\n  readonly cdkSrcDir?: string;\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\ninterface BasePDKPipelineProps extends SharePDKPipelineProps {\n  /**\n   * Whether to use codeCommit as source or not.\n   */\n  readonly useCodeCommit: boolean;\n  /**\n   * The repository to add the pipeline to.\n   */\n  readonly codeCommitRepository?: IRepository;\n  /**\n   * Name of the CodeCommit repository to create.\n   */\n  readonly repositoryName?: string;\n  /**\n   * Possible values for a resource's Removal Policy\n   * The removal policy controls what happens to the resource if it stops being managed by CloudFormation.\n   */\n  readonly codeCommitRemovalPolicy?: RemovalPolicy;\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: PDKPipeline.ALL_BRANCHES,\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   * @default undefined\n   */\n  readonly branchNamePrefixes?: string[];\n  /**\n   * If CodeConnections are used - this is the ARN of the connection.\n   */\n  readonly codeConnectionArn?: string;\n  /**\n   * The Owner and Repository name for instance, user Bob with git repository\n   * ACME becomes \"Bob/ACME\"\n   */\n  readonly repositoryOwnerAndName?: string;\n}\n\n/**\n * Properties to configure the PDKPipeline with CodeCommit as source.\n *\n * Note: Due to limitations with JSII and generic support it should be noted that\n * the synth, synthShellStepPartialProps.input and\n * synthShellStepPartialProps.primaryOutputDirectory properties will be ignored\n * if passed in to this construct.\n *\n * synthShellStepPartialProps.commands is marked as a required field, however\n * if you pass in [] the default commands of this construct will be retained.\n */\nexport interface PDKPipelineProps extends SharePDKPipelineProps {\n  /**\n   * The repository to add the pipeline to.\n   */\n  readonly codeCommitRepository?: IRepository;\n  /**\n   * Name of the CodeCommit repository to create.\n   */\n  readonly repositoryName: string;\n  /**\n   * Possible values for a resource's Removal Policy\n   * The removal policy controls what happens to the resource if it stops being managed by CloudFormation.\n   */\n  readonly codeCommitRemovalPolicy?: RemovalPolicy;\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: PDKPipeline.ALL_BRANCHES,\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   * @default undefined\n   */\n  readonly branchNamePrefixes?: string[];\n}\n\n/**\n * Properties to configure the PDKPipeline with a CodeConnections as source.\n *\n * Note: Due to limitations with JSII and generic support it should be noted that\n * the synth, synthShellStepPartialProps.input and\n * synthShellStepPartialProps.primaryOutputDirectory properties will be ignored\n * if passed in to this construct.\n *\n * synthShellStepPartialProps.commands is marked as a required field, however\n * if you pass in [] the default commands of this construct will be retained.\n */\nexport interface PDKPipelineWithCodeConnectionProps\n  extends SharePDKPipelineProps {\n  /**\n   * The Arn of the CodeConnection.\n   */\n  readonly codeConnectionArn: string;\n  /**\n   * The Owner and Repository name for instance, user Bob with git repository\n   * ACME becomes \"Bob/ACME\"\n   */\n  readonly repositoryOwnerAndName: string;\n}\n\n/**\n * Properties to help the isDefaultBranch function determine the default branch name.\n */\nexport interface IsDefaultBranchProps {\n  /**\n   * The current node to fetch defaultBranchName from context.\n   */\n  readonly node?: Node;\n\n  /**\n   * Specify the default branch name without context.\n   */\n  readonly defaultBranchName?: string;\n}\n\n/**\n * An extension to CodePipeline which configures sane defaults for a NX Monorepo\n * codebase. In addition to this, it also creates a CodeCommit repository with\n * automated PR builds and approvals.\n */\nclass BasePDKPipeline extends Construct {\n  static readonly ALL_BRANCHES = [\"\"];\n  static readonly defaultBranchName = DEFAULT_BRANCH_NAME;\n\n  /**\n   * A helper function to normalize the branch name with only alphanumeric characters and hypens ('-').\n   * @param branchName The name of the branch to normalize.\n   * @returns The normalized branch name.\n   */\n  public static normalizeBranchName(branchName: string): string {\n    return branchName.replace(/[^a-zA-Z0-9-]/g, \"-\");\n  }\n\n  /**\n   * A helper function to determine if the current branch is the default branch.\n   *\n   * If there is no BRANCH environment variable, then assume this is the default\n   * branch. Otherwise, check that BRANCH matches the default branch name.\n   *\n   * The default branch name is determined in the following priority:\n   *\n   * 1. defaultBranchName property\n   * 2. defaultBranchName context\n   * 3. PDKPipeline.defaultBranchName constant\n   *\n   * @param props? {\n   *    defaultBranchName? Specify the default branch name without context.\n   *    node? The current app to fetch defaultBranchName from context.\n   * }\n   * @returns True if the current branch is the default branch.\n   */\n  public static isDefaultBranch(\n    props: IsDefaultBranchProps = {\n      defaultBranchName: undefined,\n      node: undefined,\n    }\n  ): boolean {\n    if (!process.env.BRANCH) {\n      return true;\n    }\n    const defaultBranchName: string =\n      props.defaultBranchName ||\n      (props.node && props.node.tryGetContext(\"defaultBranchName\")) ||\n      BasePDKPipeline.defaultBranchName;\n    return defaultBranchName === process.env.BRANCH;\n  }\n\n  /**\n   * A helper function to create a branch prefix. The prefix is empty on the default branch.\n   * @param props? {\n   *    defaultBranchName? Specify the default branch name without context.\n   *    node? The current app to fetch defaultBranchName from context.\n   * }\n   * @returns The branch prefix.\n   */\n  public static getBranchPrefix(\n    props: IsDefaultBranchProps = {\n      defaultBranchName: undefined,\n      node: undefined,\n    }\n  ): string {\n    return BasePDKPipeline.isDefaultBranch(props)\n      ? \"\"\n      : BasePDKPipeline.normalizeBranchName(process.env.BRANCH!) + \"-\";\n  }\n\n  readonly codePipeline: CodePipeline;\n  readonly codeRepository: IRepository | undefined;\n  private readonly sonarCodeScannerConfig?: SonarCodeScannerConfig;\n  private readonly branchNamePrefixes?: string[];\n  private readonly defaultBranchName?: string;\n  private readonly repositoryName: string;\n\n  public constructor(\n    scope: Construct,\n    id: string,\n    props: BasePDKPipelineProps\n  ) {\n    super(scope, id);\n\n    this.node.setContext(\n      \"@aws-cdk/aws-s3:serverAccessLogsUseBucketPolicy\",\n      true\n    );\n\n    let source: CodePipelineSource;\n\n    const branch =\n      process.env.BRANCH || props.defaultBranchName || DEFAULT_BRANCH_NAME;\n\n    if (props.useCodeCommit) {\n      let codeRepository: IRepository;\n      const repositoryName = props.repositoryName || \"\";\n      if (\n        BasePDKPipeline.isDefaultBranch({\n          node: this.node,\n          defaultBranchName: props.defaultBranchName,\n        })\n      ) {\n        // In the default branch, create a CodeCommit repository\n        codeRepository = new Repository(this, \"CodeRepository\", {\n          repositoryName,\n        });\n        codeRepository.applyRemovalPolicy(\n          props.codeCommitRemovalPolicy ?? RemovalPolicy.RETAIN\n        );\n      } else {\n        // In a non-default branch, use an existing CodeCommit repository\n        codeRepository = Repository.fromRepositoryName(\n          scope,\n          \"CodeRepository\",\n          repositoryName\n        );\n      }\n\n      source = CodePipelineSource.codeCommit(codeRepository, branch);\n\n      this.codeRepository = codeRepository;\n    } else {\n      const repositoryOwnerAndName = props.repositoryOwnerAndName || \"\";\n      const codeConnectionArn = props.codeConnectionArn || \"\";\n      source = CodePipelineSource.connection(\n        repositoryOwnerAndName,\n        props.defaultBranchName || DEFAULT_BRANCH_NAME,\n        {\n          connectionArn: codeConnectionArn,\n        }\n      );\n    }\n\n    const accessLogsBucket = new Bucket(this, \"AccessLogsBucket\", {\n      versioned: false,\n      enforceSSL: true,\n      autoDeleteObjects: true,\n      removalPolicy: RemovalPolicy.DESTROY,\n      encryption: BucketEncryption.S3_MANAGED,\n      objectOwnership: ObjectOwnership.OBJECT_WRITER,\n      publicReadAccess: false,\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n    });\n\n    const artifactBucket = new Bucket(this, \"ArtifactsBucket\", {\n      enforceSSL: true,\n      autoDeleteObjects: true,\n      removalPolicy: RemovalPolicy.DESTROY,\n      encryption: props.crossAccountKeys\n        ? BucketEncryption.KMS\n        : BucketEncryption.S3_MANAGED,\n      encryptionKey: props.crossAccountKeys\n        ? new Key(this, \"ArtifactKey\", {\n            enableKeyRotation: true,\n            removalPolicy: RemovalPolicy.DESTROY,\n          })\n        : undefined,\n      objectOwnership: ObjectOwnership.BUCKET_OWNER_ENFORCED,\n      publicReadAccess: false,\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n      serverAccessLogsPrefix: \"access-logs\",\n      serverAccessLogsBucket: accessLogsBucket,\n    });\n\n    const codePipeline = new Pipeline(this, \"CodePipeline\", {\n      enableKeyRotation: props.crossAccountKeys,\n      restartExecutionOnUpdate: true,\n      crossAccountKeys: props.crossAccountKeys,\n      artifactBucket,\n      pipelineType: PipelineType.V1,\n    });\n\n    const {\n      input,\n      primaryOutputDirectory,\n      commands,\n      ...synthShellStepPartialProps\n    } = props.synthShellStepPartialProps || {};\n\n    const synthShellStep = new ShellStep(\"Synth\", {\n      input: source,\n      env:\n        props.branchNamePrefixes && props.branchNamePrefixes.length > 0\n          ? {\n              BRANCH: branch,\n            }\n          : undefined,\n      installCommands: [\"npm install -g aws-cdk pnpm\", \"npx projen install\"],\n      commands:\n        commands && commands.length > 0 ? commands : [\"npx projen build\"],\n      primaryOutputDirectory: props.primarySynthDirectory,\n      ...(synthShellStepPartialProps || {}),\n    });\n\n    synthShellStep.addOutputDirectory(\".\");\n\n    const codePipelineProps: CodePipelineProps = {\n      codePipeline,\n      ...props,\n      crossAccountKeys: undefined,\n      synth: synthShellStep,\n    };\n\n    this.codePipeline = new CodePipeline(this, id, codePipelineProps);\n    this.sonarCodeScannerConfig = props.sonarCodeScannerConfig\n      ? {\n          cdkOutDir: props.primarySynthDirectory,\n          ...props.sonarCodeScannerConfig,\n        }\n      : undefined;\n    this.branchNamePrefixes = props.branchNamePrefixes;\n    this.defaultBranchName = props.defaultBranchName;\n    this.repositoryName =\n      (props.useCodeCommit\n        ? props.repositoryName\n        : props.repositoryOwnerAndName) || \"\";\n\n    if (this.codeRepository && props.branchNamePrefixes) {\n      if (\n        PDKPipeline.isDefaultBranch({\n          node: this.node,\n          defaultBranchName: props.defaultBranchName,\n        })\n      ) {\n        new FeatureBranches(this, \"FeatureBranchPipelines\", {\n          codeRepository: this.codeRepository,\n          cdkSrcDir:\n            props.cdkSrcDir || path.dirname(props.primarySynthDirectory),\n          synthShellStepPartialProps: props.synthShellStepPartialProps,\n          cdkCommand: props.cdkCommand,\n          branchNamePrefixes: props.branchNamePrefixes,\n          defaultBranchName: props.defaultBranchName || DEFAULT_BRANCH_NAME,\n          codeBuildDefaults: props.codeBuildDefaults,\n          dockerEnabledForSynth: props.dockerEnabledForSynth,\n        });\n      } else {\n        Tags.of(Stack.of(this)).add(\"FeatureBranch\", branch);\n        Tags.of(Stack.of(this)).add(\"RepoName\", this.repositoryName);\n      }\n    }\n\n    if (props.useCodeCommit && this.codeRepository) {\n      new CfnOutput(this, \"CodeRepositoryGRCUrl\", {\n        value: this.codeRepository.repositoryCloneUrlGrc,\n      });\n    }\n  }\n\n  /**\n   * @inheritDoc\n   */\n  addStage(stage: Stage, options?: AddStageOpts): StageDeployment {\n    if (\n      this.branchNamePrefixes &&\n      !PDKPipeline.isDefaultBranch({\n        node: stage.node,\n        defaultBranchName: this.defaultBranchName,\n      })\n    ) {\n      Tags.of(stage).add(\"FeatureBranch\", process.env.BRANCH!);\n      Tags.of(stage).add(\"RepoName\", this.repositoryName);\n    }\n    // Add any root Aspects to the stage level as currently this doesn't happen automatically\n    Aspects.of(stage.node.root).all.forEach((aspect) =>\n      Aspects.of(stage).add(aspect)\n    );\n    return this.codePipeline.addStage(stage, options);\n  }\n\n  buildPipeline() {\n    this.codePipeline.buildPipeline();\n\n    this.sonarCodeScannerConfig &&\n      new SonarCodeScanner(this, \"SonarCodeScanner\", {\n        artifactBucketArn: this.codePipeline.pipeline.artifactBucket.bucketArn,\n        artifactBucketKeyArn:\n          this.codePipeline.pipeline.artifactBucket.encryptionKey?.keyArn,\n        synthBuildArn: this.codePipeline.synthProject.projectArn,\n        ...this.sonarCodeScannerConfig,\n      });\n\n    this.suppressCDKViolations();\n  }\n\n  suppressCDKViolations() {\n    this.suppressRules(\n      [\"AwsSolutions-IAM5\", \"AwsPrototyping-IAMNoWildcardPermissions\"],\n      \"Wildcards are needed for dynamically created resources.\"\n    );\n\n    this.suppressRules(\n      [\n        \"AwsSolutions-CB4\",\n        \"AwsPrototyping-CodeBuildProjectKMSEncryptedArtifacts\",\n      ],\n      \"Encryption of Codebuild is not required.\"\n    );\n\n    this.suppressRules(\n      [\"AwsSolutions-S1\", \"AwsPrototyping-S3BucketLoggingEnabled\"],\n      \"Access Log buckets should not have s3 bucket logging\"\n    );\n  }\n\n  private suppressRules(rules: string[], reason: string) {\n    NagSuppressions.addResourceSuppressions(\n      this,\n      rules.map((r) => ({\n        id: r,\n        reason,\n      })),\n      true\n    );\n  }\n}\n\n/**\n * An extension to CodePipeline which configures same defaults for a NX Monorepo\n * codebase. In addition to this, it also creates a CodeCommit repository with\n * automated PR builds and approvals.\n */\nexport class PDKPipeline extends BasePDKPipeline {\n  constructor(scope: Construct, id: string, props: PDKPipelineProps) {\n    super(scope, id, {\n      ...props,\n      useCodeCommit: true,\n      codeConnectionArn: undefined,\n      repositoryOwnerAndName: undefined,\n    });\n  }\n}\n\n/**\n * An extension to CodePipeline which configures same defaults for a NX Monorepo and using a AWS CodeConnections as source.\n */\nexport class PDKPipelineWithCodeConnection extends BasePDKPipeline {\n  constructor(\n    scope: Construct,\n    id: string,\n    props: PDKPipelineWithCodeConnectionProps\n  ) {\n    super(scope, id, {\n      ...props,\n      useCodeCommit: false,\n    });\n  }\n}\n"]}