Socket
Socket
Sign inDemoInstall

cdk-assets

Package Overview
Dependencies
Maintainers
1
Versions
571
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

cdk-assets - npm Package Compare versions

Comparing version 1.83.0 to 1.84.0

24

lib/private/handlers/container-images.d.ts

@@ -7,7 +7,29 @@ import { DockerImageManifestEntry } from '../../asset-manifest';

private readonly host;
private readonly localTagName;
private readonly docker;
constructor(workDir: string, asset: DockerImageManifestEntry, host: IHandlerHost);
publish(): Promise<void>;
/**
* Build a (local) Docker asset from a directory with a Dockerfile
*
* Tags under a deterministic, unique, local identifier wich will skip
* the build if it already exists.
*/
private buildDirectoryAsset;
/**
* Build a (local) Docker asset by running an external command
*
* External command is responsible for deduplicating the build if possible,
* and is expected to return the generated image identifier on stdout.
*/
private buildExternalAsset;
/**
* Check whether the image already exists in the ECR repo
*
* Use the fields from the destination to do the actual check. The imageUri
* should correspond to that, but is only used to print Docker image location
* for user benefit (the format is slightly different).
*/
private destinationAlreadyExists;
private buildImage;
private isImageCached;
}

@@ -8,2 +8,3 @@ "use strict";

