cdk-assets 3.3.0 → 4.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (48) hide show
  1. package/README.md +4 -4
  2. package/THIRD_PARTY_LICENSES +18764 -0
  3. package/bin/cdk-assets.js +109830 -62
  4. package/bin/docker-credential-cdk-assets.js +4 -4
  5. package/bin/list.js +3 -3
  6. package/bin/logging.js +3 -1
  7. package/bin/publish.js +6 -6
  8. package/package.json +11 -30
  9. package/lib/asset-manifest.d.ts +0 -186
  10. package/lib/asset-manifest.js +0 -270
  11. package/lib/aws-types.d.ts +0 -1666
  12. package/lib/aws-types.js +0 -144
  13. package/lib/aws.d.ts +0 -71
  14. package/lib/aws.js +0 -131
  15. package/lib/index.d.ts +0 -4
  16. package/lib/index.js +0 -21
  17. package/lib/private/archive.d.ts +0 -3
  18. package/lib/private/archive.js +0 -87
  19. package/lib/private/asset-handler.d.ts +0 -60
  20. package/lib/private/asset-handler.js +0 -3
  21. package/lib/private/docker-credentials.d.ts +0 -35
  22. package/lib/private/docker-credentials.js +0 -90
  23. package/lib/private/docker.d.ts +0 -98
  24. package/lib/private/docker.js +0 -235
  25. package/lib/private/fs-extra.d.ts +0 -3
  26. package/lib/private/fs-extra.js +0 -37
  27. package/lib/private/handlers/client-options.d.ts +0 -3
  28. package/lib/private/handlers/client-options.js +0 -12
  29. package/lib/private/handlers/container-images.d.ts +0 -22
  30. package/lib/private/handlers/container-images.js +0 -224
  31. package/lib/private/handlers/files.d.ts +0 -14
  32. package/lib/private/handlers/files.js +0 -289
  33. package/lib/private/handlers/index.d.ts +0 -3
  34. package/lib/private/handlers/index.js +0 -16
  35. package/lib/private/p-limit.d.ts +0 -10
  36. package/lib/private/p-limit.js +0 -51
  37. package/lib/private/placeholders.d.ts +0 -10
  38. package/lib/private/placeholders.js +0 -34
  39. package/lib/private/shell.d.ts +0 -24
  40. package/lib/private/shell.js +0 -131
  41. package/lib/private/util.d.ts +0 -5
  42. package/lib/private/util.js +0 -16
  43. package/lib/progress.d.ts +0 -114
  44. package/lib/progress.js +0 -104
  45. package/lib/publishing.d.ts +0 -118
  46. package/lib/publishing.js +0 -193
  47. package/scripts/manual-test-manifest.json +0 -12
  48. package/scripts/manual-test.sh +0 -22
