UNPKG

23.6 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3exports.AssetManifestBuilder = void 0;
4const fs = require("fs");
5const path = require("path");
6const cxschema = require("@aws-cdk/cloud-assembly-schema");
7const assets_1 = require("../assets");
8const cfn_fn_1 = require("../cfn-fn");
9const _shared_1 = require("./_shared");
10/**
11 * Build an manifest from assets added to a stack synthesizer
12 */
13class AssetManifestBuilder {
14 constructor() {
15 this.files = {};
16 this.dockerImages = {};
17 }
18 addFileAssetDefault(asset, stack, bucketName, bucketPrefix, role) {
19 validateFileAssetSource(asset);
20 const extension = asset.fileName != undefined ? path.extname(asset.fileName) : '';
21 const objectKey = bucketPrefix +
22 asset.sourceHash +
23 (asset.packaging === assets_1.FileAssetPackaging.ZIP_DIRECTORY
24 ? '.zip'
25 : extension);
26 // Add to manifest
27 this.files[asset.sourceHash] = {
28 source: {
29 path: asset.fileName,
30 executable: asset.executable,
31 packaging: asset.packaging,
32 },
33 destinations: {
34 [this.manifestEnvName(stack)]: {
35 bucketName: bucketName,
36 objectKey,
37 region: _shared_1.resolvedOr(stack.region, undefined),
38 assumeRoleArn: role?.assumeRoleArn,
39 assumeRoleExternalId: role?.assumeRoleExternalId,
40 },
41 },
42 };
43 const { region, urlSuffix } = stackLocationOrInstrinsics(stack);
44 const httpUrl = cfnify(`https://s3.${region}.${urlSuffix}/${bucketName}/${objectKey}`);
45 const s3ObjectUrlWithPlaceholders = `s3://${bucketName}/${objectKey}`;
46 // Return CFN expression
47 //
48 // 's3ObjectUrlWithPlaceholders' is intended for the CLI. The CLI ultimately needs a
49 // 'https://s3.REGION.amazonaws.com[.cn]/name/hash' URL to give to CloudFormation.
50 // However, there's no way for us to actually know the URL_SUFFIX in the framework, so
51 // we can't construct that URL. Instead, we record the 's3://.../...' form, and the CLI
52 // transforms it to the correct 'https://.../' URL before calling CloudFormation.
53 return {
54 bucketName: cfnify(bucketName),
55 objectKey,
56 httpUrl,
57 s3ObjectUrl: cfnify(s3ObjectUrlWithPlaceholders),
58 s3ObjectUrlWithPlaceholders,
59 s3Url: httpUrl,
60 };
61 }
62 addDockerImageAssetDefault(asset, stack, repositoryName, dockerTagPrefix, role) {
63 validateDockerImageAssetSource(asset);
64 const imageTag = dockerTagPrefix + asset.sourceHash;
65 // Add to manifest
66 this.dockerImages[asset.sourceHash] = {
67 source: {
68 executable: asset.executable,
69 directory: asset.directoryName,
70 dockerBuildArgs: asset.dockerBuildArgs,
71 dockerBuildTarget: asset.dockerBuildTarget,
72 dockerFile: asset.dockerFile,
73 networkMode: asset.networkMode,
74 platform: asset.platform,
75 },
76 destinations: {
77 [this.manifestEnvName(stack)]: {
78 repositoryName: repositoryName,
79 imageTag,
80 region: _shared_1.resolvedOr(stack.region, undefined),
81 assumeRoleArn: role?.assumeRoleArn,
82 assumeRoleExternalId: role?.assumeRoleExternalId,
83 },
84 },
85 };
86 const { account, region, urlSuffix } = stackLocationOrInstrinsics(stack);
87 // Return CFN expression
88 return {
89 repositoryName: cfnify(repositoryName),
90 imageUri: cfnify(`${account}.dkr.ecr.${region}.${urlSuffix}/${repositoryName}:${imageTag}`),
91 };
92 }
93 /**
94 * Write the manifest to disk, and add it to the synthesis session
95 *
96 * Reutrn the artifact Id
97 */
98 writeManifest(stack, session, additionalProps = {}) {
99 const artifactId = `${stack.artifactId}.assets`;
100 const manifestFile = `${artifactId}.json`;
101 const outPath = path.join(session.assembly.outdir, manifestFile);
102 const manifest = {
103 version: cxschema.Manifest.version(),
104 files: this.files,
105 dockerImages: this.dockerImages,
106 };
107 fs.writeFileSync(outPath, JSON.stringify(manifest, undefined, 2));
108 session.assembly.addArtifact(artifactId, {
109 type: cxschema.ArtifactType.ASSET_MANIFEST,
110 properties: {
111 file: manifestFile,
112 ...additionalProps,
113 },
114 });
115 return artifactId;
116 }
117 manifestEnvName(stack) {
118 return [
119 _shared_1.resolvedOr(stack.account, 'current_account'),
120 _shared_1.resolvedOr(stack.region, 'current_region'),
121 ].join('-');
122 }
123}
124exports.AssetManifestBuilder = AssetManifestBuilder;
125function validateFileAssetSource(asset) {
126 if (!!asset.executable === !!asset.fileName) {
127 throw new Error(`Exactly one of 'fileName' or 'executable' is required, got: ${JSON.stringify(asset)}`);
128 }
129 if (!!asset.packaging !== !!asset.fileName) {
130 throw new Error(`'packaging' is expected in combination with 'fileName', got: ${JSON.stringify(asset)}`);
131 }
132}
133function validateDockerImageAssetSource(asset) {
134 if (!!asset.executable === !!asset.directoryName) {
135 throw new Error(`Exactly one of 'directoryName' or 'executable' is required, got: ${JSON.stringify(asset)}`);
136 }
137 check('dockerBuildArgs');
138 check('dockerBuildTarget');
139 check('dockerFile');
140 function check(key) {
141 if (asset[key] && !asset.directoryName) {
142 throw new Error(`'${key}' is only allowed in combination with 'directoryName', got: ${JSON.stringify(asset)}`);
143 }
144 }
145}
146/**
147 * Return the stack locations if they're concrete, or the original CFN intrisics otherwise
148 *
149 * We need to return these instead of the tokenized versions of the strings,
150 * since we must accept those same ${AWS::AccountId}/${AWS::Region} placeholders
151 * in bucket names and role names (in order to allow environment-agnostic stacks).
152 *
153 * We'll wrap a single {Fn::Sub} around the final string in order to replace everything,
154 * but we can't have the token system render part of the string to {Fn::Join} because
155 * the CFN specification doesn't allow the {Fn::Sub} template string to be an arbitrary
156 * expression--it must be a string literal.
157 */
158function stackLocationOrInstrinsics(stack) {
159 return {
160 account: _shared_1.resolvedOr(stack.account, '${AWS::AccountId}'),
161 region: _shared_1.resolvedOr(stack.region, '${AWS::Region}'),
162 urlSuffix: _shared_1.resolvedOr(stack.urlSuffix, '${AWS::URLSuffix}'),
163 };
164}
165/**
166 * If the string still contains placeholders, wrap it in a Fn::Sub so they will be substituted at CFN deployment time
167 *
168 * (This happens to work because the placeholders we picked map directly onto CFN
169 * placeholders. If they didn't we'd have to do a transformation here).
170 */
171function cfnify(s) {
172 return s.indexOf('${') > -1 ? cfn_fn_1.Fn.sub(s) : s;
173}
174//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"_asset-manifest-builder.js","sourceRoot":"","sources":["_asset-manifest-builder.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\n\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport { FileAssetSource, FileAssetLocation, FileAssetPackaging, DockerImageAssetSource, DockerImageAssetLocation } from '../assets';\nimport { Fn } from '../cfn-fn';\nimport { ISynthesisSession } from '../construct-compat';\nimport { Stack } from '../stack';\nimport { resolvedOr } from './_shared';\n\n/**\n * Build an manifest from assets added to a stack synthesizer\n */\nexport class AssetManifestBuilder {\n  private readonly files: NonNullable<cxschema.AssetManifest['files']> = {};\n  private readonly dockerImages: NonNullable<cxschema.AssetManifest['dockerImages']> = {};\n\n  public addFileAssetDefault(\n    asset: FileAssetSource,\n    stack: Stack,\n    bucketName: string,\n    bucketPrefix: string,\n    role?: RoleOptions,\n  ): FileAssetLocation {\n    validateFileAssetSource(asset);\n\n    const extension =\n      asset.fileName != undefined ? path.extname(asset.fileName) : '';\n    const objectKey =\n      bucketPrefix +\n      asset.sourceHash +\n      (asset.packaging === FileAssetPackaging.ZIP_DIRECTORY\n        ? '.zip'\n        : extension);\n\n    // Add to manifest\n    this.files[asset.sourceHash] = {\n      source: {\n        path: asset.fileName,\n        executable: asset.executable,\n        packaging: asset.packaging,\n      },\n      destinations: {\n        [this.manifestEnvName(stack)]: {\n          bucketName: bucketName,\n          objectKey,\n          region: resolvedOr(stack.region, undefined),\n          assumeRoleArn: role?.assumeRoleArn,\n          assumeRoleExternalId: role?.assumeRoleExternalId,\n        },\n      },\n    };\n\n    const { region, urlSuffix } = stackLocationOrInstrinsics(stack);\n    const httpUrl = cfnify(\n      `https://s3.${region}.${urlSuffix}/${bucketName}/${objectKey}`,\n    );\n    const s3ObjectUrlWithPlaceholders = `s3://${bucketName}/${objectKey}`;\n\n    // Return CFN expression\n    //\n    // 's3ObjectUrlWithPlaceholders' is intended for the CLI. The CLI ultimately needs a\n    // 'https://s3.REGION.amazonaws.com[.cn]/name/hash' URL to give to CloudFormation.\n    // However, there's no way for us to actually know the URL_SUFFIX in the framework, so\n    // we can't construct that URL. Instead, we record the 's3://.../...' form, and the CLI\n    // transforms it to the correct 'https://.../' URL before calling CloudFormation.\n    return {\n      bucketName: cfnify(bucketName),\n      objectKey,\n      httpUrl,\n      s3ObjectUrl: cfnify(s3ObjectUrlWithPlaceholders),\n      s3ObjectUrlWithPlaceholders,\n      s3Url: httpUrl,\n    };\n  }\n\n  public addDockerImageAssetDefault(\n    asset: DockerImageAssetSource,\n    stack: Stack,\n    repositoryName: string,\n    dockerTagPrefix: string,\n    role?: RoleOptions,\n  ): DockerImageAssetLocation {\n    validateDockerImageAssetSource(asset);\n    const imageTag = dockerTagPrefix + asset.sourceHash;\n\n    // Add to manifest\n    this.dockerImages[asset.sourceHash] = {\n      source: {\n        executable: asset.executable,\n        directory: asset.directoryName,\n        dockerBuildArgs: asset.dockerBuildArgs,\n        dockerBuildTarget: asset.dockerBuildTarget,\n        dockerFile: asset.dockerFile,\n        networkMode: asset.networkMode,\n        platform: asset.platform,\n      },\n      destinations: {\n        [this.manifestEnvName(stack)]: {\n          repositoryName: repositoryName,\n          imageTag,\n          region: resolvedOr(stack.region, undefined),\n          assumeRoleArn: role?.assumeRoleArn,\n          assumeRoleExternalId: role?.assumeRoleExternalId,\n        },\n      },\n    };\n\n    const { account, region, urlSuffix } = stackLocationOrInstrinsics(stack);\n\n    // Return CFN expression\n    return {\n      repositoryName: cfnify(repositoryName),\n      imageUri: cfnify(\n        `${account}.dkr.ecr.${region}.${urlSuffix}/${repositoryName}:${imageTag}`,\n      ),\n    };\n  }\n\n  /**\n   * Write the manifest to disk, and add it to the synthesis session\n   *\n   * Reutrn the artifact Id\n   */\n  public writeManifest(\n    stack: Stack,\n    session: ISynthesisSession,\n    additionalProps: Partial<cxschema.AssetManifestProperties> = {},\n  ): string {\n    const artifactId = `${stack.artifactId}.assets`;\n    const manifestFile = `${artifactId}.json`;\n    const outPath = path.join(session.assembly.outdir, manifestFile);\n\n    const manifest: cxschema.AssetManifest = {\n      version: cxschema.Manifest.version(),\n      files: this.files,\n      dockerImages: this.dockerImages,\n    };\n\n    fs.writeFileSync(outPath, JSON.stringify(manifest, undefined, 2));\n\n    session.assembly.addArtifact(artifactId, {\n      type: cxschema.ArtifactType.ASSET_MANIFEST,\n      properties: {\n        file: manifestFile,\n        ...additionalProps,\n      },\n    });\n\n    return artifactId;\n  }\n\n  private manifestEnvName(stack: Stack): string {\n    return [\n      resolvedOr(stack.account, 'current_account'),\n      resolvedOr(stack.region, 'current_region'),\n    ].join('-');\n  }\n}\n\nexport interface RoleOptions {\n  readonly assumeRoleArn?: string;\n  readonly assumeRoleExternalId?: string;\n}\n\nfunction validateFileAssetSource(asset: FileAssetSource) {\n  if (!!asset.executable === !!asset.fileName) {\n    throw new Error(`Exactly one of 'fileName' or 'executable' is required, got: ${JSON.stringify(asset)}`);\n  }\n\n  if (!!asset.packaging !== !!asset.fileName) {\n    throw new Error(`'packaging' is expected in combination with 'fileName', got: ${JSON.stringify(asset)}`);\n  }\n}\n\nfunction validateDockerImageAssetSource(asset: DockerImageAssetSource) {\n  if (!!asset.executable === !!asset.directoryName) {\n    throw new Error(`Exactly one of 'directoryName' or 'executable' is required, got: ${JSON.stringify(asset)}`);\n  }\n\n  check('dockerBuildArgs');\n  check('dockerBuildTarget');\n  check('dockerFile');\n\n  function check<K extends keyof DockerImageAssetSource>(key: K) {\n    if (asset[key] && !asset.directoryName) {\n      throw new Error(`'${key}' is only allowed in combination with 'directoryName', got: ${JSON.stringify(asset)}`);\n    }\n  }\n}\n\n/**\n * Return the stack locations if they're concrete, or the original CFN intrisics otherwise\n *\n * We need to return these instead of the tokenized versions of the strings,\n * since we must accept those same ${AWS::AccountId}/${AWS::Region} placeholders\n * in bucket names and role names (in order to allow environment-agnostic stacks).\n *\n * We'll wrap a single {Fn::Sub} around the final string in order to replace everything,\n * but we can't have the token system render part of the string to {Fn::Join} because\n * the CFN specification doesn't allow the {Fn::Sub} template string to be an arbitrary\n * expression--it must be a string literal.\n */\nfunction stackLocationOrInstrinsics(stack: Stack) {\n  return {\n    account: resolvedOr(stack.account, '${AWS::AccountId}'),\n    region: resolvedOr(stack.region, '${AWS::Region}'),\n    urlSuffix: resolvedOr(stack.urlSuffix, '${AWS::URLSuffix}'),\n  };\n}\n\n/**\n * If the string still contains placeholders, wrap it in a Fn::Sub so they will be substituted at CFN deployment time\n *\n * (This happens to work because the placeholders we picked map directly onto CFN\n * placeholders. If they didn't we'd have to do a transformation here).\n */\nfunction cfnify(s: string): string {\n  return s.indexOf('${') > -1 ? Fn.sub(s) : s;\n}"]}
\No newline at end of file