const placeholders_1 = require("../placeholders");
const shell_1 = require("../shell");
class ContainerImageAssetHandler {

@@ -15,3 +16,2 @@ constructor(workDir, asset, host) {

this.docker = new docker_1.Docker(m => this.host.emitMessage(progress_1.EventType.DEBUG, m));
this.localTagName = `cdkasset-${this.asset.id.assetId.toLowerCase()}`;
}

@@ -27,5 +27,3 @@ async publish() {

const imageUri = `${repoUri}:${destination.imageTag}`;
this.host.emitMessage(progress_1.EventType.CHECK, `Check ${imageUri}`);
if (await imageExists(ecr, destination.repositoryName, destination.imageTag)) {
this.host.emitMessage(progress_1.EventType.FOUND, `Found ${imageUri}`);
if (await this.destinationAlreadyExists(ecr, destination, imageUri)) {
return;

@@ -38,3 +36,8 @@ }

await this.docker.login(ecr);
await this.buildImage();
const localTagName = this.asset.source.executable
? await this.buildExternalAsset(this.asset.source.executable)
: await this.buildDirectoryAsset();
if (localTagName === undefined || this.host.aborted) {
return;
}
this.host.emitMessage(progress_1.EventType.UPLOAD, `Push ${imageUri}`);

@@ -44,11 +47,54 @@ if (this.host.aborted) {

}
await this.docker.tag(this.localTagName, imageUri);
await this.docker.tag(localTagName, imageUri);
await this.docker.push(imageUri);
}
async buildImage() {
if (await this.docker.exists(this.localTagName)) {
this.host.emitMessage(progress_1.EventType.CACHED, `Cached ${this.localTagName}`);
return;
/**
* Build a (local) Docker asset from a directory with a Dockerfile
*
* Tags under a deterministic, unique, local identifier wich will skip
* the build if it already exists.
*/
async buildDirectoryAsset() {
const localTagName = `cdkasset-${this.asset.id.assetId.toLowerCase()}`;
if (!(await this.isImageCached(localTagName))) {
if (this.host.aborted) {
return undefined;
}
await this.buildImage(localTagName);
}
return localTagName;
}
/**
* Build a (local) Docker asset by running an external command
*
* External command is responsible for deduplicating the build if possible,
* and is expected to return the generated image identifier on stdout.
*/
async buildExternalAsset(executable) {
this.host.emitMessage(progress_1.EventType.BUILD, `Building Docker image using command '${executable}'`);
if (this.host.aborted) {
return undefined;
}
return (await shell_1.shell(executable, { quiet: true })).trim();
}
/**
* Check whether the image already exists in the ECR repo
*
* Use the fields from the destination to do the actual check. The imageUri
* should correspond to that, but is only used to print Docker image location
* for user benefit (the format is slightly different).
*/
async destinationAlreadyExists(ecr, destination, imageUri) {
this.host.emitMessage(progress_1.EventType.CHECK, `Check ${imageUri}`);
if (await imageExists(ecr, destination.repositoryName, destination.imageTag)) {
this.host.emitMessage(progress_1.EventType.FOUND, `Found ${imageUri}`);
return true;
}
return false;
}
async buildImage(localTagName) {
const source = this.asset.source;
if (!source.directory) {
throw new Error(`'directory' is expected in the DockerImage asset source, got: ${JSON.stringify(source)}`);
}
const fullPath = path.resolve(this.workDir, source.directory);

@@ -58,3 +104,3 @@ this.host.emitMessage(progress_1.EventType.BUILD, `Building Docker image at ${fullPath}`);

directory: fullPath,
tag: this.localTagName,
tag: localTagName,
buildArgs: source.dockerBuildArgs,

@@ -65,2 +111,9 @@ target: source.dockerBuildTarget,

}
async isImageCached(localTagName) {
if (await this.docker.exists(localTagName)) {
this.host.emitMessage(progress_1.EventType.CACHED, `Cached ${localTagName}`);
return true;
}
return false;
}
}

@@ -98,2 +151,2 @@ exports.ContainerImageAssetHandler = ContainerImageAssetHandler;

}
//# 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 { DockerImageManifestEntry } from '../../asset-manifest';\nimport { EventType } from '../../progress';\nimport { IAssetHandler, IHandlerHost } from '../asset-handler';\nimport { Docker } from '../docker';\nimport { replaceAwsPlaceholders } from '../placeholders';\n\nexport class ContainerImageAssetHandler implements IAssetHandler {\n  private readonly localTagName: string;\n  private readonly docker = new Docker(m => this.host.emitMessage(EventType.DEBUG, m));\n\n  constructor(\n    private readonly workDir: string,\n    private readonly asset: DockerImageManifestEntry,\n    private readonly host: IHandlerHost) {\n\n    this.localTagName = `cdkasset-${this.asset.id.assetId.toLowerCase()}`;\n  }\n\n  public async publish(): Promise<void> {\n    const destination = await replaceAwsPlaceholders(this.asset.destination, this.host.aws);\n\n    const ecr = await this.host.aws.ecrClient(destination);\n\n    const account = (await this.host.aws.discoverCurrentAccount()).accountId;\n\n    const repoUri = await repositoryUri(ecr, destination.repositoryName);\n    if (!repoUri) {\n      throw new Error(`No ECR repository named '${destination.repositoryName}' in account ${account}. Is this account bootstrapped?`);\n    }\n\n    const imageUri = `${repoUri}:${destination.imageTag}`;\n\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;\n    }\n\n    if (this.host.aborted) { return; }\n\n    // Login before build so that the Dockerfile can reference images in the ECR repo\n    await this.docker.login(ecr);\n    await this.buildImage();\n\n    this.host.emitMessage(EventType.UPLOAD, `Push ${imageUri}`);\n    if (this.host.aborted) { return; }\n    await this.docker.tag(this.localTagName, imageUri);\n    await this.docker.push(imageUri);\n  }\n\n  private async buildImage(): Promise<void> {\n    if (await this.docker.exists(this.localTagName)) {\n      this.host.emitMessage(EventType.CACHED, `Cached ${this.localTagName}`);\n      return;\n    }\n\n    const source = this.asset.source;\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: this.localTagName,\n      buildArgs: source.dockerBuildArgs,\n      target: source.dockerBuildTarget,\n      file: source.dockerFile,\n    });\n  }\n}\n\nasync function imageExists(ecr: AWS.ECR, repositoryName: string, imageTag: string) {\n  try {\n    await ecr.describeImages({ repositoryName, imageIds: [{ imageTag }] }).promise();\n    return true;\n  } catch (e) {\n    if (e.code !== 'ImageNotFoundException') { throw e; }\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: AWS.ECR, repositoryName: string): Promise<string | undefined> {\n  try {\n    const response = await ecr.describeRepositories({ repositoryNames: [repositoryName] }).promise();\n    return (response.repositories || [])[0]?.repositoryUri;\n  } catch (e) {\n    if (e.code !== 'RepositoryNotFoundException') { throw e; }\n    return undefined;\n  }\n}"]}
//# 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 { DockerImageDestination } from '@aws-cdk/cloud-assembly-schema';\nimport { DockerImageManifestEntry } from '../../asset-manifest';\nimport { EventType } from '../../progress';\nimport { IAssetHandler, IHandlerHost } from '../asset-handler';\nimport { Docker } from '../docker';\nimport { replaceAwsPlaceholders } from '../placeholders';\nimport { shell } from '../shell';\n\nexport class ContainerImageAssetHandler implements IAssetHandler {\n  private readonly docker = new Docker(m => this.host.emitMessage(EventType.DEBUG, m));\n\n  constructor(\n    private readonly workDir: string,\n    private readonly asset: DockerImageManifestEntry,\n    private readonly host: IHandlerHost) {\n  }\n\n  public async publish(): Promise<void> {\n    const destination = await replaceAwsPlaceholders(this.asset.destination, this.host.aws);\n    const ecr = await this.host.aws.ecrClient(destination);\n    const account = (await this.host.aws.discoverCurrentAccount()).accountId;\n    const repoUri = await repositoryUri(ecr, destination.repositoryName);\n\n    if (!repoUri) {\n      throw new Error(`No ECR repository named '${destination.repositoryName}' in account ${account}. Is this account bootstrapped?`);\n    }\n\n    const imageUri = `${repoUri}:${destination.imageTag}`;\n\n    if (await this.destinationAlreadyExists(ecr, destination, imageUri)) { return; }\n    if (this.host.aborted) { return; }\n\n    // Login before build so that the Dockerfile can reference images in the ECR repo\n    await this.docker.login(ecr);\n\n    const localTagName = this.asset.source.executable\n      ? await this.buildExternalAsset(this.asset.source.executable)\n      : await this.buildDirectoryAsset();\n\n    if (localTagName === undefined || this.host.aborted) {\n      return;\n    }\n\n    this.host.emitMessage(EventType.UPLOAD, `Push ${imageUri}`);\n    if (this.host.aborted) { return; }\n    await this.docker.tag(localTagName, imageUri);\n    await this.docker.push(imageUri);\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) { return undefined; }\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(executable: string[]): Promise<string | undefined> {\n    this.host.emitMessage(EventType.BUILD, `Building Docker image using command '${executable}'`);\n    if (this.host.aborted) { return undefined; }\n\n    return (await shell(executable, { quiet: true })).trim();\n  }\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(ecr: AWS.ECR, destination: DockerImageDestination, imageUri: string): 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  private async buildImage(localTagName: string): Promise<void> {\n    const source = this.asset.source;\n    if (!source.directory) {\n      throw new Error(`'directory' is expected in the DockerImage asset source, got: ${JSON.stringify(source)}`);\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      target: source.dockerBuildTarget,\n      file: source.dockerFile,\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: AWS.ECR, repositoryName: string, imageTag: string) {\n  try {\n    await ecr.describeImages({ repositoryName, imageIds: [{ imageTag }] }).promise();\n    return true;\n  } catch (e) {\n    if (e.code !== 'ImageNotFoundException') { throw e; }\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: AWS.ECR, repositoryName: string): Promise<string | undefined> {\n  try {\n    const response = await ecr.describeRepositories({ repositoryNames: [repositoryName] }).promise();\n    return (response.repositories || [])[0]?.repositoryUri;\n  } catch (e) {\n    if (e.code !== 'RepositoryNotFoundException') { throw e; }\n    return undefined;\n  }\n}\n"]}

@@ -11,2 +11,3 @@ import { FileManifestEntry } from '../../asset-manifest';

private packageFile;
private externalPackageFile;
}

43

lib/private/handlers/files.js

@@ -11,2 +11,3 @@ "use strict";

const placeholders_1 = require("../placeholders");
const shell_1 = require("../shell");
class FileAssetHandler {

@@ -42,4 +43,4 @@ constructor(workDir, asset, host) {

}
const publishFile = await this.packageFile();
const contentType = this.asset.source.packaging === cloud_assembly_schema_1.FileAssetPackaging.ZIP_DIRECTORY ? 'application/zip' : undefined;
const publishFile = this.asset.source.executable ?
await this.externalPackageFile(this.asset.source.executable) : await this.packageFile(this.asset.source);
this.host.emitMessage(progress_1.EventType.UPLOAD, `Upload ${s3Url}`);

@@ -49,24 +50,34 @@ await s3.upload({

Key: destination.objectKey,
Body: fs_1.createReadStream(publishFile),
ContentType: contentType,
Body: fs_1.createReadStream(publishFile.packagedPath),
ContentType: publishFile.contentType,
}).promise();
}
async packageFile() {
const source = this.asset.source;
const fullPath = path.resolve(this.workDir, this.asset.source.path);
async packageFile(source) {
if (!source.path) {
throw new Error(`'path' is expected in the File asset source, got: ${JSON.stringify(source)}`);
}
const fullPath = path.resolve(this.workDir, source.path);
if (source.packaging === cloud_assembly_schema_1.FileAssetPackaging.ZIP_DIRECTORY) {
const contentType = 'application/zip';
await fs_1.promises.mkdir(this.fileCacheRoot, { recursive: true });
const ret = path.join(this.fileCacheRoot, `${this.asset.id.assetId}.zip`);
if (await fs_extra_1.pathExists(ret)) {
this.host.emitMessage(progress_1.EventType.CACHED, `From cache ${ret}`);
return ret;
const packagedPath = path.join(this.fileCacheRoot, `${this.asset.id.assetId}.zip`);
if (await fs_extra_1.pathExists(packagedPath)) {
this.host.emitMessage(progress_1.EventType.CACHED, `From cache ${path}`);
return { packagedPath, contentType };
}
this.host.emitMessage(progress_1.EventType.BUILD, `Zip ${fullPath} -> ${ret}`);
await archive_1.zipDirectory(fullPath, ret);
return ret;
this.host.emitMessage(progress_1.EventType.BUILD, `Zip ${fullPath} -> ${path}`);
await archive_1.zipDirectory(fullPath, packagedPath);
return { packagedPath, contentType };
}
else {
return fullPath;
return { packagedPath: fullPath };
}
}
async externalPackageFile(executable) {
this.host.emitMessage(progress_1.EventType.BUILD, `Building asset source using command: '${executable}'`);
return {
packagedPath: (await shell_1.shell(executable, { quiet: true })).trim(),
contentType: 'application/zip',
};
}
}

@@ -110,2 +121,2 @@ exports.FileAssetHandler = FileAssetHandler;

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"files.js","sourceRoot":"","sources":["files.ts"],"names":[],"mappings":"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","sourcesContent":["import { createReadStream, promises as fs } from 'fs';\nimport * as path from 'path';\nimport { FileAssetPackaging } from '@aws-cdk/cloud-assembly-schema';\nimport { FileManifestEntry } from '../../asset-manifest';\nimport { EventType } from '../../progress';\nimport { zipDirectory } from '../archive';\nimport { IAssetHandler, IHandlerHost } from '../asset-handler';\nimport { pathExists } from '../fs-extra';\nimport { replaceAwsPlaceholders } from '../placeholders';\n\nexport class FileAssetHandler implements IAssetHandler {\n  private readonly fileCacheRoot: string;\n\n  constructor(\n    private readonly workDir: string,\n    private readonly asset: FileManifestEntry,\n    private readonly host: IHandlerHost) {\n    this.fileCacheRoot = path.join(workDir, '.cache');\n  }\n\n  public async publish(): Promise<void> {\n    const destination = await replaceAwsPlaceholders(this.asset.destination, this.host.aws);\n    const s3Url = `s3://${destination.bucketName}/${destination.objectKey}`;\n\n    const s3 = await this.host.aws.s3Client(destination);\n    this.host.emitMessage(EventType.CHECK, `Check ${s3Url}`);\n\n    // A thunk for describing the current account. Used when we need to format an error\n    // message, not in the success case.\n    const account = async () => (await this.host.aws.discoverCurrentAccount())?.accountId;\n    switch (await bucketOwnership(s3, destination.bucketName)) {\n      case BucketOwnership.MINE:\n        break;\n      case BucketOwnership.DOES_NOT_EXIST:\n        throw new Error(`No bucket named '${destination.bucketName}'. Is account ${await account()} bootstrapped?`);\n      case BucketOwnership.SOMEONE_ELSES_OR_NO_ACCESS:\n        throw new Error(`Bucket named '${destination.bucketName}' exists, but not in account ${await account()}. Wrong account?`);\n    }\n\n    if (await objectExists(s3, destination.bucketName, destination.objectKey)) {\n      this.host.emitMessage(EventType.FOUND, `Found ${s3Url}`);\n      return;\n    }\n\n    if (this.host.aborted) { return; }\n    const publishFile = await this.packageFile();\n    const contentType = this.asset.source.packaging === FileAssetPackaging.ZIP_DIRECTORY ? 'application/zip' : undefined;\n\n    this.host.emitMessage(EventType.UPLOAD, `Upload ${s3Url}`);\n    await s3.upload({\n      Bucket: destination.bucketName,\n      Key: destination.objectKey,\n      Body: createReadStream(publishFile),\n      ContentType: contentType,\n    }).promise();\n  }\n\n  private async packageFile(): Promise<string> {\n    const source = this.asset.source;\n    const fullPath = path.resolve(this.workDir, this.asset.source.path);\n\n    if (source.packaging === FileAssetPackaging.ZIP_DIRECTORY) {\n      await fs.mkdir(this.fileCacheRoot, { recursive: true });\n      const ret = path.join(this.fileCacheRoot, `${this.asset.id.assetId}.zip`);\n\n      if (await pathExists(ret)) {\n        this.host.emitMessage(EventType.CACHED, `From cache ${ret}`);\n        return ret;\n      }\n\n      this.host.emitMessage(EventType.BUILD, `Zip ${fullPath} -> ${ret}`);\n      await zipDirectory(fullPath, ret);\n      return ret;\n    } else {\n      return fullPath;\n    }\n  }\n}\n\nenum BucketOwnership {\n  DOES_NOT_EXIST,\n  MINE,\n  SOMEONE_ELSES_OR_NO_ACCESS\n}\n\nasync function bucketOwnership(s3: AWS.S3, bucket: string): Promise<BucketOwnership> {\n  try {\n    await s3.getBucketLocation({ Bucket: bucket }).promise();\n    return BucketOwnership.MINE;\n  } catch (e) {\n    if (e.code === 'NoSuchBucket') { return BucketOwnership.DOES_NOT_EXIST; }\n    if (['AccessDenied', 'AllAccessDisabled'].includes(e.code)) { return BucketOwnership.SOMEONE_ELSES_OR_NO_ACCESS; }\n    throw e;\n  }\n}\n\nasync function objectExists(s3: AWS.S3, bucket: string, key: string) {\n  /*\n   * The object existence check here refrains from using the `headObject` operation because this\n   * would create a negative cache entry, making GET-after-PUT eventually consistent. This has been\n   * observed to result in CloudFormation issuing \"ValidationError: S3 error: Access Denied\", for\n   * example in https://github.com/aws/aws-cdk/issues/6430.\n   *\n   * To prevent this, we are instead using the listObjectsV2 call, using the looked up key as the\n   * prefix, and limiting results to 1. Since the list operation returns keys ordered by binary\n   * UTF-8 representation, the key we are looking for is guaranteed to always be the first match\n   * returned if it exists.\n   */\n  const response = await s3.listObjectsV2({ Bucket: bucket, Prefix: key, MaxKeys: 1 }).promise();\n  return response.Contents != null && response.Contents.some(object => object.Key === key);\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"files.js","sourceRoot":"","sources":["files.ts"],"names":[],"mappings":"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","sourcesContent":["import { createReadStream, promises as fs } from 'fs';\nimport * as path from 'path';\nimport { FileAssetPackaging, FileSource } from '@aws-cdk/cloud-assembly-schema';\nimport { FileManifestEntry } from '../../asset-manifest';\nimport { EventType } from '../../progress';\nimport { zipDirectory } from '../archive';\nimport { IAssetHandler, IHandlerHost } from '../asset-handler';\nimport { pathExists } from '../fs-extra';\nimport { replaceAwsPlaceholders } from '../placeholders';\nimport { shell } from '../shell';\n\nexport class FileAssetHandler implements IAssetHandler {\n  private readonly fileCacheRoot: string;\n\n  constructor(\n    private readonly workDir: string,\n    private readonly asset: FileManifestEntry,\n    private readonly host: IHandlerHost) {\n    this.fileCacheRoot = path.join(workDir, '.cache');\n  }\n\n  public async publish(): Promise<void> {\n    const destination = await replaceAwsPlaceholders(this.asset.destination, this.host.aws);\n    const s3Url = `s3://${destination.bucketName}/${destination.objectKey}`;\n\n    const s3 = await this.host.aws.s3Client(destination);\n    this.host.emitMessage(EventType.CHECK, `Check ${s3Url}`);\n\n    // A thunk for describing the current account. Used when we need to format an error\n    // message, not in the success case.\n    const account = async () => (await this.host.aws.discoverCurrentAccount())?.accountId;\n    switch (await bucketOwnership(s3, destination.bucketName)) {\n      case BucketOwnership.MINE:\n        break;\n      case BucketOwnership.DOES_NOT_EXIST:\n        throw new Error(`No bucket named '${destination.bucketName}'. Is account ${await account()} bootstrapped?`);\n      case BucketOwnership.SOMEONE_ELSES_OR_NO_ACCESS:\n        throw new Error(`Bucket named '${destination.bucketName}' exists, but not in account ${await account()}. Wrong account?`);\n    }\n\n    if (await objectExists(s3, destination.bucketName, destination.objectKey)) {\n      this.host.emitMessage(EventType.FOUND, `Found ${s3Url}`);\n      return;\n    }\n\n    if (this.host.aborted) { return; }\n    const publishFile = this.asset.source.executable ?\n      await this.externalPackageFile(this.asset.source.executable) : await this.packageFile(this.asset.source);\n\n    this.host.emitMessage(EventType.UPLOAD, `Upload ${s3Url}`);\n    await s3.upload({\n      Bucket: destination.bucketName,\n      Key: destination.objectKey,\n      Body: createReadStream(publishFile.packagedPath),\n      ContentType: publishFile.contentType,\n    }).promise();\n  }\n\n  private async packageFile(source: FileSource): Promise<PackagedFileAsset> {\n    if (!source.path) {\n      throw new Error(`'path' is expected in the File asset source, got: ${JSON.stringify(source)}`);\n    }\n\n    const fullPath = path.resolve(this.workDir, source.path);\n\n    if (source.packaging === FileAssetPackaging.ZIP_DIRECTORY) {\n      const contentType = 'application/zip';\n\n      await fs.mkdir(this.fileCacheRoot, { recursive: true });\n      const packagedPath = path.join(this.fileCacheRoot, `${this.asset.id.assetId}.zip`);\n\n      if (await pathExists(packagedPath)) {\n        this.host.emitMessage(EventType.CACHED, `From cache ${path}`);\n        return { packagedPath, contentType };\n      }\n\n      this.host.emitMessage(EventType.BUILD, `Zip ${fullPath} -> ${path}`);\n      await zipDirectory(fullPath, packagedPath);\n      return { packagedPath, contentType };\n    } else {\n      return { packagedPath: fullPath };\n    }\n  }\n\n  private async externalPackageFile(executable: string[]): Promise<PackagedFileAsset> {\n    this.host.emitMessage(EventType.BUILD, `Building asset source using command: '${executable}'`);\n\n    return {\n      packagedPath: (await shell(executable, { quiet: true })).trim(),\n      contentType: 'application/zip',\n    };\n  }\n}\n\nenum BucketOwnership {\n  DOES_NOT_EXIST,\n  MINE,\n  SOMEONE_ELSES_OR_NO_ACCESS\n}\n\nasync function bucketOwnership(s3: AWS.S3, bucket: string): Promise<BucketOwnership> {\n  try {\n    await s3.getBucketLocation({ Bucket: bucket }).promise();\n    return BucketOwnership.MINE;\n  } catch (e) {\n    if (e.code === 'NoSuchBucket') { return BucketOwnership.DOES_NOT_EXIST; }\n    if (['AccessDenied', 'AllAccessDisabled'].includes(e.code)) { return BucketOwnership.SOMEONE_ELSES_OR_NO_ACCESS; }\n    throw e;\n  }\n}\n\nasync function objectExists(s3: AWS.S3, bucket: string, key: string) {\n  /*\n   * The object existence check here refrains from using the `headObject` operation because this\n   * would create a negative cache entry, making GET-after-PUT eventually consistent. This has been\n   * observed to result in CloudFormation issuing \"ValidationError: S3 error: Access Denied\", for\n   * example in https://github.com/aws/aws-cdk/issues/6430.\n   *\n   * To prevent this, we are instead using the listObjectsV2 call, using the looked up key as the\n   * prefix, and limiting results to 1. Since the list operation returns keys ordered by binary\n   * UTF-8 representation, the key we are looking for is guaranteed to always be the first match\n   * returned if it exists.\n   */\n  const response = await s3.listObjectsV2({ Bucket: bucket, Prefix: key, MaxKeys: 1 }).promise();\n  return response.Contents != null && response.Contents.some(object => object.Key === key);\n}\n\n\n/**\n * A packaged asset which can be uploaded (either a single file or directory)\n */\ninterface PackagedFileAsset {\n  /**\n   * Path of the file or directory\n   */\n  readonly packagedPath: string;\n\n  /**\n   * Content type to be added in the S3 upload action\n   *\n   * @default - No content type\n   */\n  readonly contentType?: string;\n}\n"]}

@@ -17,2 +17,2 @@ "use strict";

exports.makeAssetHandler = makeAssetHandler;
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx5REFBa0g7QUFFbEgseURBQWdFO0FBQ2hFLG1DQUEyQztBQUUzQyxTQUFnQixnQkFBZ0IsQ0FBQyxRQUF1QixFQUFFLEtBQXFCLEVBQUUsSUFBa0I7SUFDakcsSUFBSSxLQUFLLFlBQVksa0NBQWlCLEVBQUU7UUFDdEMsT0FBTyxJQUFJLHdCQUFnQixDQUFDLFFBQVEsQ0FBQyxTQUFTLEVBQUUsS0FBSyxFQUFFLElBQUksQ0FBQyxDQUFDO0tBQzlEO0lBQ0QsSUFBSSxLQUFLLFlBQVkseUNBQXdCLEVBQUU7UUFDN0MsT0FBTyxJQUFJLDZDQUEwQixDQUFDLFFBQVEsQ0FBQyxTQUFTLEVBQUUsS0FBSyxFQUFFLElBQUksQ0FBQyxDQUFDO0tBQ3hFO0lBRUQsTUFBTSxJQUFJLEtBQUssQ0FBQyw2QkFBNkIsS0FBSyxHQUFHLENBQUMsQ0FBQztBQUN6RCxDQUFDO0FBVEQsNENBU0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBc3NldE1hbmlmZXN0LCBEb2NrZXJJbWFnZU1hbmlmZXN0RW50cnksIEZpbGVNYW5pZmVzdEVudHJ5LCBJTWFuaWZlc3RFbnRyeSB9IGZyb20gJy4uLy4uL2Fzc2V0LW1hbmlmZXN0JztcbmltcG9ydCB7IElBc3NldEhhbmRsZXIsIElIYW5kbGVySG9zdCB9IGZyb20gJy4uL2Fzc2V0LWhhbmRsZXInO1xuaW1wb3J0IHsgQ29udGFpbmVySW1hZ2VBc3NldEhhbmRsZXIgfSBmcm9tICcuL2NvbnRhaW5lci1pbWFnZXMnO1xuaW1wb3J0IHsgRmlsZUFzc2V0SGFuZGxlciB9IGZyb20gJy4vZmlsZXMnO1xuXG5leHBvcnQgZnVuY3Rpb24gbWFrZUFzc2V0SGFuZGxlcihtYW5pZmVzdDogQXNzZXRNYW5pZmVzdCwgYXNzZXQ6IElNYW5pZmVzdEVudHJ5LCBob3N0OiBJSGFuZGxlckhvc3QpOiBJQXNzZXRIYW5kbGVyIHtcbiAgaWYgKGFzc2V0IGluc3RhbmNlb2YgRmlsZU1hbmlmZXN0RW50cnkpIHtcbiAgICByZXR1cm4gbmV3IEZpbGVBc3NldEhhbmRsZXIobWFuaWZlc3QuZGlyZWN0b3J5LCBhc3NldCwgaG9zdCk7XG4gIH1cbiAgaWYgKGFzc2V0IGluc3RhbmNlb2YgRG9ja2VySW1hZ2VNYW5pZmVzdEVudHJ5KSB7XG4gICAgcmV0dXJuIG5ldyBDb250YWluZXJJbWFnZUFzc2V0SGFuZGxlcihtYW5pZmVzdC5kaXJlY3RvcnksIGFzc2V0LCBob3N0KTtcbiAgfVxuXG4gIHRocm93IG5ldyBFcnJvcihgVW5yZWNvZ25pemVkIGFzc2V0IHR5cGU6ICcke2Fzc2V0fSdgKTtcbn0iXX0=
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx5REFBa0g7QUFFbEgseURBQWdFO0FBQ2hFLG1DQUEyQztBQUUzQyxTQUFnQixnQkFBZ0IsQ0FBQyxRQUF1QixFQUFFLEtBQXFCLEVBQUUsSUFBa0I7SUFDakcsSUFBSSxLQUFLLFlBQVksa0NBQWlCLEVBQUU7UUFDdEMsT0FBTyxJQUFJLHdCQUFnQixDQUFDLFFBQVEsQ0FBQyxTQUFTLEVBQUUsS0FBSyxFQUFFLElBQUksQ0FBQyxDQUFDO0tBQzlEO0lBQ0QsSUFBSSxLQUFLLFlBQVkseUNBQXdCLEVBQUU7UUFDN0MsT0FBTyxJQUFJLDZDQUEwQixDQUFDLFFBQVEsQ0FBQyxTQUFTLEVBQUUsS0FBSyxFQUFFLElBQUksQ0FBQyxDQUFDO0tBQ3hFO0lBRUQsTUFBTSxJQUFJLEtBQUssQ0FBQyw2QkFBNkIsS0FBSyxHQUFHLENBQUMsQ0FBQztBQUN6RCxDQUFDO0FBVEQsNENBU0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBc3NldE1hbmlmZXN0LCBEb2NrZXJJbWFnZU1hbmlmZXN0RW50cnksIEZpbGVNYW5pZmVzdEVudHJ5LCBJTWFuaWZlc3RFbnRyeSB9IGZyb20gJy4uLy4uL2Fzc2V0LW1hbmlmZXN0JztcbmltcG9ydCB7IElBc3NldEhhbmRsZXIsIElIYW5kbGVySG9zdCB9IGZyb20gJy4uL2Fzc2V0LWhhbmRsZXInO1xuaW1wb3J0IHsgQ29udGFpbmVySW1hZ2VBc3NldEhhbmRsZXIgfSBmcm9tICcuL2NvbnRhaW5lci1pbWFnZXMnO1xuaW1wb3J0IHsgRmlsZUFzc2V0SGFuZGxlciB9IGZyb20gJy4vZmlsZXMnO1xuXG5leHBvcnQgZnVuY3Rpb24gbWFrZUFzc2V0SGFuZGxlcihtYW5pZmVzdDogQXNzZXRNYW5pZmVzdCwgYXNzZXQ6IElNYW5pZmVzdEVudHJ5LCBob3N0OiBJSGFuZGxlckhvc3QpOiBJQXNzZXRIYW5kbGVyIHtcbiAgaWYgKGFzc2V0IGluc3RhbmNlb2YgRmlsZU1hbmlmZXN0RW50cnkpIHtcbiAgICByZXR1cm4gbmV3IEZpbGVBc3NldEhhbmRsZXIobWFuaWZlc3QuZGlyZWN0b3J5LCBhc3NldCwgaG9zdCk7XG4gIH1cbiAgaWYgKGFzc2V0IGluc3RhbmNlb2YgRG9ja2VySW1hZ2VNYW5pZmVzdEVudHJ5KSB7XG4gICAgcmV0dXJuIG5ldyBDb250YWluZXJJbWFnZUFzc2V0SGFuZGxlcihtYW5pZmVzdC5kaXJlY3RvcnksIGFzc2V0LCBob3N0KTtcbiAgfVxuXG4gIHRocm93IG5ldyBFcnJvcihgVW5yZWNvZ25pemVkIGFzc2V0IHR5cGU6ICcke2Fzc2V0fSdgKTtcbn1cbiJdfQ==
{
"name": "cdk-assets",
"version": "1.83.0",
"version": "1.84.0",
"lockfileVersion": 1,

@@ -8,3 +8,3 @@ "requires": true,

"@aws-cdk/cloud-assembly-schema": {
"version": "1.83.0",
"version": "1.84.0",
"requires": {

@@ -16,5 +16,5 @@ "jsonschema": "^1.4.0",

"@aws-cdk/cx-api": {
"version": "1.83.0",
"version": "1.84.0",
"requires": {
"@aws-cdk/cloud-assembly-schema": "1.83.0",
"@aws-cdk/cloud-assembly-schema": "1.84.0",
"semver": "^7.3.2"

@@ -24,5 +24,5 @@ }

"archiver": {
"version": "5.1.0",
"integrity": "sha512-iKuQUP1nuKzBC2PFlGet5twENzCfyODmvkxwDV0cEFXavwcLrIW5ssTuHi9dyTPvpWr6Faweo2eQaQiLIwyXTA==",
"resolved": "https://registry.yarnpkg.com/archiver/-/archiver-5.1.0.tgz#05b0f6f7836f3e6356a0532763d2bb91017a7e37",
"version": "5.2.0",
"integrity": "sha512-QEAKlgQuAtUxKeZB9w5/ggKXh21bZS+dzzuQ0RPBC20qtDCbTyzqmisoeJP46MP39fg4B4IcyvR+yeyEBdblsQ==",
"resolved": "https://registry.yarnpkg.com/archiver/-/archiver-5.2.0.tgz#25aa1b3d9febf7aec5b0f296e77e69960c26db94",
"requires": {

@@ -64,5 +64,5 @@ "archiver-utils": "^2.1.0",

"aws-sdk": {
"version": "2.822.0",
"integrity": "sha512-1a2L4e9ICIZLsY9mh0EFuAQYiDRFiLHLjicIa6Q4aDmN31cLqdQKCnYoWwm9yi0RHI4fnYQiQqSBF/f1Apfe9A==",
"resolved": "https://registry.yarnpkg.com/aws-sdk/-/aws-sdk-2.822.0.tgz#63a13b0124efe2a9a9d2c87eaa39ca1ab63d3309",
"version": "2.824.0",
"integrity": "sha512-9KNRQBkIMPn+6DWb4gR+RzqTMNyGLEwOgXbE4dDehOIAflfLnv3IFwLnzrhxJnleB4guYrILIsBroJFBzjiekg==",
"resolved": "https://registry.yarnpkg.com/aws-sdk/-/aws-sdk-2.824.0.tgz#a67747d4d0b53d09c6c121e93f44d8f6e76fc44b",
"requires": {

@@ -69,0 +69,0 @@ "buffer": "4.9.2",

{
"name": "cdk-assets",
"description": "CDK Asset Publishing Tool",
"version": "1.83.0",
"version": "1.84.0",
"main": "lib/index.js",

@@ -40,13 +40,13 @@ "types": "lib/index.d.ts",

"@types/yargs": "^15.0.10",
"cdk-build-tools": "1.83.0",
"cdk-build-tools": "1.84.0",
"jest": "^26.6.3",
"jszip": "^3.5.0",
"mock-fs": "^4.13.0",
"pkglint": "1.83.0"
"pkglint": "1.84.0"
},
"dependencies": {
"@aws-cdk/cloud-assembly-schema": "1.83.0",
"@aws-cdk/cx-api": "1.83.0",
"archiver": "^5.1.0",
"aws-sdk": "^2.822.0",
"@aws-cdk/cloud-assembly-schema": "1.84.0",
"@aws-cdk/cx-api": "1.84.0",
"archiver": "^5.2.0",
"aws-sdk": "^2.824.0",
"glob": "^7.1.6",

@@ -53,0 +53,0 @@ "yargs": "^16.2.0"

@@ -31,2 +31,3 @@ # cdk-assets

* Docker Images, uploaded to ECR
* Files, archives, and Docker images built by external utilities

@@ -45,2 +46,9 @@ S3 buckets and ECR repositories to upload to are expected to exist already.

location.
For assets build by external utilities, the contract is such that cdk-assets
expects the utility to manage dedupe detection as well as path/image tag generation.
This means that cdk-assets will call the external utility every time generation
is warranted, and it is up to the utility to a) determine whether to do a
full rebuild; and b) to return only one thing on stdout: the path to the file/archive
asset, or the name of the local Docker image.

@@ -87,2 +95,15 @@ ## Usage

},
"3dfe2b80b050e7e4e168f84feff678d4": {
"source": {
"executable": ["myzip"]
},
"destinations": {
"us-east-1": {
"region": "us-east-1",
"assumeRoleArn": "arn:aws:iam::12345789012:role/my-account",
"bucketName": "MySpecialBucket",
"objectKey": "3dfe2b80b050e7e4e168f84feff678d4.zip"
}
}
},
},

@@ -103,2 +124,16 @@ "dockerImages": {

}
},
"d92753c58a86f7b8c68a4591c4f9cf28": {
"source": {
"executable": ["mytool", "package", "dockerdir"],
},
"destinations": {
"us-east-1": {
"region": "us-east-1",
"assumeRoleArn": "arn:aws:iam::12345789012:role/my-account",
"repositoryName": "MyRepository2",
"imageTag": "d92753c58a86f7b8c68a4591c4f9cf28",
"imageUri": "123456789987.dkr.ecr.us-east-1.amazonaws.com/MyRepository2:1234567891b48783c58a86f7b8c68a4591c4f9be31",
}
}
}

@@ -105,0 +140,0 @@ }

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc