@aws-cdk/aws-codepipeline
Version:
Better interface to AWS Code Pipeline
84 lines • 13.8 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.CrossRegionSupportStack = exports.CrossRegionSupportConstruct = void 0;
const kms = require("@aws-cdk/aws-kms");
const s3 = require("@aws-cdk/aws-s3");
const cdk = require("@aws-cdk/core");
// keep this import separate from other imports to reduce chance for merge conflicts with v2-main
// eslint-disable-next-line no-duplicate-imports, import/order
const core_1 = require("@aws-cdk/core");
const REQUIRED_ALIAS_PREFIX = 'alias/';
/**
* A class needed to work around CodePipeline's extremely small (100 characters)
* limit for the name/ARN of the key in the ArtifactStore.
* Limits the length of the alias' auto-generated name to 50 characters.
*/
class AliasWithShorterGeneratedName extends kms.Alias {
generatePhysicalName() {
let baseName = super.generatePhysicalName();
if (baseName.startsWith(REQUIRED_ALIAS_PREFIX)) {
// remove the prefix, because we're taking the last characters of the name below
baseName = baseName.substring(REQUIRED_ALIAS_PREFIX.length);
}
const maxLength = 50 - REQUIRED_ALIAS_PREFIX.length;
// take the last characters, as they include the hash,
// and so have a higher chance of not colliding
return REQUIRED_ALIAS_PREFIX + lastNCharacters(baseName, maxLength);
}
}
function lastNCharacters(str, n) {
const startIndex = Math.max(str.length - n, 0);
return str.substring(startIndex);
}
class CrossRegionSupportConstruct extends core_1.Construct {
constructor(scope, id, props = {}) {
super(scope, id);
const createKmsKey = props.createKmsKey ?? true;
let encryptionAlias;
if (createKmsKey) {
const encryptionKey = new kms.Key(this, 'CrossRegionCodePipelineReplicationBucketEncryptionKey', {
removalPolicy: cdk.RemovalPolicy.DESTROY,
enableKeyRotation: props.enableKeyRotation,
});
encryptionAlias = new AliasWithShorterGeneratedName(this, 'CrossRegionCodePipelineReplicationBucketEncryptionAlias', {
targetKey: encryptionKey,
aliasName: cdk.PhysicalName.GENERATE_IF_NEEDED,
removalPolicy: cdk.RemovalPolicy.DESTROY,
});
}
this.replicationBucket = new s3.Bucket(this, 'CrossRegionCodePipelineReplicationBucket', {
bucketName: cdk.PhysicalName.GENERATE_IF_NEEDED,
encryption: encryptionAlias ? s3.BucketEncryption.KMS : s3.BucketEncryption.KMS_MANAGED,
encryptionKey: encryptionAlias,
enforceSSL: true,
blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,
});
}
}
exports.CrossRegionSupportConstruct = CrossRegionSupportConstruct;
/**
* A Stack containing resources required for the cross-region CodePipeline functionality to work.
* This class is private to the aws-codepipeline package.
*/
class CrossRegionSupportStack extends cdk.Stack {
constructor(scope, id, props) {
super(scope, id, {
stackName: generateStackName(props),
env: {
region: props.region,
account: props.account,
},
synthesizer: props.synthesizer,
});
const crossRegionSupportConstruct = new CrossRegionSupportConstruct(this, 'Default', {
createKmsKey: props.createKmsKey,
enableKeyRotation: props.enableKeyRotation,
});
this.replicationBucket = crossRegionSupportConstruct.replicationBucket;
}
}
exports.CrossRegionSupportStack = CrossRegionSupportStack;
function generateStackName(props) {
return `${props.pipelineStackName}-support-${props.region}`;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cross-region-support-stack.js","sourceRoot":"","sources":["cross-region-support-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import * as kms from '@aws-cdk/aws-kms';\nimport * as s3 from '@aws-cdk/aws-s3';\nimport * as cdk from '@aws-cdk/core';\n\n// keep this import separate from other imports to reduce chance for merge conflicts with v2-main\n// eslint-disable-next-line no-duplicate-imports, import/order\nimport { Construct } from '@aws-cdk/core';\n\nconst REQUIRED_ALIAS_PREFIX = 'alias/';\n\n/**\n * A class needed to work around CodePipeline's extremely small (100 characters)\n * limit for the name/ARN of the key in the ArtifactStore.\n * Limits the length of the alias' auto-generated name to 50 characters.\n */\nclass AliasWithShorterGeneratedName extends kms.Alias {\n  protected generatePhysicalName(): string {\n    let baseName = super.generatePhysicalName();\n    if (baseName.startsWith(REQUIRED_ALIAS_PREFIX)) {\n      // remove the prefix, because we're taking the last characters of the name below\n      baseName = baseName.substring(REQUIRED_ALIAS_PREFIX.length);\n    }\n    const maxLength = 50 - REQUIRED_ALIAS_PREFIX.length;\n    // take the last characters, as they include the hash,\n    // and so have a higher chance of not colliding\n    return REQUIRED_ALIAS_PREFIX + lastNCharacters(baseName, maxLength);\n  }\n}\n\nfunction lastNCharacters(str: string, n: number) {\n  const startIndex = Math.max(str.length - n, 0);\n  return str.substring(startIndex);\n}\n\n/**\n * Props for the support stack\n */\nexport interface CrossRegionSupportConstructProps {\n  /**\n   * Whether to create the KMS CMK\n   *\n   * (Required for cross-account deployments)\n   *\n   * @default true\n   */\n  readonly createKmsKey?: boolean;\n\n  /**\n   * Enables KMS key rotation for cross-account keys.\n   *\n   * @default - false (key rotation is disabled)\n   */\n  readonly enableKeyRotation?: boolean;\n}\n\nexport class CrossRegionSupportConstruct extends Construct {\n  public readonly replicationBucket: s3.IBucket;\n\n  constructor(scope: Construct, id: string, props: CrossRegionSupportConstructProps = {}) {\n    super(scope, id);\n\n    const createKmsKey = props.createKmsKey ?? true;\n\n    let encryptionAlias;\n    if (createKmsKey) {\n      const encryptionKey = new kms.Key(this, 'CrossRegionCodePipelineReplicationBucketEncryptionKey', {\n        removalPolicy: cdk.RemovalPolicy.DESTROY,\n        enableKeyRotation: props.enableKeyRotation,\n      });\n      encryptionAlias = new AliasWithShorterGeneratedName(this, 'CrossRegionCodePipelineReplicationBucketEncryptionAlias', {\n        targetKey: encryptionKey,\n        aliasName: cdk.PhysicalName.GENERATE_IF_NEEDED,\n        removalPolicy: cdk.RemovalPolicy.DESTROY,\n      });\n    }\n    this.replicationBucket = new s3.Bucket(this, 'CrossRegionCodePipelineReplicationBucket', {\n      bucketName: cdk.PhysicalName.GENERATE_IF_NEEDED,\n      encryption: encryptionAlias ? s3.BucketEncryption.KMS : s3.BucketEncryption.KMS_MANAGED,\n      encryptionKey: encryptionAlias,\n      enforceSSL: true,\n      blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,\n    });\n  }\n}\n\n/**\n * Construction properties for {@link CrossRegionSupportStack}.\n * This interface is private to the aws-codepipeline package.\n */\nexport interface CrossRegionSupportStackProps {\n  /**\n   * The name of the Stack the Pipeline itself belongs to.\n   * Used to generate a more friendly name for the support Stack.\n   */\n  readonly pipelineStackName: string;\n\n  /**\n   * The AWS region this Stack resides in.\n   */\n  readonly region: string;\n\n  /**\n   * The AWS account ID this Stack belongs to.\n   *\n   * @example '012345678901'\n   */\n  readonly account: string;\n\n  readonly synthesizer: cdk.IStackSynthesizer | undefined;\n\n  /**\n   * Whether or not to create a KMS key in the support stack\n   *\n   * (Required for cross-account deployments)\n   *\n   * @default true\n   */\n  readonly createKmsKey?: boolean;\n\n  /**\n   * Enables KMS key rotation for cross-account keys.\n   *\n   * @default - false (key rotation is disabled)\n   */\n  readonly enableKeyRotation?: boolean;\n}\n\n/**\n * A Stack containing resources required for the cross-region CodePipeline functionality to work.\n * This class is private to the aws-codepipeline package.\n */\nexport class CrossRegionSupportStack extends cdk.Stack {\n  /**\n   * The name of the S3 Bucket used for replicating the Pipeline's artifacts into the region.\n   */\n  public readonly replicationBucket: s3.IBucket;\n\n  constructor(scope: Construct, id: string, props: CrossRegionSupportStackProps) {\n    super(scope, id, {\n      stackName: generateStackName(props),\n      env: {\n        region: props.region,\n        account: props.account,\n      },\n      synthesizer: props.synthesizer,\n    });\n\n    const crossRegionSupportConstruct = new CrossRegionSupportConstruct(this, 'Default', {\n      createKmsKey: props.createKmsKey,\n      enableKeyRotation: props.enableKeyRotation,\n    });\n    this.replicationBucket = crossRegionSupportConstruct.replicationBucket;\n  }\n}\n\nfunction generateStackName(props: CrossRegionSupportStackProps): string {\n  return `${props.pipelineStackName}-support-${props.region}`;\n}\n"]}