@@ -1,235 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.DockerFactory = exports.Docker = void 0;
4
- const fs = require("fs");
5
- const os = require("os");
6
- const path = require("path");
7
- const docker_credentials_1 = require("./docker-credentials");
8
- const shell_1 = require("./shell");
9
- const util_1 = require("./util");
10
- const progress_1 = require("../progress");
11
- var InspectImageErrorCode;
12
- (function (InspectImageErrorCode) {
13
- InspectImageErrorCode[InspectImageErrorCode["Docker"] = 1] = "Docker";
14
- InspectImageErrorCode[InspectImageErrorCode["Podman"] = 125] = "Podman";
15
- })(InspectImageErrorCode || (InspectImageErrorCode = {}));
16
- class Docker {
17
- constructor(eventEmitter, subprocessOutputDestination) {
18
- this.eventEmitter = eventEmitter;
19
- this.subprocessOutputDestination = subprocessOutputDestination;
20
- this.configDir = undefined;
21
- }
22
- /**
23
- * Whether an image with the given tag exists
24
- */
25
- async exists(tag) {
26
- try {
27
- await this.execute(['inspect', tag], {
28
- subprocessOutputDestination: 'ignore',
29
- });
30
- return true;
31
- }
32
- catch (e) {
33
- const error = e;
34
- /**
35
- * The only error we expect to be thrown will have this property and value.
36
- * If it doesn't, it's unrecognized so re-throw it.
37
- */
38
- if (error.code !== 'PROCESS_FAILED') {
39
- throw error;
40
- }
41
- /**
42
- * If we know the shell command above returned an error, check to see
43
- * if the exit code is one we know to actually mean that the image doesn't
44
- * exist.
45
- */
46
- switch (error.exitCode) {
47
- case InspectImageErrorCode.Docker:
48
- case InspectImageErrorCode.Podman:
49
- // Docker and Podman will return this exit code when an image doesn't exist, return false
50
- // context: https://github.com/aws/aws-cdk/issues/16209
51
- return false;
52
- default:
53
- // This is an error but it's not an exit code we recognize, throw.
54
- throw error;
55
- }
56
- }
57
- }
58
- async build(options) {
59
- const buildCommand = [
60
- 'build',
61
- ...flatten(Object.entries(options.buildArgs || {}).map(([k, v]) => ['--build-arg', `${k}=${v}`])),
62
- ...flatten(Object.entries(options.buildSecrets || {}).map(([k, v]) => ['--secret', `id=${k},${v}`])),
63
- ...(options.buildSsh ? ['--ssh', options.buildSsh] : []),
64
- '--tag',
65
- options.tag,
66
- ...(options.target ? ['--target', options.target] : []),
67
- ...(options.file ? ['--file', options.file] : []),
68
- ...(options.networkMode ? ['--network', options.networkMode] : []),
69
- ...(options.platform ? ['--platform', options.platform] : []),
70
- ...(options.outputs ? options.outputs.map((output) => [`--output=${output}`]) : []),
71
- ...(options.cacheFrom
72
- ? [
73
- ...options.cacheFrom
74
- .map((cacheFrom) => ['--cache-from', this.cacheOptionToFlag(cacheFrom)])
75
- .flat(),
76
- ]
77
- : []),
78
- ...(options.cacheTo ? ['--cache-to', this.cacheOptionToFlag(options.cacheTo)] : []),
79
- ...(options.cacheDisabled ? ['--no-cache'] : []),
80
- '.',
81
- ];
82
- await this.execute(buildCommand, {
83
- cwd: options.directory,
84
- subprocessOutputDestination: this.subprocessOutputDestination,
85
- env: {
86
- BUILDX_NO_DEFAULT_ATTESTATIONS: '1', // Docker Build adds provenance attestations by default that confuse cdk-assets
87
- },
88
- });
89
- }
90
- /**
91
- * Get credentials from ECR and run docker login
92
- */
93
- async login(ecr) {
94
- const credentials = await (0, docker_credentials_1.obtainEcrCredentials)(ecr, this.eventEmitter);
95
- // Use --password-stdin otherwise docker will complain. Loudly.
96
- await this.execute(['login', '--username', credentials.username, '--password-stdin', credentials.endpoint.replace(/^https?:\/\/|\/$/g, '')], {
97
- input: credentials.password,
98
- // Need to ignore otherwise Docker will complain
99
- // 'WARNING! Your password will be stored unencrypted'
100
- // doesn't really matter since it's a token.
101
- subprocessOutputDestination: 'ignore',
102
- });
103
- }
104
- async tag(sourceTag, targetTag) {
105
- await this.execute(['tag', sourceTag, targetTag]);
106
- }
107
- async push(options) {
108
- await this.execute(['push', options.tag], {
109
- subprocessOutputDestination: this.subprocessOutputDestination,
110
- });
111
- }
112
- /**
113
- * If a CDK Docker Credentials file exists, creates a new Docker config directory.
114
- * Sets up `docker-credential-cdk-assets` to be the credential helper for each domain in the CDK config.
115
- * All future commands (e.g., `build`, `push`) will use this config.
116
- *
117
- * See https://docs.docker.com/engine/reference/commandline/login/#credential-helpers for more details on cred helpers.
118
- *
119
- * @returns true if CDK config was found and configured, false otherwise
120
- */
121
- configureCdkCredentials() {
122
- const config = (0, docker_credentials_1.cdkCredentialsConfig)();
123
- if (!config) {
124
- return false;
125
- }
126
- this.configDir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdkDockerConfig'));
127
- const domains = Object.keys(config.domainCredentials);
128
- const credHelpers = domains.reduce((map, domain) => {
129
- map[domain] = 'cdk-assets'; // Use docker-credential-cdk-assets for this domain
130
- return map;
131
- }, {});
132
- fs.writeFileSync(path.join(this.configDir, 'config.json'), JSON.stringify({ credHelpers }), {
133
- encoding: 'utf-8',
134
- });
135
- return true;
136
- }
137
- /**
138
- * Removes any configured Docker config directory.
139
- * All future commands (e.g., `build`, `push`) will use the default config.
140
- *
141
- * This is useful after calling `configureCdkCredentials` to reset to default credentials.
142
- */
143
- resetAuthPlugins() {
144
- this.configDir = undefined;
145
- }
146
- async execute(args, options = {}) {
147
- const configArgs = this.configDir ? ['--config', this.configDir] : [];
148
- const pathToCdkAssets = path.resolve(__dirname, '..', '..', 'bin');
149
- const shellEventPublisher = (0, progress_1.shellEventPublisherFromEventEmitter)(this.eventEmitter);
150
- try {
151
- await (0, shell_1.shell)([getDockerCmd(), ...configArgs, ...args], {
152
- ...options,
153
- shellEventPublisher: shellEventPublisher,
154
- env: {
155
- ...process.env,
156
- ...options.env,
157
- PATH: `${pathToCdkAssets}${path.delimiter}${options.env?.PATH ?? process.env.PATH}`,
158
- },
159
- });
160
- }
161
- catch (e) {
162
- if (e.code === 'ENOENT') {
163
- throw new Error(`Failed to find and execute '${getDockerCmd()}' while attempting to build a container asset. Please install '${getDockerCmd()}' and try again. (Or set the 'CDK_DOCKER ' environment variable to choose a different compatible container client.)`);
164
- }
165
- throw e;
166
- }
167
- }
168
- cacheOptionToFlag(option) {
169
- let flag = `type=${option.type}`;
170
- if (option.params) {
171
- flag +=
172
- ',' +
173
- Object.entries(option.params)
174
- .map(([k, v]) => `${k}=${v}`)
175
- .join(',');
176
- }
177
- return flag;
178
- }
179
- }
180
- exports.Docker = Docker;
181
- /**
182
- * Helps get appropriately configured Docker instances during the container
183
- * image publishing process.
184
- */
185
- class DockerFactory {
186
- constructor() {
187
- this.enterLoggedInDestinationsCriticalSection = (0, util_1.createCriticalSection)();
188
- this.loggedInDestinations = new Set();
189
- }
190
- /**
191
- * Gets a Docker instance for building images.
192
- */
193
- async forBuild(options) {
194
- const docker = new Docker(options.eventEmitter, options.subprocessOutputDestination);
195
- // Default behavior is to login before build so that the Dockerfile can reference images in the ECR repo
196
- // However, if we're in a pipelines environment (for example),
197
- // we may have alternative credentials to the default ones to use for the build itself.
198
- // If the special config file is present, delay the login to the default credentials until the push.
199
- // If the config file is present, we will configure and use those credentials for the build.
200
- let cdkDockerCredentialsConfigured = docker.configureCdkCredentials();
201
- if (!cdkDockerCredentialsConfigured) {
202
- await this.loginOncePerDestination(docker, options);
203
- }
204
- return docker;
205
- }
206
- /**
207
- * Gets a Docker instance for pushing images to ECR.
208
- */
209
- async forEcrPush(options) {
210
- const docker = new Docker(options.eventEmitter, options.subprocessOutputDestination);
211
- await this.loginOncePerDestination(docker, options);
212
- return docker;
213
- }
214
- async loginOncePerDestination(docker, options) {
215
- // Changes: 012345678910.dkr.ecr.us-west-2.amazonaws.com/tagging-test
216
- // To this: 012345678910.dkr.ecr.us-west-2.amazonaws.com
217
- const repositoryDomain = options.repoUri.split('/')[0];
218
- // Ensure one-at-a-time access to loggedInDestinations.
219
- await this.enterLoggedInDestinationsCriticalSection(async () => {
220
- if (this.loggedInDestinations.has(repositoryDomain)) {
221
- return;
222
- }
223
- await docker.login(options.ecr);
224
- this.loggedInDestinations.add(repositoryDomain);
225
- });
226
- }
227
- }
228
- exports.DockerFactory = DockerFactory;
229
- function getDockerCmd() {
230
- return process.env.CDK_DOCKER ?? 'docker';
231
- }
232
- function flatten(x) {
233
- return Array.prototype.concat([], ...x);
234
- }
235
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docker.js","sourceRoot":"","sources":["docker.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { cdkCredentialsConfig, obtainEcrCredentials } from './docker-credentials';\nimport type { ShellOptions, ProcessFailedError } from './shell';\nimport { shell } from './shell';\nimport { createCriticalSection } from './util';\nimport type { IECRClient } from '../aws';\nimport type { SubprocessOutputDestination } from './asset-handler';\nimport type { EventEmitter } from '../progress';\nimport { shellEventPublisherFromEventEmitter } from '../progress';\n\ninterface BuildOptions {\n  readonly directory: string;\n\n  /**\n   * Tag the image with a given repoName:tag combination\n   */\n  readonly tag: string;\n  readonly target?: string;\n  readonly file?: string;\n  readonly buildArgs?: Record<string, string>;\n  readonly buildSecrets?: Record<string, string>;\n  readonly buildSsh?: string;\n  readonly networkMode?: string;\n  readonly platform?: string;\n  readonly outputs?: string[];\n  readonly cacheFrom?: DockerCacheOption[];\n  readonly cacheTo?: DockerCacheOption;\n  readonly cacheDisabled?: boolean;\n}\n\ninterface PushOptions {\n  readonly tag: string;\n}\n\nexport interface DockerCredentialsConfig {\n  readonly version: string;\n  readonly domainCredentials: Record<string, DockerDomainCredentials>;\n}\n\nexport interface DockerDomainCredentials {\n  readonly secretsManagerSecretId?: string;\n  readonly ecrRepository?: string;\n}\n\nenum InspectImageErrorCode {\n  Docker = 1,\n  Podman = 125,\n}\n\nexport interface DockerCacheOption {\n  readonly type: string;\n  readonly params?: { [key: string]: string };\n}\n\nexport class Docker {\n  private configDir: string | undefined = undefined;\n\n  constructor(\n    private readonly eventEmitter: EventEmitter,\n    private readonly subprocessOutputDestination: SubprocessOutputDestination,\n  ) {\n  }\n\n  /**\n   * Whether an image with the given tag exists\n   */\n  public async exists(tag: string) {\n    try {\n      await this.execute(['inspect', tag], {\n        subprocessOutputDestination: 'ignore',\n      });\n      return true;\n    } catch (e: any) {\n      const error: ProcessFailedError = e;\n\n      /**\n       * The only error we expect to be thrown will have this property and value.\n       * If it doesn't, it's unrecognized so re-throw it.\n       */\n      if (error.code !== 'PROCESS_FAILED') {\n        throw error;\n      }\n\n      /**\n       * If we know the shell command above returned an error, check to see\n       * if the exit code is one we know to actually mean that the image doesn't\n       * exist.\n       */\n      switch (error.exitCode) {\n        case InspectImageErrorCode.Docker:\n        case InspectImageErrorCode.Podman:\n          // Docker and Podman will return this exit code when an image doesn't exist, return false\n          // context: https://github.com/aws/aws-cdk/issues/16209\n          return false;\n        default:\n          // This is an error but it's not an exit code we recognize, throw.\n          throw error;\n      }\n    }\n  }\n\n  public async build(options: BuildOptions) {\n    const buildCommand = [\n      'build',\n      ...flatten(\n        Object.entries(options.buildArgs || {}).map(([k, v]) => ['--build-arg', `${k}=${v}`]),\n      ),\n      ...flatten(\n        Object.entries(options.buildSecrets || {}).map(([k, v]) => ['--secret', `id=${k},${v}`]),\n      ),\n      ...(options.buildSsh ? ['--ssh', options.buildSsh] : []),\n      '--tag',\n      options.tag,\n      ...(options.target ? ['--target', options.target] : []),\n      ...(options.file ? ['--file', options.file] : []),\n      ...(options.networkMode ? ['--network', options.networkMode] : []),\n      ...(options.platform ? ['--platform', options.platform] : []),\n      ...(options.outputs ? options.outputs.map((output) => [`--output=${output}`]) : []),\n      ...(options.cacheFrom\n        ? [\n          ...options.cacheFrom\n            .map((cacheFrom) => ['--cache-from', this.cacheOptionToFlag(cacheFrom)])\n            .flat(),\n        ]\n        : []),\n      ...(options.cacheTo ? ['--cache-to', this.cacheOptionToFlag(options.cacheTo)] : []),\n      ...(options.cacheDisabled ? ['--no-cache'] : []),\n      '.',\n    ];\n    await this.execute(buildCommand, {\n      cwd: options.directory,\n      subprocessOutputDestination: this.subprocessOutputDestination,\n      env: {\n        BUILDX_NO_DEFAULT_ATTESTATIONS: '1', // Docker Build adds provenance attestations by default that confuse cdk-assets\n      },\n    });\n  }\n\n  /**\n   * Get credentials from ECR and run docker login\n   */\n  public async login(ecr: IECRClient) {\n    const credentials = await obtainEcrCredentials(ecr, this.eventEmitter);\n\n    // Use --password-stdin otherwise docker will complain. Loudly.\n    await this.execute(\n      ['login', '--username', credentials.username, '--password-stdin', credentials.endpoint.replace(/^https?:\\/\\/|\\/$/g, '')],\n      {\n        input: credentials.password,\n\n        // Need to ignore otherwise Docker will complain\n        // 'WARNING! Your password will be stored unencrypted'\n        // doesn't really matter since it's a token.\n        subprocessOutputDestination: 'ignore',\n      },\n    );\n  }\n\n  public async tag(sourceTag: string, targetTag: string) {\n    await this.execute(['tag', sourceTag, targetTag]);\n  }\n\n  public async push(options: PushOptions) {\n    await this.execute(['push', options.tag], {\n      subprocessOutputDestination: this.subprocessOutputDestination,\n    });\n  }\n\n  /**\n   * If a CDK Docker Credentials file exists, creates a new Docker config directory.\n   * Sets up `docker-credential-cdk-assets` to be the credential helper for each domain in the CDK config.\n   * All future commands (e.g., `build`, `push`) will use this config.\n   *\n   * See https://docs.docker.com/engine/reference/commandline/login/#credential-helpers for more details on cred helpers.\n   *\n   * @returns true if CDK config was found and configured, false otherwise\n   */\n  public configureCdkCredentials(): boolean {\n    const config = cdkCredentialsConfig();\n    if (!config) {\n      return false;\n    }\n\n    this.configDir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdkDockerConfig'));\n\n    const domains = Object.keys(config.domainCredentials);\n    const credHelpers = domains.reduce((map: Record<string, string>, domain) => {\n      map[domain] = 'cdk-assets'; // Use docker-credential-cdk-assets for this domain\n      return map;\n    }, {});\n    fs.writeFileSync(path.join(this.configDir, 'config.json'), JSON.stringify({ credHelpers }), {\n      encoding: 'utf-8',\n    });\n\n    return true;\n  }\n\n  /**\n   * Removes any configured Docker config directory.\n   * All future commands (e.g., `build`, `push`) will use the default config.\n   *\n   * This is useful after calling `configureCdkCredentials` to reset to default credentials.\n   */\n  public resetAuthPlugins() {\n    this.configDir = undefined;\n  }\n\n  private async execute(args: string[], options: Omit<ShellOptions, 'shellEventPublisher'> = {}) {\n    const configArgs = this.configDir ? ['--config', this.configDir] : [];\n\n    const pathToCdkAssets = path.resolve(__dirname, '..', '..', 'bin');\n\n    const shellEventPublisher = shellEventPublisherFromEventEmitter(this.eventEmitter);\n    try {\n      await shell([getDockerCmd(), ...configArgs, ...args], {\n        ...options,\n        shellEventPublisher: shellEventPublisher,\n        env: {\n          ...process.env,\n          ...options.env,\n          PATH: `${pathToCdkAssets}${path.delimiter}${options.env?.PATH ?? process.env.PATH}`,\n        },\n      });\n    } catch (e: any) {\n      if (e.code === 'ENOENT') {\n        throw new Error(\n          `Failed to find and execute '${getDockerCmd()}' while attempting to build a container asset. Please install '${getDockerCmd()}' and try again. (Or set the 'CDK_DOCKER ' environment variable to choose a different compatible container client.)`,\n        );\n      }\n      throw e;\n    }\n  }\n\n  private cacheOptionToFlag(option: DockerCacheOption): string {\n    let flag = `type=${option.type}`;\n    if (option.params) {\n      flag +=\n        ',' +\n        Object.entries(option.params)\n          .map(([k, v]) => `${k}=${v}`)\n          .join(',');\n    }\n    return flag;\n  }\n}\n\nexport interface DockerFactoryOptions {\n  readonly repoUri: string;\n  readonly ecr: IECRClient;\n  readonly eventEmitter: EventEmitter;\n  readonly subprocessOutputDestination: SubprocessOutputDestination;\n}\n\n/**\n * Helps get appropriately configured Docker instances during the container\n * image publishing process.\n */\nexport class DockerFactory {\n  private enterLoggedInDestinationsCriticalSection = createCriticalSection();\n  private loggedInDestinations = new Set<string>();\n\n  /**\n   * Gets a Docker instance for building images.\n   */\n  public async forBuild(options: DockerFactoryOptions): Promise<Docker> {\n    const docker = new Docker(options.eventEmitter, options.subprocessOutputDestination);\n\n    // Default behavior is to login before build so that the Dockerfile can reference images in the ECR repo\n    // However, if we're in a pipelines environment (for example),\n    // we may have alternative credentials to the default ones to use for the build itself.\n    // If the special config file is present, delay the login to the default credentials until the push.\n    // If the config file is present, we will configure and use those credentials for the build.\n    let cdkDockerCredentialsConfigured = docker.configureCdkCredentials();\n    if (!cdkDockerCredentialsConfigured) {\n      await this.loginOncePerDestination(docker, options);\n    }\n\n    return docker;\n  }\n\n  /**\n   * Gets a Docker instance for pushing images to ECR.\n   */\n  public async forEcrPush(options: DockerFactoryOptions) {\n    const docker = new Docker(options.eventEmitter, options.subprocessOutputDestination);\n    await this.loginOncePerDestination(docker, options);\n    return docker;\n  }\n\n  private async loginOncePerDestination(docker: Docker, options: DockerFactoryOptions) {\n    // Changes: 012345678910.dkr.ecr.us-west-2.amazonaws.com/tagging-test\n    // To this: 012345678910.dkr.ecr.us-west-2.amazonaws.com\n    const repositoryDomain = options.repoUri.split('/')[0];\n\n    // Ensure one-at-a-time access to loggedInDestinations.\n    await this.enterLoggedInDestinationsCriticalSection(async () => {\n      if (this.loggedInDestinations.has(repositoryDomain)) {\n        return;\n      }\n\n      await docker.login(options.ecr);\n      this.loggedInDestinations.add(repositoryDomain);\n    });\n  }\n}\n\nfunction getDockerCmd(): string {\n  return process.env.CDK_DOCKER ?? 'docker';\n}\n\nfunction flatten(x: string[][]) {\n  return Array.prototype.concat([], ...x);\n}\n"]}
@@ -1,3 +0,0 @@
1
- export declare function pathExists(pathName: string): Promise<boolean>;
2
- export declare function emptyDirSync(dir: string): void;
3
- export declare function rmRfSync(dir: string): void;
@@ -1,37 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.pathExists = pathExists;
4
- exports.emptyDirSync = emptyDirSync;
5
- exports.rmRfSync = rmRfSync;
6
- const fs = require("fs");
7
- const path = require("path");
8
- const pfs = fs.promises;
9
- async function pathExists(pathName) {
10
- try {
11
- await pfs.stat(pathName);
12
- return true;
13
- }
14
- catch (e) {
15
- if (e.code !== 'ENOENT') {
16
- throw e;
17
- }
18
- return false;
19
- }
20
- }
21
- function emptyDirSync(dir) {
22
- fs.readdirSync(dir, { withFileTypes: true }).forEach((dirent) => {
23
- const fullPath = path.join(dir, dirent.name);
24
- if (dirent.isDirectory()) {
25
- emptyDirSync(fullPath);
26
- fs.rmdirSync(fullPath);
27
- }
28
- else {
29
- fs.unlinkSync(fullPath);
30
- }
31
- });
32
- }
33
- function rmRfSync(dir) {
34
- emptyDirSync(dir);
35
- fs.rmdirSync(dir);
36
- }
37
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZnMtZXh0cmEuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJmcy1leHRyYS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUtBLGdDQVVDO0FBRUQsb0NBVUM7QUFFRCw0QkFHQztBQWhDRCx5QkFBeUI7QUFDekIsNkJBQTZCO0FBRTdCLE1BQU0sR0FBRyxHQUFHLEVBQUUsQ0FBQyxRQUFRLENBQUM7QUFFakIsS0FBSyxVQUFVLFVBQVUsQ0FBQyxRQUFnQjtJQUMvQyxJQUFJLENBQUM7UUFDSCxNQUFNLEdBQUcsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUM7UUFDekIsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0lBQUMsT0FBTyxDQUFNLEVBQUUsQ0FBQztRQUNoQixJQUFJLENBQUMsQ0FBQyxJQUFJLEtBQUssUUFBUSxFQUFFLENBQUM7WUFDeEIsTUFBTSxDQUFDLENBQUM7UUFDVixDQUFDO1FBQ0QsT0FBTyxLQUFLLENBQUM7SUFDZixDQUFDO0FBQ0gsQ0FBQztBQUVELFNBQWdCLFlBQVksQ0FBQyxHQUFXO0lBQ3RDLEVBQUUsQ0FBQyxXQUFXLENBQUMsR0FBRyxFQUFFLEVBQUUsYUFBYSxFQUFFLElBQUksRUFBRSxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUU7UUFDOUQsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLEVBQUUsTUFBTSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQzdDLElBQUksTUFBTSxDQUFDLFdBQVcsRUFBRSxFQUFFLENBQUM7WUFDekIsWUFBWSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1lBQ3ZCLEVBQUUsQ0FBQyxTQUFTLENBQUMsUUFBUSxDQUFDLENBQUM7UUFDekIsQ0FBQzthQUFNLENBQUM7WUFDTixFQUFFLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBQzFCLENBQUM7SUFDSCxDQUFDLENBQUMsQ0FBQztBQUNMLENBQUM7QUFFRCxTQUFnQixRQUFRLENBQUMsR0FBVztJQUNsQyxZQUFZLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDbEIsRUFBRSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsQ0FBQztBQUNwQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgZnMgZnJvbSAnZnMnO1xuaW1wb3J0ICogYXMgcGF0aCBmcm9tICdwYXRoJztcblxuY29uc3QgcGZzID0gZnMucHJvbWlzZXM7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBwYXRoRXhpc3RzKHBhdGhOYW1lOiBzdHJpbmcpIHtcbiAgdHJ5IHtcbiAgICBhd2FpdCBwZnMuc3RhdChwYXRoTmFtZSk7XG4gICAgcmV0dXJuIHRydWU7XG4gIH0gY2F0Y2ggKGU6IGFueSkge1xuICAgIGlmIChlLmNvZGUgIT09ICdFTk9FTlQnKSB7XG4gICAgICB0aHJvdyBlO1xuICAgIH1cbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGVtcHR5RGlyU3luYyhkaXI6IHN0cmluZykge1xuICBmcy5yZWFkZGlyU3luYyhkaXIsIHsgd2l0aEZpbGVUeXBlczogdHJ1ZSB9KS5mb3JFYWNoKChkaXJlbnQpID0+IHtcbiAgICBjb25zdCBmdWxsUGF0aCA9IHBhdGguam9pbihkaXIsIGRpcmVudC5uYW1lKTtcbiAgICBpZiAoZGlyZW50LmlzRGlyZWN0b3J5KCkpIHtcbiAgICAgIGVtcHR5RGlyU3luYyhmdWxsUGF0aCk7XG4gICAgICBmcy5ybWRpclN5bmMoZnVsbFBhdGgpO1xuICAgIH0gZWxzZSB7XG4gICAgICBmcy51bmxpbmtTeW5jKGZ1bGxQYXRoKTtcbiAgICB9XG4gIH0pO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcm1SZlN5bmMoZGlyOiBzdHJpbmcpIHtcbiAgZW1wdHlEaXJTeW5jKGRpcik7XG4gIGZzLnJtZGlyU3luYyhkaXIpO1xufVxuIl19
@@ -1,3 +0,0 @@
1
- import type { AwsDestination } from '@aws-cdk/cloud-assembly-schema';
2
- import type { ClientOptions } from '../../aws';
3
- export declare function destinationToClientOptions(destination: AwsDestination): ClientOptions;
@@ -1,12 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.destinationToClientOptions = destinationToClientOptions;
4
- function destinationToClientOptions(destination) {
5
- return {
6
- assumeRoleArn: destination.assumeRoleArn,
7
- assumeRoleExternalId: destination.assumeRoleExternalId,
8
- assumeRoleAdditionalOptions: destination.assumeRoleAdditionalOptions,
9
- region: destination.region,
10
- };
11
- }
12
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xpZW50LW9wdGlvbnMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJjbGllbnQtb3B0aW9ucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUdBLGdFQU9DO0FBUEQsU0FBZ0IsMEJBQTBCLENBQUMsV0FBMkI7SUFDcEUsT0FBTztRQUNMLGFBQWEsRUFBRSxXQUFXLENBQUMsYUFBYTtRQUN4QyxvQkFBb0IsRUFBRSxXQUFXLENBQUMsb0JBQW9CO1FBQ3RELDJCQUEyQixFQUFFLFdBQVcsQ0FBQywyQkFBMkI7UUFDcEUsTUFBTSxFQUFFLFdBQVcsQ0FBQyxNQUFNO0tBQzNCLENBQUM7QUFDSixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBBd3NEZXN0aW5hdGlvbiB9IGZyb20gJ0Bhd3MtY2RrL2Nsb3VkLWFzc2VtYmx5LXNjaGVtYSc7XG5pbXBvcnQgdHlwZSB7IENsaWVudE9wdGlvbnMgfSBmcm9tICcuLi8uLi9hd3MnO1xuXG5leHBvcnQgZnVuY3Rpb24gZGVzdGluYXRpb25Ub0NsaWVudE9wdGlvbnMoZGVzdGluYXRpb246IEF3c0Rlc3RpbmF0aW9uKTogQ2xpZW50T3B0aW9ucyB7XG4gIHJldHVybiB7XG4gICAgYXNzdW1lUm9sZUFybjogZGVzdGluYXRpb24uYXNzdW1lUm9sZUFybixcbiAgICBhc3N1bWVSb2xlRXh0ZXJuYWxJZDogZGVzdGluYXRpb24uYXNzdW1lUm9sZUV4dGVybmFsSWQsXG4gICAgYXNzdW1lUm9sZUFkZGl0aW9uYWxPcHRpb25zOiBkZXN0aW5hdGlvbi5hc3N1bWVSb2xlQWRkaXRpb25hbE9wdGlvbnMsXG4gICAgcmVnaW9uOiBkZXN0aW5hdGlvbi5yZWdpb24sXG4gIH07XG59XG4iXX0=
@@ -1,22 +0,0 @@
1
- import type { DockerImageManifestEntry } from '../../asset-manifest';
2
- import type { IAssetHandler, IHandlerHost, IHandlerOptions } from '../asset-handler';
3
- export declare class ContainerImageAssetHandler implements IAssetHandler {
4
- private readonly workDir;
5
- private readonly asset;
6
- private readonly host;
7
- private readonly options;
8
- private init?;
9
- constructor(workDir: string, asset: DockerImageManifestEntry, host: IHandlerHost, options: IHandlerOptions);
10
- build(): Promise<void>;
11
- isPublished(): Promise<boolean>;
12
- publish(): Promise<void>;
13
- private initOnce;
14
- /**
15
- * Check whether the image already exists in the ECR repo
16
- *
17
- * Use the fields from the destination to do the actual check. The imageUri
18
- * should correspond to that, but is only used to print Docker image location
19
- * for user benefit (the format is slightly different).
20
- */
21
- private destinationAlreadyExists;
22
- }
@@ -1,224 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.ContainerImageAssetHandler = void 0;
4
- const path = require("path");
5
- const client_options_1 = require("./client-options");
6
- const progress_1 = require("../../progress");
7
- const placeholders_1 = require("../placeholders");
8
- const shell_1 = require("../shell");
9
- class ContainerImageAssetHandler {
10
- constructor(workDir, asset, host, options) {
11
- this.workDir = workDir;
12
- this.asset = asset;
13
- this.host = host;
14
- this.options = options;
15
- }
16
- async build() {
17
- const initOnce = await this.initOnce();
18
- if (initOnce.destinationAlreadyExists) {
19
- return;
20
- }
21
- if (this.host.aborted) {
22
- return;
23
- }
24
- const dockerForBuilding = await this.host.dockerFactory.forBuild({
25
- repoUri: initOnce.repoUri,
26
- eventEmitter: (m) => this.host.emitMessage(progress_1.EventType.DEBUG, m),
27
- ecr: initOnce.ecr,
28
- subprocessOutputDestination: this.options.subprocessOutputDestination,
29
- });
30
- const builder = new ContainerImageBuilder(dockerForBuilding, this.workDir, this.asset, this.host);
31
- const localTagName = await builder.build();
32
- if (localTagName === undefined || this.host.aborted) {
33
- return;
34
- }
35
- if (this.host.aborted) {
36
- return;
37
- }
38
- await dockerForBuilding.tag(localTagName, initOnce.imageUri);
39
- }
40
- async isPublished() {
41
- try {
42
- const initOnce = await this.initOnce({ quiet: true });
43
- return initOnce.destinationAlreadyExists;
44
- }
45
- catch (e) {
46
- this.host.emitMessage(progress_1.EventType.DEBUG, `${e.message}`);
47
- }
48
- return false;
49
- }
50
- async publish() {
51
- const initOnce = await this.initOnce();
52
- if (initOnce.destinationAlreadyExists) {
53
- return;
54
- }
55
- if (this.host.aborted) {
56
- return;
57
- }
58
- const dockerForPushing = await this.host.dockerFactory.forEcrPush({
59
- repoUri: initOnce.repoUri,
60
- eventEmitter: this.host.emitMessage,
61
- ecr: initOnce.ecr,
62
- subprocessOutputDestination: this.options.subprocessOutputDestination,
63
- });
64
- if (this.host.aborted) {
65
- return;
66
- }
67
- this.host.emitMessage(progress_1.EventType.UPLOAD, `Push ${initOnce.imageUri}`);
68
- await dockerForPushing.push({
69
- tag: initOnce.imageUri,
70
- });
71
- }
72
- async initOnce(options = {}) {
73
- if (this.init) {
74
- return this.init;
75
- }
76
- const destination = await (0, placeholders_1.replaceAwsPlaceholders)(this.asset.destination, this.host.aws);
77
- const ecr = await this.host.aws.ecrClient({
78
- ...(0, client_options_1.destinationToClientOptions)(destination),
79
- quiet: options.quiet,
80
- });
81
- const account = async () => (await this.host.aws.discoverCurrentAccount())?.accountId;
82
- const repoUri = await repositoryUri(ecr, destination.repositoryName);
83
- if (!repoUri) {
84
- throw new Error(`No ECR repository named '${destination.repositoryName}' in account ${await account()}. Is this account bootstrapped?`);
85
- }
86
- const imageUri = `${repoUri}:${destination.imageTag}`;
87
- this.init = {
88
- imageUri,
89
- ecr,
90
- repoUri,
91
- destinationAlreadyExists: await this.destinationAlreadyExists(ecr, destination, imageUri),
92
- };
93
- return this.init;
94
- }
95
- /**
96
- * Check whether the image already exists in the ECR repo
97
- *
98
- * Use the fields from the destination to do the actual check. The imageUri
99
- * should correspond to that, but is only used to print Docker image location
100
- * for user benefit (the format is slightly different).
101
- */
102
- async destinationAlreadyExists(ecr, destination, imageUri) {
103
- this.host.emitMessage(progress_1.EventType.CHECK, `Check ${imageUri}`);
104
- if (await imageExists(ecr, destination.repositoryName, destination.imageTag)) {
105
- this.host.emitMessage(progress_1.EventType.FOUND, `Found ${imageUri}`);
106
- return true;
107
- }
108
- return false;
109
- }
110
- }
111
- exports.ContainerImageAssetHandler = ContainerImageAssetHandler;
112
- class ContainerImageBuilder {
113
- constructor(docker, workDir, asset, host) {
114
- this.docker = docker;
115
- this.workDir = workDir;
116
- this.asset = asset;
117
- this.host = host;
118
- }
119
- async build() {
120
- return this.asset.source.executable
121
- ? this.buildExternalAsset(this.asset.source.executable)
122
- : this.buildDirectoryAsset();
123
- }
124
- /**
125
- * Build a (local) Docker asset from a directory with a Dockerfile
126
- *
127
- * Tags under a deterministic, unique, local identifier wich will skip
128
- * the build if it already exists.
129
- */
130
- async buildDirectoryAsset() {
131
- const localTagName = `cdkasset-${this.asset.id.assetId.toLowerCase()}`;
132
- if (!(await this.isImageCached(localTagName))) {
133
- if (this.host.aborted) {
134
- return undefined;
135
- }
136
- await this.buildImage(localTagName);
137
- }
138
- return localTagName;
139
- }
140
- /**
141
- * Build a (local) Docker asset by running an external command
142
- *
143
- * External command is responsible for deduplicating the build if possible,
144
- * and is expected to return the generated image identifier on stdout.
145
- */
146
- async buildExternalAsset(executable, cwd) {
147
- const assetPath = cwd ?? this.workDir;
148
- this.host.emitMessage(progress_1.EventType.BUILD, `Building Docker image using command '${executable}'`);
149
- if (this.host.aborted) {
150
- return undefined;
151
- }
152
- const shellEventPublisher = (0, progress_1.shellEventPublisherFromEventEmitter)(this.host.emitMessage);
153
- return (await (0, shell_1.shell)(executable, {
154
- cwd: assetPath,
155
- shellEventPublisher,
156
- subprocessOutputDestination: 'ignore',
157
- })).trim();
158
- }
159
- async buildImage(localTagName) {
160
- const source = this.asset.source;
161
- if (!source.directory) {
162
- throw new Error(`'directory' is expected in the DockerImage asset source, got: ${JSON.stringify(source)}`);
163
- }
164
- const fullPath = path.resolve(this.workDir, source.directory);
165
- this.host.emitMessage(progress_1.EventType.BUILD, `Building Docker image at ${fullPath}`);
166
- await this.docker.build({
167
- directory: fullPath,
168
- tag: localTagName,
169
- buildArgs: source.dockerBuildArgs,
170
- buildSecrets: source.dockerBuildSecrets,
171
- buildSsh: source.dockerBuildSsh,
172
- target: source.dockerBuildTarget,
173
- file: source.dockerFile,
174
- networkMode: source.networkMode,
175
- platform: source.platform,
176
- outputs: source.dockerOutputs,
177
- cacheFrom: source.cacheFrom,
178
- cacheTo: source.cacheTo,
179
- cacheDisabled: source.cacheDisabled,
180
- });
181
- }
182
- async isImageCached(localTagName) {
183
- if (await this.docker.exists(localTagName)) {
184
- this.host.emitMessage(progress_1.EventType.CACHED, `Cached ${localTagName}`);
185
- return true;
186
- }
187
- return false;
188
- }
189
- }
190
- async function imageExists(ecr, repositoryName, imageTag) {
191
- try {
192
- await ecr.describeImages({
193
- repositoryName,
194
- imageIds: [{ imageTag }],
195
- });
196
- return true;
197
- }
198
- catch (e) {
199
- if (e.name !== 'ImageNotFoundException') {
200
- throw e;
201
- }
202
- return false;
203
- }
204
- }
205
- /**
206
- * Return the URI for the repository with the given name
207
- *
208
- * Returns undefined if the repository does not exist.
209
- */
210
- async function repositoryUri(ecr, repositoryName) {
211
- try {
212
- const response = await ecr.describeRepositories({
213
- repositoryNames: [repositoryName],
214
- });
215
- return (response.repositories || [])[0]?.repositoryUri;
216
- }
217
- catch (e) {
218
- if (e.name !== 'RepositoryNotFoundException') {
219
- throw e;
220
- }
221
- return undefined;
222
- }
223
- }
224
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"container-images.js","sourceRoot":"","sources":["container-images.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport type { DockerImageDestination } from '@aws-cdk/cloud-assembly-schema';\nimport { destinationToClientOptions } from './client-options';\nimport type { DockerImageManifestEntry } from '../../asset-manifest';\nimport type { IECRClient } from '../../aws';\nimport { EventType, shellEventPublisherFromEventEmitter } from '../../progress';\nimport type { IAssetHandler, IHandlerHost, IHandlerOptions } from '../asset-handler';\nimport type { Docker } from '../docker';\nimport { replaceAwsPlaceholders } from '../placeholders';\nimport { shell } from '../shell';\n\ninterface ContainerImageAssetHandlerInit {\n  readonly ecr: IECRClient;\n  readonly repoUri: string;\n  readonly imageUri: string;\n  readonly destinationAlreadyExists: boolean;\n}\n\nexport class ContainerImageAssetHandler implements IAssetHandler {\n  private init?: ContainerImageAssetHandlerInit;\n\n  constructor(\n    private readonly workDir: string,\n    private readonly asset: DockerImageManifestEntry,\n    private readonly host: IHandlerHost,\n    private readonly options: IHandlerOptions,\n  ) {\n  }\n\n  public async build(): Promise<void> {\n    const initOnce = await this.initOnce();\n\n    if (initOnce.destinationAlreadyExists) {\n      return;\n    }\n    if (this.host.aborted) {\n      return;\n    }\n\n    const dockerForBuilding = await this.host.dockerFactory.forBuild({\n      repoUri: initOnce.repoUri,\n      eventEmitter: (m: string) => this.host.emitMessage(EventType.DEBUG, m),\n      ecr: initOnce.ecr,\n      subprocessOutputDestination: this.options.subprocessOutputDestination,\n    });\n\n    const builder = new ContainerImageBuilder(\n      dockerForBuilding,\n      this.workDir,\n      this.asset,\n      this.host,\n    );\n    const localTagName = await builder.build();\n\n    if (localTagName === undefined || this.host.aborted) {\n      return;\n    }\n    if (this.host.aborted) {\n      return;\n    }\n\n    await dockerForBuilding.tag(localTagName, initOnce.imageUri);\n  }\n\n  public async isPublished(): Promise<boolean> {\n    try {\n      const initOnce = await this.initOnce({ quiet: true });\n      return initOnce.destinationAlreadyExists;\n    } catch (e: any) {\n      this.host.emitMessage(EventType.DEBUG, `${e.message}`);\n    }\n    return false;\n  }\n\n  public async publish(): Promise<void> {\n    const initOnce = await this.initOnce();\n\n    if (initOnce.destinationAlreadyExists) {\n      return;\n    }\n    if (this.host.aborted) {\n      return;\n    }\n\n    const dockerForPushing = await this.host.dockerFactory.forEcrPush({\n      repoUri: initOnce.repoUri,\n      eventEmitter: this.host.emitMessage,\n      ecr: initOnce.ecr,\n      subprocessOutputDestination: this.options.subprocessOutputDestination,\n    });\n\n    if (this.host.aborted) {\n      return;\n    }\n\n    this.host.emitMessage(EventType.UPLOAD, `Push ${initOnce.imageUri}`);\n    await dockerForPushing.push({\n      tag: initOnce.imageUri,\n    });\n  }\n\n  private async initOnce(\n    options: { quiet?: boolean } = {},\n  ): Promise<ContainerImageAssetHandlerInit> {\n    if (this.init) {\n      return this.init;\n    }\n\n    const destination = await replaceAwsPlaceholders(this.asset.destination, this.host.aws);\n    const ecr = await this.host.aws.ecrClient({\n      ...destinationToClientOptions(destination),\n      quiet: options.quiet,\n    });\n    const account = async () => (await this.host.aws.discoverCurrentAccount())?.accountId;\n\n    const repoUri = await repositoryUri(ecr, destination.repositoryName);\n    if (!repoUri) {\n      throw new Error(\n        `No ECR repository named '${destination.repositoryName}' in account ${await account()}. Is this account bootstrapped?`,\n      );\n    }\n\n    const imageUri = `${repoUri}:${destination.imageTag}`;\n\n    this.init = {\n      imageUri,\n      ecr,\n      repoUri,\n      destinationAlreadyExists: await this.destinationAlreadyExists(ecr, destination, imageUri),\n    };\n\n    return this.init;\n  }\n\n  /**\n   * Check whether the image already exists in the ECR repo\n   *\n   * Use the fields from the destination to do the actual check. The imageUri\n   * should correspond to that, but is only used to print Docker image location\n   * for user benefit (the format is slightly different).\n   */\n  private async destinationAlreadyExists(\n    ecr: IECRClient,\n    destination: DockerImageDestination,\n    imageUri: string,\n  ): Promise<boolean> {\n    this.host.emitMessage(EventType.CHECK, `Check ${imageUri}`);\n    if (await imageExists(ecr, destination.repositoryName, destination.imageTag)) {\n      this.host.emitMessage(EventType.FOUND, `Found ${imageUri}`);\n      return true;\n    }\n\n    return false;\n  }\n}\n\nclass ContainerImageBuilder {\n  constructor(\n    private readonly docker: Docker,\n    private readonly workDir: string,\n    private readonly asset: DockerImageManifestEntry,\n    private readonly host: IHandlerHost,\n  ) {\n  }\n\n  async build(): Promise<string | undefined> {\n    return this.asset.source.executable\n      ? this.buildExternalAsset(this.asset.source.executable)\n      : this.buildDirectoryAsset();\n  }\n\n  /**\n   * Build a (local) Docker asset from a directory with a Dockerfile\n   *\n   * Tags under a deterministic, unique, local identifier wich will skip\n   * the build if it already exists.\n   */\n  private async buildDirectoryAsset(): Promise<string | undefined> {\n    const localTagName = `cdkasset-${this.asset.id.assetId.toLowerCase()}`;\n\n    if (!(await this.isImageCached(localTagName))) {\n      if (this.host.aborted) {\n        return undefined;\n      }\n\n      await this.buildImage(localTagName);\n    }\n\n    return localTagName;\n  }\n\n  /**\n   * Build a (local) Docker asset by running an external command\n   *\n   * External command is responsible for deduplicating the build if possible,\n   * and is expected to return the generated image identifier on stdout.\n   */\n  private async buildExternalAsset(\n    executable: string[],\n    cwd?: string,\n  ): Promise<string | undefined> {\n    const assetPath = cwd ?? this.workDir;\n\n    this.host.emitMessage(EventType.BUILD, `Building Docker image using command '${executable}'`);\n    if (this.host.aborted) {\n      return undefined;\n    }\n\n    const shellEventPublisher = shellEventPublisherFromEventEmitter(this.host.emitMessage);\n\n    return (\n      await shell(executable, {\n        cwd: assetPath,\n        shellEventPublisher,\n        subprocessOutputDestination: 'ignore',\n      })\n    ).trim();\n  }\n\n  private async buildImage(localTagName: string): Promise<void> {\n    const source = this.asset.source;\n    if (!source.directory) {\n      throw new Error(\n        `'directory' is expected in the DockerImage asset source, got: ${JSON.stringify(source)}`,\n      );\n    }\n\n    const fullPath = path.resolve(this.workDir, source.directory);\n    this.host.emitMessage(EventType.BUILD, `Building Docker image at ${fullPath}`);\n\n    await this.docker.build({\n      directory: fullPath,\n      tag: localTagName,\n      buildArgs: source.dockerBuildArgs,\n      buildSecrets: source.dockerBuildSecrets,\n      buildSsh: source.dockerBuildSsh,\n      target: source.dockerBuildTarget,\n      file: source.dockerFile,\n      networkMode: source.networkMode,\n      platform: source.platform,\n      outputs: source.dockerOutputs,\n      cacheFrom: source.cacheFrom,\n      cacheTo: source.cacheTo,\n      cacheDisabled: source.cacheDisabled,\n    });\n  }\n\n  private async isImageCached(localTagName: string): Promise<boolean> {\n    if (await this.docker.exists(localTagName)) {\n      this.host.emitMessage(EventType.CACHED, `Cached ${localTagName}`);\n      return true;\n    }\n\n    return false;\n  }\n}\n\nasync function imageExists(ecr: IECRClient, repositoryName: string, imageTag: string) {\n  try {\n    await ecr.describeImages({\n      repositoryName,\n      imageIds: [{ imageTag }],\n    });\n    return true;\n  } catch (e: any) {\n    if (e.name !== 'ImageNotFoundException') {\n      throw e;\n    }\n    return false;\n  }\n}\n\n/**\n * Return the URI for the repository with the given name\n *\n * Returns undefined if the repository does not exist.\n */\nasync function repositoryUri(ecr: IECRClient, repositoryName: string): Promise<string | undefined> {\n  try {\n    const response = await ecr.describeRepositories({\n      repositoryNames: [repositoryName],\n    });\n    return (response.repositories || [])[0]?.repositoryUri;\n  } catch (e: any) {\n    if (e.name !== 'RepositoryNotFoundException') {\n      throw e;\n    }\n    return undefined;\n  }\n}\n"]}
@@ -1,14 +0,0 @@
1
- import type { FileManifestEntry } from '../../asset-manifest';
2
- import type { IAssetHandler, IHandlerHost, PublishOptions } from '../asset-handler';
3
- export declare class FileAssetHandler implements IAssetHandler {
4
- private readonly workDir;
5
- private readonly asset;
6
- private readonly host;
7
- private readonly fileCacheRoot;
8
- constructor(workDir: string, asset: FileManifestEntry, host: IHandlerHost);
9
- build(): Promise<void>;
10
- isPublished(): Promise<boolean>;
11
- publish(options?: PublishOptions): Promise<void>;
12
- private packageFile;
13
- private externalPackageFile;
14
- }