aws-delivlib
Version:
A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.
138 lines • 18.1 kB
JavaScript
;
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 __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
__setModuleDefault(result, mod);
return result;
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.MirrorSource = void 0;
const path = __importStar(require("node:path"));
const aws_cdk_lib_1 = require("aws-cdk-lib");
/**
* Source of the image.
*/
class MirrorSource {
/**
* Configure an image from DockerHub.
*
* @param image e.g jsii/superchain
* @param tag optional, defaults to 'latest'
*
* @deprecated This method's name inaccurately expresses that the image comes
* from DockerHub, when any publicly-accessible repository can be used. Prefer
* using `fromImageName(string, string?)` instead, which is more aptly named.
*/
static fromDockerHub(image, tag = 'latest') {
return this.fromPublicImage(image, tag);
}
/**
* Configure an image from DockerHub or a repository-qualified image name.
*
* @param image e.g public.ecr.aws/jsii/superchain
* @param tag optional, defaults to 'latest'
* @param ecrRepositoryName the name of the ECR Repository to use (e.g: jsii/superchain)
*/
static fromPublicImage(image, tag = 'latest', ecrRepositoryName = image.includes('/') ? image : `library/${image}`) {
class DockerHubMirrorSource extends MirrorSource {
constructor() {
if (image.includes(':')) {
throw new Error('image must not include tag');
}
// simulates DockerHub by prefixing library/ to official images
const repositoryName = image.includes('/') ? image : `library/${image}`;
super(repositoryName, tag, undefined, ecrRepositoryName);
}
bind(options) {
const ecrImageUri = `${options.ecrRegistry}/${this.ecrRepositoryName}:${this.tag}`;
return {
commands: [
`docker pull ${this.repositoryName}:${this.tag}`,
`docker tag ${this.repositoryName}:${this.tag} ${ecrImageUri}`,
],
repositoryName: this.ecrRepositoryName,
tag: this.tag,
};
}
}
return new DockerHubMirrorSource();
}
/**
* DEPRECATED
* @deprecated use fromDir()
*/
static fromDirectory(directory, repositoryName, tag) {
return this.fromDir(directory, repositoryName, { tag });
}
/**
* Configure an image from a local directory.
*
* @param directory Path to directory containing the Dockerfile.
* @param repositoryName Repository name of the built image.
* @param options additional configuration options
*/
static fromDir(directory, repositoryName, opts = {}) {
class DirectoryMirrorSource extends MirrorSource {
constructor() {
super(repositoryName, opts.tag ?? 'latest', directory);
}
bind(options) {
const asset = new aws_cdk_lib_1.aws_s3_assets.Asset(options.scope, `BuildContext${this.directory}${JSON.stringify(opts.buildArgs ?? {})}`, {
path: this.directory,
// Need to give an explicit displayName, because the directory might
// be absolute, and the directories and args are liable to change on
// every pipeline run.
displayName: `EcrMirror directory ${path.basename(directory)}`,
});
if (options.syncJob) {
asset.grantRead(options.syncJob);
}
const ecrImageUri = `${options.ecrRegistry}/${this.ecrRepositoryName}:${this.tag}`;
const cmdFlags = [];
cmdFlags.push('--pull');
cmdFlags.push('-t', ecrImageUri);
if (opts.buildArgs) {
Object.entries(opts.buildArgs).forEach(([k, v]) => cmdFlags.push('--build-arg', `${k}=${v}`));
}
const zipFile = `${this.repositoryName}.zip`;
const tmpDir = this.repositoryName;
return {
commands: [
`rm -rf ${zipFile} ${tmpDir}`,
`aws s3 cp ${asset.s3ObjectUrl} ${zipFile}`,
`unzip ${zipFile} -d ${tmpDir}`,
`docker build ${cmdFlags.join(' ')} ${tmpDir}`,
],
repositoryName: this.ecrRepositoryName,
tag: this.tag,
};
}
}
return new DirectoryMirrorSource();
}
constructor(repositoryName, tag, directory, ecrRepositoryName = repositoryName) {
this.repositoryName = repositoryName;
this.tag = tag;
this.directory = directory;
this.ecrRepositoryName = ecrRepositoryName;
}
}
exports.MirrorSource = MirrorSource;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mirror-source.js","sourceRoot":"","sources":["mirror-source.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'node:path';\nimport {\n  aws_codebuild as codebuild,\n  aws_s3_assets as s3Assets,\n} from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\n\nexport interface MirrorSourceBindOptions {\n  /**\n   * The target ECR registry\n   */\n  readonly ecrRegistry: string;\n  /**\n   * The scope to attach any constructs that may also be needed.\n   */\n  readonly scope: Construct;\n\n  /**\n   * The CodeBuild project that will run the synchronization between DockerHub and ECR.\n   * @default - either no sync job is present or it's not defined yet.\n   */\n  readonly syncJob?: codebuild.IProject;\n}\n\nexport interface MirrorSourceConfig {\n  /**\n   * The commands to run to retrieve the docker image.\n   * e.g. ['docker pull <image-id>']\n   */\n  readonly commands: string[];\n\n  /**\n   * The name of the target ECR repository.\n   */\n  readonly repositoryName: string;\n\n  /**\n   * The tag to be use for the target ECR image.\n   */\n  readonly tag: string;\n}\n\n/** Additional options when configuring a Mirror Source from a local directory */\nexport interface MirrorSourceDirectoryOptions {\n  /**\n   * Tag of the built image.\n   * @default 'latest'\n   */\n  readonly tag?: string;\n\n  /**\n   * Build args to pass to the `docker build` command.\n   *\n   * @default - no build args are passed\n   */\n  readonly buildArgs?: { [key: string]: string };\n}\n\n/**\n * Source of the image.\n */\nexport abstract class MirrorSource {\n\n  /**\n   * Configure an image from DockerHub.\n   *\n   * @param image e.g jsii/superchain\n   * @param tag optional, defaults to 'latest'\n   *\n   * @deprecated This method's name inaccurately expresses that the image comes\n   * from DockerHub, when any publicly-accessible repository can be used. Prefer\n   * using `fromImageName(string, string?)` instead, which is more aptly named.\n   */\n  public static fromDockerHub(image: string, tag: string = 'latest'): MirrorSource {\n    return this.fromPublicImage(image, tag);\n  }\n\n  /**\n   * Configure an image from DockerHub or a repository-qualified image name.\n   *\n   * @param image e.g public.ecr.aws/jsii/superchain\n   * @param tag optional, defaults to 'latest'\n   * @param ecrRepositoryName the name of the ECR Repository to use (e.g: jsii/superchain)\n   */\n  public static fromPublicImage(image: string, tag: string = 'latest', ecrRepositoryName: string = image.includes('/') ? image : `library/${image}`): MirrorSource {\n    class DockerHubMirrorSource extends MirrorSource {\n      constructor() {\n        if (image.includes(':')) {\n          throw new Error('image must not include tag');\n        }\n        // simulates DockerHub by prefixing library/ to official images\n        const repositoryName = image.includes('/') ? image : `library/${image}`;\n        super(repositoryName, tag, undefined, ecrRepositoryName);\n      }\n\n      public bind(options: MirrorSourceBindOptions): MirrorSourceConfig {\n        const ecrImageUri = `${options.ecrRegistry}/${this.ecrRepositoryName}:${this.tag}`;\n        return {\n          commands: [\n            `docker pull ${this.repositoryName}:${this.tag}`,\n            `docker tag ${this.repositoryName}:${this.tag} ${ecrImageUri}`,\n          ],\n          repositoryName: this.ecrRepositoryName,\n          tag: this.tag,\n        };\n      }\n    }\n\n    return new DockerHubMirrorSource();\n  }\n\n  /**\n   * DEPRECATED\n   * @deprecated use fromDir()\n   */\n  public static fromDirectory(directory: string, repositoryName: string, tag?: string): MirrorSource {\n    return this.fromDir(directory, repositoryName, { tag });\n  }\n\n  /**\n   * Configure an image from a local directory.\n   *\n   * @param directory Path to directory containing the Dockerfile.\n   * @param repositoryName Repository name of the built image.\n   * @param options additional configuration options\n   */\n  public static fromDir(directory: string, repositoryName: string, opts: MirrorSourceDirectoryOptions = {}): MirrorSource {\n    class DirectoryMirrorSource extends MirrorSource {\n      constructor() {\n        super(repositoryName, opts.tag ?? 'latest', directory);\n      }\n\n      public bind(options: MirrorSourceBindOptions): MirrorSourceConfig {\n        const asset = new s3Assets.Asset(options.scope, `BuildContext${this.directory}${JSON.stringify(opts.buildArgs ?? {})}`, {\n          path: this.directory!,\n\n          // Need to give an explicit displayName, because the directory might\n          // be absolute, and the directories and args are liable to change on\n          // every pipeline run.\n          displayName: `EcrMirror directory ${path.basename(directory)}`,\n        });\n        if (options.syncJob) {\n          asset.grantRead(options.syncJob);\n        }\n        const ecrImageUri = `${options.ecrRegistry}/${this.ecrRepositoryName}:${this.tag}`;\n        const cmdFlags = [];\n        cmdFlags.push('--pull');\n        cmdFlags.push('-t', ecrImageUri);\n\n        if (opts.buildArgs) {\n          Object.entries(opts.buildArgs).forEach(([k, v]) => cmdFlags.push('--build-arg', `${k}=${v}`));\n        }\n\n        const zipFile = `${this.repositoryName}.zip`;\n        const tmpDir = this.repositoryName;\n\n        return {\n          commands: [\n            `rm -rf ${zipFile} ${tmpDir}`,\n            `aws s3 cp ${asset.s3ObjectUrl} ${zipFile}`,\n            `unzip ${zipFile} -d ${tmpDir}`,\n            `docker build ${cmdFlags.join(' ')} ${tmpDir}`,\n          ],\n          repositoryName: this.ecrRepositoryName,\n          tag: this.tag,\n        };\n      }\n    }\n    return new DirectoryMirrorSource();\n  }\n\n  private constructor(\n    protected readonly repositoryName: string,\n    protected readonly tag: string,\n    protected readonly directory?: string,\n    protected readonly ecrRepositoryName = repositoryName,\n  ) {\n  }\n\n  /**\n   * Bind the source with the EcrMirror construct.\n   */\n  public abstract bind(options: MirrorSourceBindOptions): MirrorSourceConfig;\n}\n"]}