UNPKG

aws-delivlib

Version:

A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.

138 lines • 18.1 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 __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"]}