cdk-assets 3.0.0 → 3.1.1
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.
- package/lib/asset-manifest.d.ts +28 -4
- package/lib/asset-manifest.js +31 -6
- package/lib/aws.d.ts +1 -0
- package/lib/aws.js +11 -5
- package/lib/private/asset-handler.d.ts +6 -0
- package/lib/private/asset-handler.js +1 -1
- package/lib/private/handlers/files.js +2 -2
- package/lib/publishing.js +7 -7
- package/package.json +4 -5
package/lib/asset-manifest.d.ts
CHANGED
|
@@ -67,6 +67,19 @@ export interface IManifestEntry {
|
|
|
67
67
|
* Type-dependent destination data
|
|
68
68
|
*/
|
|
69
69
|
readonly genericDestination: unknown;
|
|
70
|
+
/**
|
|
71
|
+
* Return a display name for this asset
|
|
72
|
+
*
|
|
73
|
+
* The `includeDestination` parameter controls whether or not to include the
|
|
74
|
+
* destination ID in the display name.
|
|
75
|
+
*
|
|
76
|
+
* - Pass `false` if you are displaying notifications about building the
|
|
77
|
+
* asset, or if you are describing the work of building the asset and publishing
|
|
78
|
+
* to all destinations at the same time.
|
|
79
|
+
* - Pass `true` if you are displaying notifications about publishing to a
|
|
80
|
+
* specific destination.
|
|
81
|
+
*/
|
|
82
|
+
displayName(includeDestination: boolean): string;
|
|
70
83
|
}
|
|
71
84
|
/**
|
|
72
85
|
* A manifest entry for a file asset
|
|
@@ -74,6 +87,7 @@ export interface IManifestEntry {
|
|
|
74
87
|
export declare class FileManifestEntry implements IManifestEntry {
|
|
75
88
|
/** Identifier for this asset */
|
|
76
89
|
readonly id: DestinationIdentifier;
|
|
90
|
+
private readonly _displayName;
|
|
77
91
|
/** Source of the file asset */
|
|
78
92
|
readonly source: FileSource;
|
|
79
93
|
/** Destination for the file asset */
|
|
@@ -83,11 +97,12 @@ export declare class FileManifestEntry implements IManifestEntry {
|
|
|
83
97
|
readonly type = "file";
|
|
84
98
|
constructor(
|
|
85
99
|
/** Identifier for this asset */
|
|
86
|
-
id: DestinationIdentifier,
|
|
100
|
+
id: DestinationIdentifier, _displayName: string | undefined,
|
|
87
101
|
/** Source of the file asset */
|
|
88
102
|
source: FileSource,
|
|
89
103
|
/** Destination for the file asset */
|
|
90
104
|
destination: FileDestination);
|
|
105
|
+
displayName(includeDestination: boolean): string;
|
|
91
106
|
}
|
|
92
107
|
/**
|
|
93
108
|
* A manifest entry for a docker image asset
|
|
@@ -95,6 +110,7 @@ export declare class FileManifestEntry implements IManifestEntry {
|
|
|
95
110
|
export declare class DockerImageManifestEntry implements IManifestEntry {
|
|
96
111
|
/** Identifier for this asset */
|
|
97
112
|
readonly id: DestinationIdentifier;
|
|
113
|
+
private readonly _displayName;
|
|
98
114
|
/** Source of the file asset */
|
|
99
115
|
readonly source: DockerImageSource;
|
|
100
116
|
/** Destination for the file asset */
|
|
@@ -104,17 +120,25 @@ export declare class DockerImageManifestEntry implements IManifestEntry {
|
|
|
104
120
|
readonly type = "docker-image";
|
|
105
121
|
constructor(
|
|
106
122
|
/** Identifier for this asset */
|
|
107
|
-
id: DestinationIdentifier,
|
|
123
|
+
id: DestinationIdentifier, _displayName: string | undefined,
|
|
108
124
|
/** Source of the file asset */
|
|
109
125
|
source: DockerImageSource,
|
|
110
126
|
/** Destination for the file asset */
|
|
111
127
|
destination: DockerImageDestination);
|
|
128
|
+
displayName(includeDestination: boolean): string;
|
|
112
129
|
}
|
|
113
130
|
/**
|
|
114
131
|
* Identify an asset destination in an asset manifest
|
|
115
132
|
*
|
|
116
|
-
*
|
|
117
|
-
*
|
|
133
|
+
* This class is used to identify both an asset to be built as well as a
|
|
134
|
+
* destination where an asset will be published. However, when reasoning about
|
|
135
|
+
* building assets the destination part can be ignored, because the same asset
|
|
136
|
+
* being sent to multiple destinations will only need to be built once and their
|
|
137
|
+
* assetIds are all the same.
|
|
138
|
+
*
|
|
139
|
+
* When stringified, this will be a combination of the source and destination
|
|
140
|
+
* IDs; if a string representation of the source is necessary, use `id.assetId`
|
|
141
|
+
* instead.
|
|
118
142
|
*/
|
|
119
143
|
export declare class DestinationIdentifier {
|
|
120
144
|
/**
|
package/lib/asset-manifest.js
CHANGED
|
@@ -117,7 +117,7 @@ function makeEntries(assets, ctor) {
|
|
|
117
117
|
const ret = new Array();
|
|
118
118
|
for (const [assetId, asset] of Object.entries(assets)) {
|
|
119
119
|
for (const [destId, destination] of Object.entries(asset.destinations)) {
|
|
120
|
-
ret.push(new ctor(new DestinationIdentifier(assetId, destId), asset.source, destination));
|
|
120
|
+
ret.push(new ctor(new DestinationIdentifier(assetId, destId), asset.displayName, asset.source, destination));
|
|
121
121
|
}
|
|
122
122
|
}
|
|
123
123
|
return ret;
|
|
@@ -129,18 +129,27 @@ const ASSET_TYPES = ['files', 'dockerImages'];
|
|
|
129
129
|
class FileManifestEntry {
|
|
130
130
|
constructor(
|
|
131
131
|
/** Identifier for this asset */
|
|
132
|
-
id,
|
|
132
|
+
id, _displayName,
|
|
133
133
|
/** Source of the file asset */
|
|
134
134
|
source,
|
|
135
135
|
/** Destination for the file asset */
|
|
136
136
|
destination) {
|
|
137
137
|
this.id = id;
|
|
138
|
+
this._displayName = _displayName;
|
|
138
139
|
this.source = source;
|
|
139
140
|
this.destination = destination;
|
|
140
141
|
this.type = 'file';
|
|
141
142
|
this.genericSource = source;
|
|
142
143
|
this.genericDestination = destination;
|
|
143
144
|
}
|
|
145
|
+
displayName(includeDestination) {
|
|
146
|
+
if (includeDestination) {
|
|
147
|
+
return this._displayName ? `${this._displayName} (${this.id.destinationId})` : `${this.id}`;
|
|
148
|
+
}
|
|
149
|
+
else {
|
|
150
|
+
return this._displayName ? this._displayName : this.id.assetId;
|
|
151
|
+
}
|
|
152
|
+
}
|
|
144
153
|
}
|
|
145
154
|
exports.FileManifestEntry = FileManifestEntry;
|
|
146
155
|
/**
|
|
@@ -149,25 +158,41 @@ exports.FileManifestEntry = FileManifestEntry;
|
|
|
149
158
|
class DockerImageManifestEntry {
|
|
150
159
|
constructor(
|
|
151
160
|
/** Identifier for this asset */
|
|
152
|
-
id,
|
|
161
|
+
id, _displayName,
|
|
153
162
|
/** Source of the file asset */
|
|
154
163
|
source,
|
|
155
164
|
/** Destination for the file asset */
|
|
156
165
|
destination) {
|
|
157
166
|
this.id = id;
|
|
167
|
+
this._displayName = _displayName;
|
|
158
168
|
this.source = source;
|
|
159
169
|
this.destination = destination;
|
|
160
170
|
this.type = 'docker-image';
|
|
161
171
|
this.genericSource = source;
|
|
162
172
|
this.genericDestination = destination;
|
|
163
173
|
}
|
|
174
|
+
displayName(includeDestination) {
|
|
175
|
+
if (includeDestination) {
|
|
176
|
+
return this._displayName ? `${this._displayName} (${this.id.destinationId})` : `${this.id}`;
|
|
177
|
+
}
|
|
178
|
+
else {
|
|
179
|
+
return this._displayName ? this._displayName : this.id.assetId;
|
|
180
|
+
}
|
|
181
|
+
}
|
|
164
182
|
}
|
|
165
183
|
exports.DockerImageManifestEntry = DockerImageManifestEntry;
|
|
166
184
|
/**
|
|
167
185
|
* Identify an asset destination in an asset manifest
|
|
168
186
|
*
|
|
169
|
-
*
|
|
170
|
-
*
|
|
187
|
+
* This class is used to identify both an asset to be built as well as a
|
|
188
|
+
* destination where an asset will be published. However, when reasoning about
|
|
189
|
+
* building assets the destination part can be ignored, because the same asset
|
|
190
|
+
* being sent to multiple destinations will only need to be built once and their
|
|
191
|
+
* assetIds are all the same.
|
|
192
|
+
*
|
|
193
|
+
* When stringified, this will be a combination of the source and destination
|
|
194
|
+
* IDs; if a string representation of the source is necessary, use `id.assetId`
|
|
195
|
+
* instead.
|
|
171
196
|
*/
|
|
172
197
|
class DestinationIdentifier {
|
|
173
198
|
constructor(assetId, destinationId) {
|
|
@@ -242,4 +267,4 @@ function prefixTreeChars(xs, prefix = '') {
|
|
|
242
267
|
}
|
|
243
268
|
return ret;
|
|
244
269
|
}
|
|
245
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"asset-manifest.js","sourceRoot":"","sources":["asset-manifest.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport {\n  AssetManifest as AssetManifestSchema,\n  DockerImageDestination,\n  DockerImageSource,\n  FileDestination,\n  FileSource,\n  Manifest,\n} from '@aws-cdk/cloud-assembly-schema';\n\n/**\n * A manifest of assets\n */\nexport class AssetManifest {\n  /**\n   * The default name of the asset manifest in a cdk.out directory\n   */\n  public static readonly DEFAULT_FILENAME = 'assets.json';\n\n  /**\n   * Load an asset manifest from the given file\n   */\n  public static fromFile(fileName: string) {\n    try {\n      const obj = Manifest.loadAssetManifest(fileName);\n      return new AssetManifest(path.dirname(fileName), obj);\n    } catch (e: any) {\n      throw new Error(`Cannot read asset manifest '${fileName}': ${e.message}`);\n    }\n  }\n\n  /**\n   * Load an asset manifest from the given file or directory\n   *\n   * If the argument given is a directoy, the default asset file name will be used.\n   */\n  public static fromPath(filePath: string) {\n    let st;\n    try {\n      st = fs.statSync(filePath);\n    } catch (e: any) {\n      throw new Error(`Cannot read asset manifest at '${filePath}': ${e.message}`);\n    }\n    if (st.isDirectory()) {\n      return AssetManifest.fromFile(path.join(filePath, AssetManifest.DEFAULT_FILENAME));\n    }\n    return AssetManifest.fromFile(filePath);\n  }\n\n  /**\n   * The directory where the manifest was found\n   */\n  public readonly directory: string;\n\n  constructor(\n    directory: string,\n    private readonly manifest: AssetManifestSchema,\n  ) {\n    this.directory = directory;\n  }\n\n  /**\n   * Select a subset of assets and destinations from this manifest.\n   *\n   * Only assets with at least 1 selected destination are retained.\n   *\n   * If selection is not given, everything is returned.\n   */\n  public select(selection?: DestinationPattern[]): AssetManifest {\n    if (selection === undefined) {\n      return this;\n    }\n\n    const ret: AssetManifestSchema & Required<Pick<AssetManifestSchema, AssetType>> = {\n      version: this.manifest.version,\n      dockerImages: {},\n      files: {},\n    };\n\n    for (const assetType of ASSET_TYPES) {\n      for (const [assetId, asset] of Object.entries(this.manifest[assetType] || {})) {\n        const filteredDestinations = filterDict(asset.destinations, (_, destId) =>\n          selection.some((sel) => sel.matches(new DestinationIdentifier(assetId, destId))),\n        );\n\n        if (Object.keys(filteredDestinations).length > 0) {\n          ret[assetType][assetId] = {\n            ...asset,\n            destinations: filteredDestinations,\n          };\n        }\n      }\n    }\n\n    return new AssetManifest(this.directory, ret);\n  }\n\n  /**\n   * Describe the asset manifest as a list of strings\n   */\n  public list() {\n    return [\n      ...describeAssets('file', this.manifest.files || {}),\n      ...describeAssets('docker-image', this.manifest.dockerImages || {}),\n    ];\n\n    function describeAssets(\n      type: string,\n      assets: Record<string, { source: any; destinations: Record<string, any> }>,\n    ) {\n      const ret = new Array<string>();\n      for (const [assetId, asset] of Object.entries(assets || {})) {\n        ret.push(`${assetId} ${type} ${JSON.stringify(asset.source)}`);\n\n        const destStrings = Object.entries(asset.destinations).map(\n          ([destId, dest]) => ` ${assetId}:${destId} ${JSON.stringify(dest)}`,\n        );\n        ret.push(...prefixTreeChars(destStrings, '  '));\n      }\n      return ret;\n    }\n  }\n\n  /**\n   * List of assets per destination\n   *\n   * Returns one asset for every publishable destination. Multiple asset\n   * destinations may share the same asset source.\n   */\n  public get entries(): IManifestEntry[] {\n    return [\n      ...makeEntries(this.manifest.files || {}, FileManifestEntry),\n      ...makeEntries(this.manifest.dockerImages || {}, DockerImageManifestEntry),\n    ];\n  }\n\n  /**\n   * List of file assets, splat out to destinations\n   */\n  public get files(): FileManifestEntry[] {\n    return makeEntries(this.manifest.files || {}, FileManifestEntry);\n  }\n}\n\nfunction makeEntries<A, B, C>(\n  assets: Record<string, { source: A; destinations: Record<string, B> }>,\n  ctor: new (id: DestinationIdentifier, source: A, destination: B) => C,\n): C[] {\n  const ret = new Array<C>();\n  for (const [assetId, asset] of Object.entries(assets)) {\n    for (const [destId, destination] of Object.entries(asset.destinations)) {\n      ret.push(new ctor(new DestinationIdentifier(assetId, destId), asset.source, destination));\n    }\n  }\n  return ret;\n}\n\ntype AssetType = 'files' | 'dockerImages';\n\nconst ASSET_TYPES: AssetType[] = ['files', 'dockerImages'];\n\n/**\n * A single asset from an asset manifest'\n */\nexport interface IManifestEntry {\n  /**\n   * The identifier of the asset and its destination\n   */\n  readonly id: DestinationIdentifier;\n\n  /**\n   * The type of asset\n   */\n  readonly type: string;\n\n  /**\n   * Type-dependent source data\n   */\n  readonly genericSource: unknown;\n\n  /**\n   * Type-dependent destination data\n   */\n  readonly genericDestination: unknown;\n}\n\n/**\n * A manifest entry for a file asset\n */\nexport class FileManifestEntry implements IManifestEntry {\n  public readonly genericSource: unknown;\n  public readonly genericDestination: unknown;\n  public readonly type = 'file';\n\n  constructor(\n    /** Identifier for this asset */\n    public readonly id: DestinationIdentifier,\n    /** Source of the file asset */\n    public readonly source: FileSource,\n    /** Destination for the file asset */\n    public readonly destination: FileDestination,\n  ) {\n    this.genericSource = source;\n    this.genericDestination = destination;\n  }\n}\n\n/**\n * A manifest entry for a docker image asset\n */\nexport class DockerImageManifestEntry implements IManifestEntry {\n  public readonly genericSource: unknown;\n  public readonly genericDestination: unknown;\n  public readonly type = 'docker-image';\n\n  constructor(\n    /** Identifier for this asset */\n    public readonly id: DestinationIdentifier,\n    /** Source of the file asset */\n    public readonly source: DockerImageSource,\n    /** Destination for the file asset */\n    public readonly destination: DockerImageDestination,\n  ) {\n    this.genericSource = source;\n    this.genericDestination = destination;\n  }\n}\n\n/**\n * Identify an asset destination in an asset manifest\n *\n * When stringified, this will be a combination of the source\n * and destination IDs.\n */\nexport class DestinationIdentifier {\n  /**\n   * Identifies the asset, by source.\n   *\n   * The assetId will be the same between assets that represent\n   * the same physical file or image.\n   */\n  public readonly assetId: string;\n\n  /**\n   * Identifies the destination where this asset will be published\n   */\n  public readonly destinationId: string;\n\n  constructor(assetId: string, destinationId: string) {\n    this.assetId = assetId;\n    this.destinationId = destinationId;\n  }\n\n  /**\n   * Return a string representation for this asset identifier\n   */\n  public toString() {\n    return this.destinationId ? `${this.assetId}:${this.destinationId}` : this.assetId;\n  }\n}\n\nfunction filterDict<A>(\n  xs: Record<string, A>,\n  pred: (x: A, key: string) => boolean,\n): Record<string, A> {\n  const ret: Record<string, A> = {};\n  for (const [key, value] of Object.entries(xs)) {\n    if (pred(value, key)) {\n      ret[key] = value;\n    }\n  }\n  return ret;\n}\n\n/**\n * A filter pattern for an destination identifier\n */\nexport class DestinationPattern {\n  /**\n   * Parse a ':'-separated string into an asset/destination identifier\n   */\n  public static parse(s: string) {\n    if (!s) {\n      throw new Error('Empty string is not a valid destination identifier');\n    }\n    const parts = s.split(':').map((x) => (x !== '*' ? x : undefined));\n    if (parts.length === 1) {\n      return new DestinationPattern(parts[0]);\n    }\n    if (parts.length === 2) {\n      return new DestinationPattern(parts[0] || undefined, parts[1] || undefined);\n    }\n    throw new Error(`Asset identifier must contain at most 2 ':'-separated parts, got '${s}'`);\n  }\n\n  /**\n   * Identifies the asset, by source.\n   */\n  public readonly assetId?: string;\n\n  /**\n   * Identifies the destination where this asset will be published\n   */\n  public readonly destinationId?: string;\n\n  constructor(assetId?: string, destinationId?: string) {\n    this.assetId = assetId;\n    this.destinationId = destinationId;\n  }\n\n  /**\n   * Whether or not this pattern matches the given identifier\n   */\n  public matches(id: DestinationIdentifier) {\n    return (\n      (this.assetId === undefined || this.assetId === id.assetId) &&\n      (this.destinationId === undefined || this.destinationId === id.destinationId)\n    );\n  }\n\n  /**\n   * Return a string representation for this asset identifier\n   */\n  public toString() {\n    return `${this.assetId ?? '*'}:${this.destinationId ?? '*'}`;\n  }\n}\n\n/**\n * Prefix box-drawing characters to make lines look like a hanging tree\n */\nfunction prefixTreeChars(xs: string[], prefix = '') {\n  const ret = new Array<string>();\n  for (let i = 0; i < xs.length; i++) {\n    const isLast = i === xs.length - 1;\n    const boxChar = isLast ? '└' : '├';\n    ret.push(`${prefix}${boxChar}${xs[i]}`);\n  }\n  return ret;\n}\n"]}
|
|
270
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"asset-manifest.js","sourceRoot":"","sources":["asset-manifest.ts"],"names":[],"mappings":";;;AAAA,yBAAyB;AACzB,6BAA6B;AAC7B,0EAOwC;AAExC;;GAEG;AACH,MAAa,aAAa;IAMxB;;OAEG;IACI,MAAM,CAAC,QAAQ,CAAC,QAAgB;QACrC,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,gCAAQ,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC;YACjD,OAAO,IAAI,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,GAAG,CAAC,CAAC;QACxD,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,MAAM,IAAI,KAAK,CAAC,+BAA+B,QAAQ,MAAM,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;QAC5E,CAAC;IACH,CAAC;IAED;;;;OAIG;IACI,MAAM,CAAC,QAAQ,CAAC,QAAgB;QACrC,IAAI,EAAE,CAAC;QACP,IAAI,CAAC;YACH,EAAE,GAAG,EAAE,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;QAC7B,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,MAAM,IAAI,KAAK,CAAC,kCAAkC,QAAQ,MAAM,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;QAC/E,CAAC;QACD,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC;YACrB,OAAO,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,aAAa,CAAC,gBAAgB,CAAC,CAAC,CAAC;QACrF,CAAC;QACD,OAAO,aAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;IAC1C,CAAC;IAOD,YACE,SAAiB,EACA,QAA6B;QAA7B,aAAQ,GAAR,QAAQ,CAAqB;QAE9C,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC;IAC7B,CAAC;IAED;;;;;;OAMG;IACI,MAAM,CAAC,SAAgC;QAC5C,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;YAC5B,OAAO,IAAI,CAAC;QACd,CAAC;QAED,MAAM,GAAG,GAAyE;YAChF,OAAO,EAAE,IAAI,CAAC,QAAQ,CAAC,OAAO;YAC9B,YAAY,EAAE,EAAE;YAChB,KAAK,EAAE,EAAE;SACV,CAAC;QAEF,KAAK,MAAM,SAAS,IAAI,WAAW,EAAE,CAAC;YACpC,KAAK,MAAM,CAAC,OAAO,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC;gBAC9E,MAAM,oBAAoB,GAAG,UAAU,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC,CAAC,EAAE,MAAM,EAAE,EAAE,CACxE,SAAS,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,qBAAqB,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC,CAAC,CACjF,CAAC;gBAEF,IAAI,MAAM,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACjD,GAAG,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,GAAG;wBACxB,GAAG,KAAK;wBACR,YAAY,EAAE,oBAAoB;qBACnC,CAAC;gBACJ,CAAC;YACH,CAAC;QACH,CAAC;QAED,OAAO,IAAI,aAAa,CAAC,IAAI,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC;IAChD,CAAC;IAED;;OAEG;IACI,IAAI;QACT,OAAO;YACL,GAAG,cAAc,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,KAAK,IAAI,EAAE,CAAC;YACpD,GAAG,cAAc,CAAC,cAAc,EAAE,IAAI,CAAC,QAAQ,CAAC,YAAY,IAAI,EAAE,CAAC;SACpE,CAAC;QAEF,SAAS,cAAc,CACrB,IAAY,EACZ,MAA0E;YAE1E,MAAM,GAAG,GAAG,IAAI,KAAK,EAAU,CAAC;YAChC,KAAK,MAAM,CAAC,OAAO,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,IAAI,EAAE,CAAC,EAAE,CAAC;gBAC5D,GAAG,CAAC,IAAI,CAAC,GAAG,OAAO,IAAI,IAAI,IAAI,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;gBAE/D,MAAM,WAAW,GAAG,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC,GAAG,CACxD,CAAC,CAAC,MAAM,EAAE,IAAI,CAAC,EAAE,EAAE,CAAC,IAAI,OAAO,IAAI,MAAM,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,CACpE,CAAC;gBACF,GAAG,CAAC,IAAI,CAAC,GAAG,eAAe,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC,CAAC;YAClD,CAAC;YACD,OAAO,GAAG,CAAC;QACb,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,IAAW,OAAO;QAChB,OAAO;YACL,GAAG,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,IAAI,EAAE,EAAE,iBAAiB,CAAC;YAC5D,GAAG,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,YAAY,IAAI,EAAE,EAAE,wBAAwB,CAAC;SAC3E,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,IAAW,KAAK;QACd,OAAO,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,IAAI,EAAE,EAAE,iBAAiB,CAAC,CAAC;IACnE,CAAC;;AAhIH,sCAiIC;AAhIC;;GAEG;AACoB,8BAAgB,GAAG,aAAa,CAAC;AA+H1D,SAAS,WAAW,CAClB,MAA4F,EAC5F,IAAsG;IAEtG,MAAM,GAAG,GAAG,IAAI,KAAK,EAAK,CAAC;IAC3B,KAAK,MAAM,CAAC,OAAO,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;QACtD,KAAK,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,YAAY,CAAC,EAAE,CAAC;YACvE,GAAG,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,IAAI,qBAAqB,CAAC,OAAO,EAAE,MAAM,CAAC,EAAE,KAAK,CAAC,WAAW,EAAE,KAAK,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,CAAC;QAC/G,CAAC;IACH,CAAC;IACD,OAAO,GAAG,CAAC;AACb,CAAC;AAID,MAAM,WAAW,GAAgB,CAAC,OAAO,EAAE,cAAc,CAAC,CAAC;AAyC3D;;GAEG;AACH,MAAa,iBAAiB;IAK5B;IACE,gCAAgC;IAChB,EAAyB,EACxB,YAAgC;IACjD,+BAA+B;IACf,MAAkB;IAClC,qCAAqC;IACrB,WAA4B;QAL5B,OAAE,GAAF,EAAE,CAAuB;QACxB,iBAAY,GAAZ,YAAY,CAAoB;QAEjC,WAAM,GAAN,MAAM,CAAY;QAElB,gBAAW,GAAX,WAAW,CAAiB;QAT9B,SAAI,GAAG,MAAM,CAAC;QAW5B,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC;QAC5B,IAAI,CAAC,kBAAkB,GAAG,WAAW,CAAC;IACxC,CAAC;IAEM,WAAW,CAAC,kBAA2B;QAC5C,IAAI,kBAAkB,EAAE,CAAC;YACvB,OAAO,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,YAAY,KAAK,IAAI,CAAC,EAAE,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,EAAE,EAAE,CAAC;QAC9F,CAAC;aAAM,CAAC;YACN,OAAO,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,CAAC;QACjE,CAAC;IACH,CAAC;CACF;AAzBD,8CAyBC;AAED;;GAEG;AACH,MAAa,wBAAwB;IAKnC;IACE,gCAAgC;IAChB,EAAyB,EACxB,YAAgC;IACjD,+BAA+B;IACf,MAAyB;IACzC,qCAAqC;IACrB,WAAmC;QALnC,OAAE,GAAF,EAAE,CAAuB;QACxB,iBAAY,GAAZ,YAAY,CAAoB;QAEjC,WAAM,GAAN,MAAM,CAAmB;QAEzB,gBAAW,GAAX,WAAW,CAAwB;QATrC,SAAI,GAAG,cAAc,CAAC;QAWpC,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC;QAC5B,IAAI,CAAC,kBAAkB,GAAG,WAAW,CAAC;IACxC,CAAC;IAEM,WAAW,CAAC,kBAA2B;QAC5C,IAAI,kBAAkB,EAAE,CAAC;YACvB,OAAO,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,YAAY,KAAK,IAAI,CAAC,EAAE,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,EAAE,EAAE,CAAC;QAC9F,CAAC;aAAM,CAAC;YACN,OAAO,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,CAAC;QACjE,CAAC;IACH,CAAC;CACF;AAzBD,4DAyBC;AAED;;;;;;;;;;;;GAYG;AACH,MAAa,qBAAqB;IAchC,YAAY,OAAe,EAAE,aAAqB;QAChD,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACvB,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;IACrC,CAAC;IAED;;OAEG;IACI,QAAQ;QACb,OAAO,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC;IACrF,CAAC;CACF;AAzBD,sDAyBC;AAED,SAAS,UAAU,CACjB,EAAqB,EACrB,IAAoC;IAEpC,MAAM,GAAG,GAAsB,EAAE,CAAC;IAClC,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC;QAC9C,IAAI,IAAI,CAAC,KAAK,EAAE,GAAG,CAAC,EAAE,CAAC;YACrB,GAAG,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;QACnB,CAAC;IACH,CAAC;IACD,OAAO,GAAG,CAAC;AACb,CAAC;AAED;;GAEG;AACH,MAAa,kBAAkB;IAC7B;;OAEG;IACI,MAAM,CAAC,KAAK,CAAC,CAAS;QAC3B,IAAI,CAAC,CAAC,EAAE,CAAC;YACP,MAAM,IAAI,KAAK,CAAC,oDAAoD,CAAC,CAAC;QACxE,CAAC;QACD,MAAM,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC;QACnE,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACvB,OAAO,IAAI,kBAAkB,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QAC1C,CAAC;QACD,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACvB,OAAO,IAAI,kBAAkB,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,SAAS,EAAE,KAAK,CAAC,CAAC,CAAC,IAAI,SAAS,CAAC,CAAC;QAC9E,CAAC;QACD,MAAM,IAAI,KAAK,CAAC,qEAAqE,CAAC,GAAG,CAAC,CAAC;IAC7F,CAAC;IAYD,YAAY,OAAgB,EAAE,aAAsB;QAClD,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACvB,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;IACrC,CAAC;IAED;;OAEG;IACI,OAAO,CAAC,EAAyB;QACtC,OAAO,CACL,CAAC,IAAI,CAAC,OAAO,KAAK,SAAS,IAAI,IAAI,CAAC,OAAO,KAAK,EAAE,CAAC,OAAO,CAAC;YAC3D,CAAC,IAAI,CAAC,aAAa,KAAK,SAAS,IAAI,IAAI,CAAC,aAAa,KAAK,EAAE,CAAC,aAAa,CAAC,CAC9E,CAAC;IACJ,CAAC;IAED;;OAEG;IACI,QAAQ;QACb,OAAO,GAAG,IAAI,CAAC,OAAO,IAAI,GAAG,IAAI,IAAI,CAAC,aAAa,IAAI,GAAG,EAAE,CAAC;IAC/D,CAAC;CACF;AAjDD,gDAiDC;AAED;;GAEG;AACH,SAAS,eAAe,CAAC,EAAY,EAAE,MAAM,GAAG,EAAE;IAChD,MAAM,GAAG,GAAG,IAAI,KAAK,EAAU,CAAC;IAChC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACnC,MAAM,MAAM,GAAG,CAAC,KAAK,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC;QACnC,MAAM,OAAO,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;QACnC,GAAG,CAAC,IAAI,CAAC,GAAG,MAAM,GAAG,OAAO,GAAG,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IAC1C,CAAC;IACD,OAAO,GAAG,CAAC;AACb,CAAC","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport {\n  AssetManifest as AssetManifestSchema,\n  DockerImageDestination,\n  DockerImageSource,\n  FileDestination,\n  FileSource,\n  Manifest,\n} from '@aws-cdk/cloud-assembly-schema';\n\n/**\n * A manifest of assets\n */\nexport class AssetManifest {\n  /**\n   * The default name of the asset manifest in a cdk.out directory\n   */\n  public static readonly DEFAULT_FILENAME = 'assets.json';\n\n  /**\n   * Load an asset manifest from the given file\n   */\n  public static fromFile(fileName: string) {\n    try {\n      const obj = Manifest.loadAssetManifest(fileName);\n      return new AssetManifest(path.dirname(fileName), obj);\n    } catch (e: any) {\n      throw new Error(`Cannot read asset manifest '${fileName}': ${e.message}`);\n    }\n  }\n\n  /**\n   * Load an asset manifest from the given file or directory\n   *\n   * If the argument given is a directoy, the default asset file name will be used.\n   */\n  public static fromPath(filePath: string) {\n    let st;\n    try {\n      st = fs.statSync(filePath);\n    } catch (e: any) {\n      throw new Error(`Cannot read asset manifest at '${filePath}': ${e.message}`);\n    }\n    if (st.isDirectory()) {\n      return AssetManifest.fromFile(path.join(filePath, AssetManifest.DEFAULT_FILENAME));\n    }\n    return AssetManifest.fromFile(filePath);\n  }\n\n  /**\n   * The directory where the manifest was found\n   */\n  public readonly directory: string;\n\n  constructor(\n    directory: string,\n    private readonly manifest: AssetManifestSchema,\n  ) {\n    this.directory = directory;\n  }\n\n  /**\n   * Select a subset of assets and destinations from this manifest.\n   *\n   * Only assets with at least 1 selected destination are retained.\n   *\n   * If selection is not given, everything is returned.\n   */\n  public select(selection?: DestinationPattern[]): AssetManifest {\n    if (selection === undefined) {\n      return this;\n    }\n\n    const ret: AssetManifestSchema & Required<Pick<AssetManifestSchema, AssetType>> = {\n      version: this.manifest.version,\n      dockerImages: {},\n      files: {},\n    };\n\n    for (const assetType of ASSET_TYPES) {\n      for (const [assetId, asset] of Object.entries(this.manifest[assetType] || {})) {\n        const filteredDestinations = filterDict(asset.destinations, (_, destId) =>\n          selection.some((sel) => sel.matches(new DestinationIdentifier(assetId, destId))),\n        );\n\n        if (Object.keys(filteredDestinations).length > 0) {\n          ret[assetType][assetId] = {\n            ...asset,\n            destinations: filteredDestinations,\n          };\n        }\n      }\n    }\n\n    return new AssetManifest(this.directory, ret);\n  }\n\n  /**\n   * Describe the asset manifest as a list of strings\n   */\n  public list() {\n    return [\n      ...describeAssets('file', this.manifest.files || {}),\n      ...describeAssets('docker-image', this.manifest.dockerImages || {}),\n    ];\n\n    function describeAssets(\n      type: string,\n      assets: Record<string, { source: any; destinations: Record<string, any> }>,\n    ) {\n      const ret = new Array<string>();\n      for (const [assetId, asset] of Object.entries(assets || {})) {\n        ret.push(`${assetId} ${type} ${JSON.stringify(asset.source)}`);\n\n        const destStrings = Object.entries(asset.destinations).map(\n          ([destId, dest]) => ` ${assetId}:${destId} ${JSON.stringify(dest)}`,\n        );\n        ret.push(...prefixTreeChars(destStrings, '  '));\n      }\n      return ret;\n    }\n  }\n\n  /**\n   * List of assets per destination\n   *\n   * Returns one asset for every publishable destination. Multiple asset\n   * destinations may share the same asset source.\n   */\n  public get entries(): IManifestEntry[] {\n    return [\n      ...makeEntries(this.manifest.files || {}, FileManifestEntry),\n      ...makeEntries(this.manifest.dockerImages || {}, DockerImageManifestEntry),\n    ];\n  }\n\n  /**\n   * List of file assets, splat out to destinations\n   */\n  public get files(): FileManifestEntry[] {\n    return makeEntries(this.manifest.files || {}, FileManifestEntry);\n  }\n}\n\nfunction makeEntries<A, B, C>(\n  assets: Record<string, { source: A; displayName?: string; destinations: Record<string, B> }>,\n  ctor: new (id: DestinationIdentifier, displayName: string | undefined, source: A, destination: B) => C,\n): C[] {\n  const ret = new Array<C>();\n  for (const [assetId, asset] of Object.entries(assets)) {\n    for (const [destId, destination] of Object.entries(asset.destinations)) {\n      ret.push(new ctor(new DestinationIdentifier(assetId, destId), asset.displayName, asset.source, destination));\n    }\n  }\n  return ret;\n}\n\ntype AssetType = 'files' | 'dockerImages';\n\nconst ASSET_TYPES: AssetType[] = ['files', 'dockerImages'];\n\n/**\n * A single asset from an asset manifest'\n */\nexport interface IManifestEntry {\n  /**\n   * The identifier of the asset and its destination\n   */\n  readonly id: DestinationIdentifier;\n\n  /**\n   * The type of asset\n   */\n  readonly type: string;\n\n  /**\n   * Type-dependent source data\n   */\n  readonly genericSource: unknown;\n\n  /**\n   * Type-dependent destination data\n   */\n  readonly genericDestination: unknown;\n\n  /**\n   * Return a display name for this asset\n   *\n   * The `includeDestination` parameter controls whether or not to include the\n   * destination ID in the display name.\n   *\n   * - Pass `false` if you are displaying notifications about building the\n   *   asset, or if you are describing the work of building the asset and publishing\n   *   to all destinations at the same time.\n   * - Pass `true` if you are displaying notifications about publishing to a\n   *   specific destination.\n   */\n  displayName(includeDestination: boolean): string;\n}\n\n/**\n * A manifest entry for a file asset\n */\nexport class FileManifestEntry implements IManifestEntry {\n  public readonly genericSource: unknown;\n  public readonly genericDestination: unknown;\n  public readonly type = 'file';\n\n  constructor(\n    /** Identifier for this asset */\n    public readonly id: DestinationIdentifier,\n    private readonly _displayName: string | undefined,\n    /** Source of the file asset */\n    public readonly source: FileSource,\n    /** Destination for the file asset */\n    public readonly destination: FileDestination,\n  ) {\n    this.genericSource = source;\n    this.genericDestination = destination;\n  }\n\n  public displayName(includeDestination: boolean): string {\n    if (includeDestination) {\n      return this._displayName ? `${this._displayName} (${this.id.destinationId})` : `${this.id}`;\n    } else {\n      return this._displayName ? this._displayName : this.id.assetId;\n    }\n  }\n}\n\n/**\n * A manifest entry for a docker image asset\n */\nexport class DockerImageManifestEntry implements IManifestEntry {\n  public readonly genericSource: unknown;\n  public readonly genericDestination: unknown;\n  public readonly type = 'docker-image';\n\n  constructor(\n    /** Identifier for this asset */\n    public readonly id: DestinationIdentifier,\n    private readonly _displayName: string | undefined,\n    /** Source of the file asset */\n    public readonly source: DockerImageSource,\n    /** Destination for the file asset */\n    public readonly destination: DockerImageDestination,\n  ) {\n    this.genericSource = source;\n    this.genericDestination = destination;\n  }\n\n  public displayName(includeDestination: boolean): string {\n    if (includeDestination) {\n      return this._displayName ? `${this._displayName} (${this.id.destinationId})` : `${this.id}`;\n    } else {\n      return this._displayName ? this._displayName : this.id.assetId;\n    }\n  }\n}\n\n/**\n * Identify an asset destination in an asset manifest\n *\n * This class is used to identify both an asset to be built as well as a\n * destination where an asset will be published. However, when reasoning about\n * building assets the destination part can be ignored, because the same asset\n * being sent to multiple destinations will only need to be built once and their\n * assetIds are all the same.\n *\n * When stringified, this will be a combination of the source and destination\n * IDs; if a string representation of the source is necessary, use `id.assetId`\n * instead.\n */\nexport class DestinationIdentifier {\n  /**\n   * Identifies the asset, by source.\n   *\n   * The assetId will be the same between assets that represent\n   * the same physical file or image.\n   */\n  public readonly assetId: string;\n\n  /**\n   * Identifies the destination where this asset will be published\n   */\n  public readonly destinationId: string;\n\n  constructor(assetId: string, destinationId: string) {\n    this.assetId = assetId;\n    this.destinationId = destinationId;\n  }\n\n  /**\n   * Return a string representation for this asset identifier\n   */\n  public toString() {\n    return this.destinationId ? `${this.assetId}:${this.destinationId}` : this.assetId;\n  }\n}\n\nfunction filterDict<A>(\n  xs: Record<string, A>,\n  pred: (x: A, key: string) => boolean,\n): Record<string, A> {\n  const ret: Record<string, A> = {};\n  for (const [key, value] of Object.entries(xs)) {\n    if (pred(value, key)) {\n      ret[key] = value;\n    }\n  }\n  return ret;\n}\n\n/**\n * A filter pattern for an destination identifier\n */\nexport class DestinationPattern {\n  /**\n   * Parse a ':'-separated string into an asset/destination identifier\n   */\n  public static parse(s: string) {\n    if (!s) {\n      throw new Error('Empty string is not a valid destination identifier');\n    }\n    const parts = s.split(':').map((x) => (x !== '*' ? x : undefined));\n    if (parts.length === 1) {\n      return new DestinationPattern(parts[0]);\n    }\n    if (parts.length === 2) {\n      return new DestinationPattern(parts[0] || undefined, parts[1] || undefined);\n    }\n    throw new Error(`Asset identifier must contain at most 2 ':'-separated parts, got '${s}'`);\n  }\n\n  /**\n   * Identifies the asset, by source.\n   */\n  public readonly assetId?: string;\n\n  /**\n   * Identifies the destination where this asset will be published\n   */\n  public readonly destinationId?: string;\n\n  constructor(assetId?: string, destinationId?: string) {\n    this.assetId = assetId;\n    this.destinationId = destinationId;\n  }\n\n  /**\n   * Whether or not this pattern matches the given identifier\n   */\n  public matches(id: DestinationIdentifier) {\n    return (\n      (this.assetId === undefined || this.assetId === id.assetId) &&\n      (this.destinationId === undefined || this.destinationId === id.destinationId)\n    );\n  }\n\n  /**\n   * Return a string representation for this asset identifier\n   */\n  public toString() {\n    return `${this.assetId ?? '*'}:${this.destinationId ?? '*'}`;\n  }\n}\n\n/**\n * Prefix box-drawing characters to make lines look like a hanging tree\n */\nfunction prefixTreeChars(xs: string[], prefix = '') {\n  const ret = new Array<string>();\n  for (let i = 0; i < xs.length; i++) {\n    const isLast = i === xs.length - 1;\n    const boxChar = isLast ? '└' : '├';\n    ret.push(`${prefix}${boxChar}${xs[i]}`);\n  }\n  return ret;\n}\n"]}
|
package/lib/aws.d.ts
CHANGED
|
@@ -57,6 +57,7 @@ export declare class DefaultAwsClient implements IAws {
|
|
|
57
57
|
private readonly profile?;
|
|
58
58
|
private account?;
|
|
59
59
|
private config;
|
|
60
|
+
private readonly mainCredentials;
|
|
60
61
|
constructor(profile?: string | undefined);
|
|
61
62
|
s3Client(options: ClientOptions): Promise<IS3Client>;
|
|
62
63
|
ecrClient(options: ClientOptions): Promise<IECRClient>;
|
package/lib/aws.js
CHANGED
|
@@ -20,12 +20,16 @@ class DefaultAwsClient {
|
|
|
20
20
|
const clientConfig = {
|
|
21
21
|
customUserAgent: USER_AGENT,
|
|
22
22
|
};
|
|
23
|
+
// storing the main credentials separately because
|
|
24
|
+
// the `config` object changes every time we assume the file publishing role.
|
|
25
|
+
// TODO refactor to make `config` a readonly property and avoid state mutations.
|
|
26
|
+
this.mainCredentials = (0, credential_providers_1.fromNodeProviderChain)({
|
|
27
|
+
profile: this.profile,
|
|
28
|
+
clientConfig,
|
|
29
|
+
});
|
|
23
30
|
this.config = {
|
|
24
31
|
clientConfig,
|
|
25
|
-
credentials:
|
|
26
|
-
profile: this.profile,
|
|
27
|
-
clientConfig,
|
|
28
|
-
}),
|
|
32
|
+
credentials: this.mainCredentials,
|
|
29
33
|
};
|
|
30
34
|
}
|
|
31
35
|
async s3Client(options) {
|
|
@@ -92,6 +96,8 @@ class DefaultAwsClient {
|
|
|
92
96
|
config.region = options.region;
|
|
93
97
|
if (options.assumeRoleArn) {
|
|
94
98
|
config.credentials = (0, credential_providers_1.fromTemporaryCredentials)({
|
|
99
|
+
// dont forget the credentials chain.
|
|
100
|
+
masterCredentials: this.mainCredentials,
|
|
95
101
|
params: {
|
|
96
102
|
RoleArn: options.assumeRoleArn,
|
|
97
103
|
ExternalId: options.assumeRoleExternalId,
|
|
@@ -122,4 +128,4 @@ function safeUsername() {
|
|
|
122
128
|
return 'noname';
|
|
123
129
|
}
|
|
124
130
|
}
|
|
125
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws.js","sourceRoot":"","sources":["aws.ts"],"names":[],"mappings":"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","sourcesContent":["import * as os from 'os';\nimport {\n  DescribeImagesCommand,\n  DescribeRepositoriesCommand,\n  ECRClient,\n  GetAuthorizationTokenCommand,\n} from '@aws-sdk/client-ecr';\nimport {\n  GetBucketEncryptionCommand,\n  GetBucketLocationCommand,\n  ListObjectsV2Command,\n  S3Client,\n} from '@aws-sdk/client-s3';\nimport { GetSecretValueCommand, SecretsManagerClient } from '@aws-sdk/client-secrets-manager';\nimport {\n  AssumeRoleCommandInput,\n  GetCallerIdentityCommand,\n  STSClient,\n  STSClientConfig,\n} from '@aws-sdk/client-sts';\nimport { fromNodeProviderChain, fromTemporaryCredentials } from '@aws-sdk/credential-providers';\nimport { Upload } from '@aws-sdk/lib-storage';\nimport {\n  NODE_REGION_CONFIG_FILE_OPTIONS,\n  NODE_REGION_CONFIG_OPTIONS,\n} from '@smithy/config-resolver';\nimport { loadConfig } from '@smithy/node-config-provider';\nimport {\n  AwsCredentialIdentityProvider,\n  CompleteMultipartUploadCommandOutput,\n  DescribeImagesCommandInput,\n  DescribeImagesCommandOutput,\n  DescribeRepositoriesCommandInput,\n  DescribeRepositoriesCommandOutput,\n  GetAuthorizationTokenCommandInput,\n  GetAuthorizationTokenCommandOutput,\n  GetBucketEncryptionCommandInput,\n  GetBucketEncryptionCommandOutput,\n  GetBucketLocationCommandInput,\n  GetBucketLocationCommandOutput,\n  GetSecretValueCommandInput,\n  GetSecretValueCommandOutput,\n  ListObjectsV2CommandInput,\n  ListObjectsV2CommandOutput,\n  PutObjectCommandInput,\n} from './aws-types';\n\nexport type AssumeRoleAdditionalOptions = Partial<\n  Omit<AssumeRoleCommandInput, 'ExternalId' | 'RoleArn'>\n>;\n\nexport interface IS3Client {\n  getBucketEncryption(\n    input: GetBucketEncryptionCommandInput\n  ): Promise<GetBucketEncryptionCommandOutput>;\n  getBucketLocation(input: GetBucketLocationCommandInput): Promise<GetBucketLocationCommandOutput>;\n  listObjectsV2(input: ListObjectsV2CommandInput): Promise<ListObjectsV2CommandOutput>;\n  upload(input: PutObjectCommandInput): Promise<CompleteMultipartUploadCommandOutput>;\n}\n\nexport interface IECRClient {\n  describeImages(input: DescribeImagesCommandInput): Promise<DescribeImagesCommandOutput>;\n  describeRepositories(\n    input: DescribeRepositoriesCommandInput\n  ): Promise<DescribeRepositoriesCommandOutput>;\n  getAuthorizationToken(\n    input?: GetAuthorizationTokenCommandInput\n  ): Promise<GetAuthorizationTokenCommandOutput>;\n}\n\nexport interface ISecretsManagerClient {\n  getSecretValue(input: GetSecretValueCommandInput): Promise<GetSecretValueCommandOutput>;\n}\n\n/**\n * AWS SDK operations required by Asset Publishing\n */\nexport interface IAws {\n  discoverPartition(): Promise<string>;\n  discoverDefaultRegion(): Promise<string>;\n  discoverCurrentAccount(): Promise<Account>;\n\n  discoverTargetAccount(options: ClientOptions): Promise<Account>;\n  s3Client(options: ClientOptions): Promise<IS3Client>;\n  ecrClient(options: ClientOptions): Promise<IECRClient>;\n  secretsManagerClient(options: ClientOptions): Promise<ISecretsManagerClient>;\n}\n\nexport interface ClientOptions {\n  region?: string;\n  assumeRoleArn?: string;\n  assumeRoleExternalId?: string;\n  assumeRoleAdditionalOptions?: AssumeRoleAdditionalOptions;\n  quiet?: boolean;\n}\n\nconst USER_AGENT = 'cdk-assets';\n\ninterface Configuration {\n  clientConfig: STSClientConfig;\n  region?: string;\n  credentials: AwsCredentialIdentityProvider;\n}\n\n/**\n * An AWS account\n *\n * An AWS account always exists in only one partition. Usually we don't care about\n * the partition, but when we need to form ARNs we do.\n */\nexport interface Account {\n  /**\n   * The account number\n   */\n  readonly accountId: string;\n\n  /**\n   * The partition ('aws' or 'aws-cn' or otherwise)\n   */\n  readonly partition: string;\n}\n\n/**\n * AWS client using the AWS SDK for JS with no special configuration\n */\nexport class DefaultAwsClient implements IAws {\n  private account?: Account;\n  private config: Configuration;\n\n  constructor(private readonly profile?: string) {\n    const clientConfig: STSClientConfig = {\n      customUserAgent: USER_AGENT,\n    };\n    this.config = {\n      clientConfig,\n      credentials: fromNodeProviderChain({\n        profile: this.profile,\n        clientConfig,\n      }),\n    };\n  }\n\n  public async s3Client(options: ClientOptions): Promise<IS3Client> {\n    const client = new S3Client(await this.awsOptions(options));\n    return {\n      getBucketEncryption: (\n        input: GetBucketEncryptionCommandInput,\n      ): Promise<GetBucketEncryptionCommandOutput> =>\n        client.send(new GetBucketEncryptionCommand(input)),\n      getBucketLocation: (\n        input: GetBucketLocationCommandInput,\n      ): Promise<GetBucketLocationCommandOutput> =>\n        client.send(new GetBucketLocationCommand(input)),\n      listObjectsV2: (input: ListObjectsV2CommandInput): Promise<ListObjectsV2CommandOutput> =>\n        client.send(new ListObjectsV2Command(input)),\n      upload: (input: PutObjectCommandInput): Promise<CompleteMultipartUploadCommandOutput> => {\n        const upload = new Upload({\n          client,\n          params: input,\n        });\n        return upload.done();\n      },\n    };\n  }\n\n  public async ecrClient(options: ClientOptions): Promise<IECRClient> {\n    const client = new ECRClient(await this.awsOptions(options));\n    return {\n      describeImages: (input: DescribeImagesCommandInput): Promise<DescribeImagesCommandOutput> =>\n        client.send(new DescribeImagesCommand(input)),\n      describeRepositories: (\n        input: DescribeRepositoriesCommandInput,\n      ): Promise<DescribeRepositoriesCommandOutput> =>\n        client.send(new DescribeRepositoriesCommand(input)),\n      getAuthorizationToken: (\n        input: GetAuthorizationTokenCommandInput,\n      ): Promise<GetAuthorizationTokenCommandOutput> =>\n        client.send(new GetAuthorizationTokenCommand(input ?? {})),\n    };\n  }\n\n  public async secretsManagerClient(options: ClientOptions): Promise<ISecretsManagerClient> {\n    const client = new SecretsManagerClient(await this.awsOptions(options));\n    return {\n      getSecretValue: (input: GetSecretValueCommandInput): Promise<GetSecretValueCommandOutput> =>\n        client.send(new GetSecretValueCommand(input)),\n    };\n  }\n\n  public async discoverPartition(): Promise<string> {\n    return (await this.discoverCurrentAccount()).partition;\n  }\n\n  public async discoverDefaultRegion(): Promise<string> {\n    return loadConfig(NODE_REGION_CONFIG_OPTIONS, NODE_REGION_CONFIG_FILE_OPTIONS)() || 'us-east-1';\n  }\n\n  public async discoverCurrentAccount(): Promise<Account> {\n    if (this.account === undefined) {\n      this.account = await this.getAccount();\n    }\n    return this.account;\n  }\n\n  public async discoverTargetAccount(options: ClientOptions): Promise<Account> {\n    return this.getAccount(await this.awsOptions(options));\n  }\n\n  private async getAccount(options?: ClientOptions): Promise<Account> {\n    this.config.clientConfig = options ?? this.config.clientConfig;\n    const stsClient = new STSClient(await this.awsOptions(options));\n\n    const command = new GetCallerIdentityCommand();\n    const response = await stsClient.send(command);\n    if (!response.Account || !response.Arn) {\n      throw new Error(`Unrecognized response from STS: '${JSON.stringify(response)}'`);\n    }\n    return {\n      accountId: response.Account!,\n      partition: response.Arn!.split(':')[1],\n    };\n  }\n\n  private async awsOptions(options?: ClientOptions) {\n    const config = this.config;\n    config.region = options?.region;\n    if (options) {\n      config.region = options.region;\n      if (options.assumeRoleArn) {\n        config.credentials = fromTemporaryCredentials({\n          params: {\n            RoleArn: options.assumeRoleArn,\n            ExternalId: options.assumeRoleExternalId,\n            RoleSessionName: `${USER_AGENT}-${safeUsername()}`,\n            TransitiveTagKeys: options.assumeRoleAdditionalOptions?.Tags\n              ? options.assumeRoleAdditionalOptions.Tags.map((t) => t.Key!)\n              : undefined,\n            ...options.assumeRoleAdditionalOptions,\n          },\n          clientConfig: this.config.clientConfig,\n        });\n      }\n    }\n    return config;\n  }\n}\n\n/**\n * Return the username with characters invalid for a RoleSessionName removed\n *\n * @see https://docs.aws.amazon.com/STS/latest/APIReference/API_AssumeRole.html#API_AssumeRole_RequestParameters\n */\nfunction safeUsername() {\n  try {\n    return os.userInfo().username.replace(/[^\\w+=,.@-]/g, '@');\n  } catch {\n    return 'noname';\n  }\n}\n"]}
|
|
131
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws.js","sourceRoot":"","sources":["aws.ts"],"names":[],"mappings":"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","sourcesContent":["import * as os from 'os';\nimport {\n  DescribeImagesCommand,\n  DescribeRepositoriesCommand,\n  ECRClient,\n  GetAuthorizationTokenCommand,\n} from '@aws-sdk/client-ecr';\nimport {\n  GetBucketEncryptionCommand,\n  GetBucketLocationCommand,\n  ListObjectsV2Command,\n  S3Client,\n} from '@aws-sdk/client-s3';\nimport { GetSecretValueCommand, SecretsManagerClient } from '@aws-sdk/client-secrets-manager';\nimport {\n  AssumeRoleCommandInput,\n  GetCallerIdentityCommand,\n  STSClient,\n  STSClientConfig,\n} from '@aws-sdk/client-sts';\nimport { fromNodeProviderChain, fromTemporaryCredentials } from '@aws-sdk/credential-providers';\nimport { Upload } from '@aws-sdk/lib-storage';\nimport {\n  NODE_REGION_CONFIG_FILE_OPTIONS,\n  NODE_REGION_CONFIG_OPTIONS,\n} from '@smithy/config-resolver';\nimport { loadConfig } from '@smithy/node-config-provider';\nimport {\n  AwsCredentialIdentityProvider,\n  CompleteMultipartUploadCommandOutput,\n  DescribeImagesCommandInput,\n  DescribeImagesCommandOutput,\n  DescribeRepositoriesCommandInput,\n  DescribeRepositoriesCommandOutput,\n  GetAuthorizationTokenCommandInput,\n  GetAuthorizationTokenCommandOutput,\n  GetBucketEncryptionCommandInput,\n  GetBucketEncryptionCommandOutput,\n  GetBucketLocationCommandInput,\n  GetBucketLocationCommandOutput,\n  GetSecretValueCommandInput,\n  GetSecretValueCommandOutput,\n  ListObjectsV2CommandInput,\n  ListObjectsV2CommandOutput,\n  PutObjectCommandInput,\n} from './aws-types';\n\nexport type AssumeRoleAdditionalOptions = Partial<\n  Omit<AssumeRoleCommandInput, 'ExternalId' | 'RoleArn'>\n>;\n\nexport interface IS3Client {\n  getBucketEncryption(\n    input: GetBucketEncryptionCommandInput\n  ): Promise<GetBucketEncryptionCommandOutput>;\n  getBucketLocation(input: GetBucketLocationCommandInput): Promise<GetBucketLocationCommandOutput>;\n  listObjectsV2(input: ListObjectsV2CommandInput): Promise<ListObjectsV2CommandOutput>;\n  upload(input: PutObjectCommandInput): Promise<CompleteMultipartUploadCommandOutput>;\n}\n\nexport interface IECRClient {\n  describeImages(input: DescribeImagesCommandInput): Promise<DescribeImagesCommandOutput>;\n  describeRepositories(\n    input: DescribeRepositoriesCommandInput\n  ): Promise<DescribeRepositoriesCommandOutput>;\n  getAuthorizationToken(\n    input?: GetAuthorizationTokenCommandInput\n  ): Promise<GetAuthorizationTokenCommandOutput>;\n}\n\nexport interface ISecretsManagerClient {\n  getSecretValue(input: GetSecretValueCommandInput): Promise<GetSecretValueCommandOutput>;\n}\n\n/**\n * AWS SDK operations required by Asset Publishing\n */\nexport interface IAws {\n  discoverPartition(): Promise<string>;\n  discoverDefaultRegion(): Promise<string>;\n  discoverCurrentAccount(): Promise<Account>;\n\n  discoverTargetAccount(options: ClientOptions): Promise<Account>;\n  s3Client(options: ClientOptions): Promise<IS3Client>;\n  ecrClient(options: ClientOptions): Promise<IECRClient>;\n  secretsManagerClient(options: ClientOptions): Promise<ISecretsManagerClient>;\n}\n\nexport interface ClientOptions {\n  region?: string;\n  assumeRoleArn?: string;\n  assumeRoleExternalId?: string;\n  assumeRoleAdditionalOptions?: AssumeRoleAdditionalOptions;\n  quiet?: boolean;\n}\n\nconst USER_AGENT = 'cdk-assets';\n\ninterface Configuration {\n  clientConfig: STSClientConfig;\n  region?: string;\n  credentials: AwsCredentialIdentityProvider;\n}\n\n/**\n * An AWS account\n *\n * An AWS account always exists in only one partition. Usually we don't care about\n * the partition, but when we need to form ARNs we do.\n */\nexport interface Account {\n  /**\n   * The account number\n   */\n  readonly accountId: string;\n\n  /**\n   * The partition ('aws' or 'aws-cn' or otherwise)\n   */\n  readonly partition: string;\n}\n\n/**\n * AWS client using the AWS SDK for JS with no special configuration\n */\nexport class DefaultAwsClient implements IAws {\n  private account?: Account;\n  private config: Configuration;\n  private readonly mainCredentials: AwsCredentialIdentityProvider;\n\n  constructor(private readonly profile?: string) {\n    const clientConfig: STSClientConfig = {\n      customUserAgent: USER_AGENT,\n    };\n\n    // storing the main credentials separately because\n    // the `config` object changes every time we assume the file publishing role.\n    // TODO refactor to make `config` a readonly property and avoid state mutations.\n    this.mainCredentials = fromNodeProviderChain({\n      profile: this.profile,\n      clientConfig,\n    });\n\n    this.config = {\n      clientConfig,\n      credentials: this.mainCredentials,\n    };\n  }\n\n  public async s3Client(options: ClientOptions): Promise<IS3Client> {\n    const client = new S3Client(await this.awsOptions(options));\n    return {\n      getBucketEncryption: (\n        input: GetBucketEncryptionCommandInput,\n      ): Promise<GetBucketEncryptionCommandOutput> =>\n        client.send(new GetBucketEncryptionCommand(input)),\n      getBucketLocation: (\n        input: GetBucketLocationCommandInput,\n      ): Promise<GetBucketLocationCommandOutput> =>\n        client.send(new GetBucketLocationCommand(input)),\n      listObjectsV2: (input: ListObjectsV2CommandInput): Promise<ListObjectsV2CommandOutput> =>\n        client.send(new ListObjectsV2Command(input)),\n      upload: (input: PutObjectCommandInput): Promise<CompleteMultipartUploadCommandOutput> => {\n        const upload = new Upload({\n          client,\n          params: input,\n        });\n        return upload.done();\n      },\n    };\n  }\n\n  public async ecrClient(options: ClientOptions): Promise<IECRClient> {\n    const client = new ECRClient(await this.awsOptions(options));\n    return {\n      describeImages: (input: DescribeImagesCommandInput): Promise<DescribeImagesCommandOutput> =>\n        client.send(new DescribeImagesCommand(input)),\n      describeRepositories: (\n        input: DescribeRepositoriesCommandInput,\n      ): Promise<DescribeRepositoriesCommandOutput> =>\n        client.send(new DescribeRepositoriesCommand(input)),\n      getAuthorizationToken: (\n        input: GetAuthorizationTokenCommandInput,\n      ): Promise<GetAuthorizationTokenCommandOutput> =>\n        client.send(new GetAuthorizationTokenCommand(input ?? {})),\n    };\n  }\n\n  public async secretsManagerClient(options: ClientOptions): Promise<ISecretsManagerClient> {\n    const client = new SecretsManagerClient(await this.awsOptions(options));\n    return {\n      getSecretValue: (input: GetSecretValueCommandInput): Promise<GetSecretValueCommandOutput> =>\n        client.send(new GetSecretValueCommand(input)),\n    };\n  }\n\n  public async discoverPartition(): Promise<string> {\n    return (await this.discoverCurrentAccount()).partition;\n  }\n\n  public async discoverDefaultRegion(): Promise<string> {\n    return loadConfig(NODE_REGION_CONFIG_OPTIONS, NODE_REGION_CONFIG_FILE_OPTIONS)() || 'us-east-1';\n  }\n\n  public async discoverCurrentAccount(): Promise<Account> {\n    if (this.account === undefined) {\n      this.account = await this.getAccount();\n    }\n    return this.account;\n  }\n\n  public async discoverTargetAccount(options: ClientOptions): Promise<Account> {\n    return this.getAccount(await this.awsOptions(options));\n  }\n\n  private async getAccount(options?: ClientOptions): Promise<Account> {\n    this.config.clientConfig = options ?? this.config.clientConfig;\n    const stsClient = new STSClient(await this.awsOptions(options));\n\n    const command = new GetCallerIdentityCommand();\n    const response = await stsClient.send(command);\n    if (!response.Account || !response.Arn) {\n      throw new Error(`Unrecognized response from STS: '${JSON.stringify(response)}'`);\n    }\n    return {\n      accountId: response.Account!,\n      partition: response.Arn!.split(':')[1],\n    };\n  }\n\n  private async awsOptions(options?: ClientOptions) {\n    const config = this.config;\n    config.region = options?.region;\n    if (options) {\n      config.region = options.region;\n      if (options.assumeRoleArn) {\n        config.credentials = fromTemporaryCredentials({\n          // dont forget the credentials chain.\n          masterCredentials: this.mainCredentials,\n          params: {\n            RoleArn: options.assumeRoleArn,\n            ExternalId: options.assumeRoleExternalId,\n            RoleSessionName: `${USER_AGENT}-${safeUsername()}`,\n            TransitiveTagKeys: options.assumeRoleAdditionalOptions?.Tags\n              ? options.assumeRoleAdditionalOptions.Tags.map((t) => t.Key!)\n              : undefined,\n            ...options.assumeRoleAdditionalOptions,\n          },\n          clientConfig: this.config.clientConfig,\n        });\n      }\n    }\n    return config;\n  }\n}\n\n/**\n * Return the username with characters invalid for a RoleSessionName removed\n *\n * @see https://docs.aws.amazon.com/STS/latest/APIReference/API_AssumeRole.html#API_AssumeRole_RequestParameters\n */\nfunction safeUsername() {\n  try {\n    return os.userInfo().username.replace(/[^\\w+=,.@-]/g, '@');\n  } catch {\n    return 'noname';\n  }\n}\n"]}
|
|
@@ -12,6 +12,12 @@ export interface PublishOptions {
|
|
|
12
12
|
* @default true
|
|
13
13
|
*/
|
|
14
14
|
readonly allowCrossAccount?: boolean;
|
|
15
|
+
/**
|
|
16
|
+
* Always upload, even if the target file already exists
|
|
17
|
+
*
|
|
18
|
+
* @default false
|
|
19
|
+
*/
|
|
20
|
+
readonly force?: boolean;
|
|
15
21
|
}
|
|
16
22
|
/**
|
|
17
23
|
* Handler for asset building and publishing.
|
|
@@ -1,3 +1,3 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXNzZXQtaGFuZGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImFzc2V0LWhhbmRsZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERvY2tlckZhY3RvcnkgfSBmcm9tICcuL2RvY2tlcic7XG5pbXBvcnQgeyBJQXdzIH0gZnJvbSAnLi4vYXdzJztcbmltcG9ydCB7IEV2ZW50RW1pdHRlciB9IGZyb20gJy4uL3Byb2dyZXNzJztcblxuLyoqXG4gKiBPcHRpb25zIGZvciBwdWJsaXNoaW5nIGFuIGFzc2V0LlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFB1Ymxpc2hPcHRpb25zIHtcbiAgLyoqXG4gICAqIFdoZXRoZXIgb3Igbm90IHRvIGFsbG93IGNyb3NzIGFjY291bnQgcHVibGlzaGluZy4gVGhhdCBpcyxcbiAgICogcHVibGlzaCB0byBhIGJ1Y2tldCBiZWxvbmdpbmcgdG8gYSBkaWZmZXJlbnQgYWNjb3VudCB0aGFuIHRoZSB0YXJnZXQgYWNjb3VudC5cbiAgICpcbiAgICogQGRlZmF1bHQgdHJ1ZVxuICAgKi9cbiAgcmVhZG9ubHkgYWxsb3dDcm9zc0FjY291bnQ/OiBib29sZWFuO1xufVxuXG4vKipcbiAqIEhhbmRsZXIgZm9yIGFzc2V0IGJ1aWxkaW5nIGFuZCBwdWJsaXNoaW5nLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIElBc3NldEhhbmRsZXIge1xuICAvKipcbiAgICogQnVpbGQgdGhlIGFzc2V0LlxuICAgKi9cbiAgYnVpbGQoKTogUHJvbWlzZTx2b2lkPjtcblxuICAvKipcbiAgICogUHVibGlzaCB0aGUgYXNzZXQuXG4gICAqL1xuICBwdWJsaXNoKG9wdGlvbnM/OiBQdWJsaXNoT3B0aW9ucyk6IFByb21pc2U8dm9pZD47XG5cbiAgLyoqXG4gICAqIFJldHVybiB3aGV0aGVyIHRoZSBhc3NldCBhbHJlYWR5IGV4aXN0c1xuICAgKi9cbiAgaXNQdWJsaXNoZWQoKTogUHJvbWlzZTxib29sZWFuPjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBJSGFuZGxlckhvc3Qge1xuICByZWFkb25seSBhd3M6IElBd3M7XG4gIHJlYWRvbmx5IGFib3J0ZWQ6IGJvb2xlYW47XG4gIHJlYWRvbmx5IGRvY2tlckZhY3Rvcnk6IERvY2tlckZhY3Rvcnk7XG5cbiAgZW1pdE1lc3NhZ2U6IEV2ZW50RW1pdHRlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBJSGFuZGxlck9wdGlvbnMge1xuICAvKipcbiAgICogV2hlcmUgdG8gc2VuZCBvdXRwdXQgb2YgYSBzdWJwcm9jZXNzZXNcbiAgICpcbiAgICogQGRlZmF1bHQgJ3N0ZGlvJ1xuICAgKi9cbiAgcmVhZG9ubHkgc3VicHJvY2Vzc091dHB1dERlc3RpbmF0aW9uOiBTdWJwcm9jZXNzT3V0cHV0RGVzdGluYXRpb247XG59XG5cbi8qKlxuICogVGhlIHBvdGVudGlhbCBkZXN0aW5hdGlvbnMgZm9yIHN1YnByb2Nlc3Mgb3V0cHV0LlxuICpcbiAqICdzdGRpbycgd2lsbCBzZW5kIG91dHB1dCBkaXJlY3RseSB0byBzdGRvdXQvc3RkZXJyLFxuICogJ3B1Ymxpc2gnIHdpbGwgcHVibGlzaCB0aGUgb3V0cHV0IHRvIHRoZSB7QGxpbmsgSVB1Ymxpc2hQcm9ncmVzc0xpc3RlbmVyfSxcbiAqICdpZ25vcmUnIHdpbGwgaWdub3JlIHRoZSBvdXRwdXQsIGFuZCBlbWl0IGl0IG5vd2hlcmUuXG4gKi9cbmV4cG9ydCB0eXBlIFN1YnByb2Nlc3NPdXRwdXREZXN0aW5hdGlvbiA9ICdzdGRpbycgfCAnaWdub3JlJyB8ICdwdWJsaXNoJztcbiJdfQ==
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXNzZXQtaGFuZGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImFzc2V0LWhhbmRsZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERvY2tlckZhY3RvcnkgfSBmcm9tICcuL2RvY2tlcic7XG5pbXBvcnQgeyBJQXdzIH0gZnJvbSAnLi4vYXdzJztcbmltcG9ydCB7IEV2ZW50RW1pdHRlciB9IGZyb20gJy4uL3Byb2dyZXNzJztcblxuLyoqXG4gKiBPcHRpb25zIGZvciBwdWJsaXNoaW5nIGFuIGFzc2V0LlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFB1Ymxpc2hPcHRpb25zIHtcbiAgLyoqXG4gICAqIFdoZXRoZXIgb3Igbm90IHRvIGFsbG93IGNyb3NzIGFjY291bnQgcHVibGlzaGluZy4gVGhhdCBpcyxcbiAgICogcHVibGlzaCB0byBhIGJ1Y2tldCBiZWxvbmdpbmcgdG8gYSBkaWZmZXJlbnQgYWNjb3VudCB0aGFuIHRoZSB0YXJnZXQgYWNjb3VudC5cbiAgICpcbiAgICogQGRlZmF1bHQgdHJ1ZVxuICAgKi9cbiAgcmVhZG9ubHkgYWxsb3dDcm9zc0FjY291bnQ/OiBib29sZWFuO1xuXG4gIC8qKlxuICAgKiBBbHdheXMgdXBsb2FkLCBldmVuIGlmIHRoZSB0YXJnZXQgZmlsZSBhbHJlYWR5IGV4aXN0c1xuICAgKlxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKi9cbiAgcmVhZG9ubHkgZm9yY2U/OiBib29sZWFuO1xufVxuXG4vKipcbiAqIEhhbmRsZXIgZm9yIGFzc2V0IGJ1aWxkaW5nIGFuZCBwdWJsaXNoaW5nLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIElBc3NldEhhbmRsZXIge1xuICAvKipcbiAgICogQnVpbGQgdGhlIGFzc2V0LlxuICAgKi9cbiAgYnVpbGQoKTogUHJvbWlzZTx2b2lkPjtcblxuICAvKipcbiAgICogUHVibGlzaCB0aGUgYXNzZXQuXG4gICAqL1xuICBwdWJsaXNoKG9wdGlvbnM/OiBQdWJsaXNoT3B0aW9ucyk6IFByb21pc2U8dm9pZD47XG5cbiAgLyoqXG4gICAqIFJldHVybiB3aGV0aGVyIHRoZSBhc3NldCBhbHJlYWR5IGV4aXN0c1xuICAgKi9cbiAgaXNQdWJsaXNoZWQoKTogUHJvbWlzZTxib29sZWFuPjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBJSGFuZGxlckhvc3Qge1xuICByZWFkb25seSBhd3M6IElBd3M7XG4gIHJlYWRvbmx5IGFib3J0ZWQ6IGJvb2xlYW47XG4gIHJlYWRvbmx5IGRvY2tlckZhY3Rvcnk6IERvY2tlckZhY3Rvcnk7XG5cbiAgZW1pdE1lc3NhZ2U6IEV2ZW50RW1pdHRlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBJSGFuZGxlck9wdGlvbnMge1xuICAvKipcbiAgICogV2hlcmUgdG8gc2VuZCBvdXRwdXQgb2YgYSBzdWJwcm9jZXNzZXNcbiAgICpcbiAgICogQGRlZmF1bHQgJ3N0ZGlvJ1xuICAgKi9cbiAgcmVhZG9ubHkgc3VicHJvY2Vzc091dHB1dERlc3RpbmF0aW9uOiBTdWJwcm9jZXNzT3V0cHV0RGVzdGluYXRpb247XG59XG5cbi8qKlxuICogVGhlIHBvdGVudGlhbCBkZXN0aW5hdGlvbnMgZm9yIHN1YnByb2Nlc3Mgb3V0cHV0LlxuICpcbiAqICdzdGRpbycgd2lsbCBzZW5kIG91dHB1dCBkaXJlY3RseSB0byBzdGRvdXQvc3RkZXJyLFxuICogJ3B1Ymxpc2gnIHdpbGwgcHVibGlzaCB0aGUgb3V0cHV0IHRvIHRoZSB7QGxpbmsgSVB1Ymxpc2hQcm9ncmVzc0xpc3RlbmVyfSxcbiAqICdpZ25vcmUnIHdpbGwgaWdub3JlIHRoZSBvdXRwdXQsIGFuZCBlbWl0IGl0IG5vd2hlcmUuXG4gKi9cbmV4cG9ydCB0eXBlIFN1YnByb2Nlc3NPdXRwdXREZXN0aW5hdGlvbiA9ICdzdGRpbycgfCAnaWdub3JlJyB8ICdwdWJsaXNoJztcbiJdfQ==
|
|
@@ -85,7 +85,7 @@ class FileAssetHandler {
|
|
|
85
85
|
}
|
|
86
86
|
break;
|
|
87
87
|
}
|
|
88
|
-
if (await objectExists(s3, destination.bucketName, destination.objectKey)) {
|
|
88
|
+
if (!options.force && await objectExists(s3, destination.bucketName, destination.objectKey)) {
|
|
89
89
|
this.host.emitMessage(progress_1.EventType.FOUND, `Found ${s3Url}`);
|
|
90
90
|
return;
|
|
91
91
|
}
|
|
@@ -286,4 +286,4 @@ async function cached(cache, key, factory) {
|
|
|
286
286
|
cache.set(key, fresh);
|
|
287
287
|
return fresh;
|
|
288
288
|
}
|
|
289
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"files.js","sourceRoot":"","sources":["files.ts"],"names":[],"mappings":";;;AAAA,2BAAsD;AACtD,6BAA6B;AAC7B,0EAAgF;AAChF,6BAA6B;AAC7B,qDAA8D;AAI9D,6CAAgF;AAChF,wCAA0C;AAE1C,0CAAyC;AACzC,kDAAyD;AACzD,oCAAiC;AAEjC;;;;GAIG;AACH,MAAM,mBAAmB,GAAG,EAAE,CAAC;AAE/B,MAAa,gBAAgB;IAG3B,YACmB,OAAe,EACf,KAAwB,EACxB,IAAkB;QAFlB,YAAO,GAAP,OAAO,CAAQ;QACf,UAAK,GAAL,KAAK,CAAmB;QACxB,SAAI,GAAJ,IAAI,CAAc;QAEnC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;IACpD,CAAC;IAEM,KAAK,CAAC,KAAK;IAClB,CAAC;IAEM,KAAK,CAAC,WAAW;QACtB,MAAM,WAAW,GAAG,MAAM,IAAA,qCAAsB,EAAC,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACxF,MAAM,KAAK,GAAG,QAAQ,WAAW,CAAC,UAAU,IAAI,WAAW,CAAC,SAAS,EAAE,CAAC;QACxE,IAAI,CAAC;YACH,MAAM,EAAE,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC;gBACtC,GAAG,IAAA,2CAA0B,EAAC,WAAW,CAAC;gBAC1C,KAAK,EAAE,IAAI;aACZ,CAAC,CAAC;YACH,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,oBAAS,CAAC,KAAK,EAAE,SAAS,KAAK,EAAE,CAAC,CAAC;YAEzD,IAAI,MAAM,YAAY,CAAC,EAAE,EAAE,WAAW,CAAC,UAAU,EAAE,WAAW,CAAC,SAAS,CAAC,EAAE,CAAC;gBAC1E,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,oBAAS,CAAC,KAAK,EAAE,SAAS,KAAK,EAAE,CAAC,CAAC;gBACzD,OAAO,IAAI,CAAC;YACd,CAAC;QACH,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,oBAAS,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;QACzD,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAEM,KAAK,CAAC,OAAO,CAAC,UAA0B,EAAE;QAC/C,MAAM,WAAW,GAAG,MAAM,IAAA,qCAAsB,EAAC,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACxF,MAAM,KAAK,GAAG,QAAQ,WAAW,CAAC,UAAU,IAAI,WAAW,CAAC,SAAS,EAAE,CAAC;QAExE,MAAM,aAAa,GAAG,IAAA,2CAA0B,EAAC,WAAW,CAAC,CAAC;QAC9D,MAAM,EAAE,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC;QACvD,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,oBAAS,CAAC,KAAK,EAAE,SAAS,KAAK,EAAE,CAAC,CAAC;QAEzD,MAAM,UAAU,GAAG,iBAAiB,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAEpD,mFAAmF;QACnF,oCAAoC;QACpC,MAAM,OAAO,GAAG,KAAK,IAAI,EAAE,CACzB,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,qBAAqB,CAAC,aAAa,CAAC,CAAC,CAAC,SAAS,CAAC;QAEvE,MAAM,iBAAiB,GAAG,OAAO,CAAC,iBAAiB,IAAI,IAAI,CAAC;QAC5D,QACE,MAAM,UAAU,CAAC,eAAe,CAC9B,EAAE,EACF,WAAW,CAAC,UAAU,EACtB,iBAAiB,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,OAAO,EAAE,CAChD,EACD,CAAC;YACD,KAAK,eAAe,CAAC,IAAI;gBACvB,MAAM;YACR,KAAK,eAAe,CAAC,cAAc;gBACjC,MAAM,IAAI,KAAK,CACb,oBAAoB,WAAW,CAAC,UAAU,iBAAiB,MAAM,OAAO,EAAE,gBAAgB,CAC3F,CAAC;YACJ,KAAK,eAAe,CAAC,SAAS;gBAC5B,MAAM,IAAI,KAAK,CACb,iBAAiB,WAAW,CAAC,UAAU,0CAA0C,CAClF,CAAC;YACJ,KAAK,eAAe,CAAC,6BAA6B;gBAChD,IAAI,CAAC,iBAAiB,EAAE,CAAC;oBACvB,MAAM,IAAI,KAAK,CACb;;kDAEsC,WAAW,CAAC,UAAU;8CAC1B,MAAM,OAAO,EAAE;;;;;;;;;;;;;iDAaZ,CACtC,CAAC;gBACJ,CAAC;gBACD,MAAM;QACV,CAAC;QAED,IAAI,MAAM,YAAY,CAAC,EAAE,EAAE,WAAW,CAAC,UAAU,EAAE,WAAW,CAAC,SAAS,CAAC,EAAE,CAAC;YAC1E,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,oBAAS,CAAC,KAAK,EAAE,SAAS,KAAK,EAAE,CAAC,CAAC;YACzD,OAAO;QACT,CAAC;QAED,sEAAsE;QACtE,mEAAmE;QACnE,IAAI,gBAAgB,GAA6B,EAAE,CAAC;QACpD,MAAM,WAAW,GAAG,MAAM,UAAU,CAAC,gBAAgB,CAAC,EAAE,EAAE,WAAW,CAAC,UAAU,CAAC,CAAC;QAClF,QAAQ,WAAW,CAAC,IAAI,EAAE,CAAC;YACzB,KAAK,eAAe;gBAClB,MAAM;YACR,KAAK,QAAQ;gBACX,gBAAgB,GAAG,EAAE,oBAAoB,EAAE,QAAQ,EAAE,CAAC;gBACtD,MAAM;YACR,KAAK,KAAK;gBACR,4EAA4E;gBAC5E,gBAAgB,GAAG;oBACjB,oBAAoB,EAAE,SAAS;oBAC/B,WAAW,EAAE,WAAW,CAAC,QAAQ;iBAClC,CAAC;gBACF,MAAM;YACR,KAAK,gBAAgB;gBACnB,IAAI,CAAC,IAAI,CAAC,WAAW,CACnB,oBAAS,CAAC,KAAK,EACf,oBAAoB,WAAW,CAAC,UAAU,iBAAiB,MAAM,OAAO,EAAE,gBAAgB,CAC3F,CAAC;gBACF,MAAM;YACR,KAAK,eAAe;gBAClB,IAAI,CAAC,IAAI,CAAC,WAAW,CACnB,oBAAS,CAAC,KAAK,EACf,iDAAiD,WAAW,CAAC,UAAU,8JAA8J,CACtO,CAAC;gBACF,MAAM;QACV,CAAC;QAED,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;YACtB,OAAO;QACT,CAAC;QACD,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,UAAU;YAC9C,CAAC,CAAC,MAAM,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,UAAU,CAAC;YAC9D,CAAC,CAAC,MAAM,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QAE9C,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,oBAAS,CAAC,MAAM,EAAE,UAAU,KAAK,EAAE,CAAC,CAAC;QAE3D,MAAM,MAAM,GAAG,MAAM,CAAC,MAAM,CAC1B,EAAE,EACF;YACE,MAAM,EAAE,WAAW,CAAC,UAAU;YAC9B,GAAG,EAAE,WAAW,CAAC,SAAS;YAC1B,IAAI,EAAE,IAAA,qBAAgB,EAAC,WAAW,CAAC,YAAY,CAAC;YAChD,WAAW,EAAE,WAAW,CAAC,WAAW;YACpC,iBAAiB,EAAE,QAAQ;SACI,EACjC,gBAAgB,CACjB,CAAC;QAEF,MAAM,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;IAC1B,CAAC;IAEO,KAAK,CAAC,WAAW,CAAC,MAAkB;QAC1C,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;YACjB,MAAM,IAAI,KAAK,CACb,qDAAqD,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAC9E,CAAC;QACJ,CAAC;QAED,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC;QAEzD,IAAI,MAAM,CAAC,SAAS,KAAK,0CAAkB,CAAC,aAAa,EAAE,CAAC;YAC1D,MAAM,WAAW,GAAG,iBAAiB,CAAC;YAEtC,MAAM,aAAE,CAAC,KAAK,CAAC,IAAI,CAAC,aAAa,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;YACxD,MAAM,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,OAAO,MAAM,CAAC,CAAC;YAEnF,IAAI,MAAM,IAAA,qBAAU,EAAC,YAAY,CAAC,EAAE,CAAC;gBACnC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,oBAAS,CAAC,MAAM,EAAE,cAAc,YAAY,EAAE,CAAC,CAAC;gBACtE,OAAO,EAAE,YAAY,EAAE,WAAW,EAAE,CAAC;YACvC,CAAC;YAED,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,oBAAS,CAAC,KAAK,EAAE,OAAO,QAAQ,OAAO,YAAY,EAAE,CAAC,CAAC;YAC7E,MAAM,IAAA,sBAAY,EAAC,QAAQ,EAAE,YAAY,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,oBAAS,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC;YAC7F,OAAO,EAAE,YAAY,EAAE,WAAW,EAAE,CAAC;QACvC,CAAC;aAAM,CAAC;YACN,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,0BAA0B,CAAC;YACzE,OAAO,EAAE,YAAY,EAAE,QAAQ,EAAE,WAAW,EAAE,CAAC;QACjD,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,mBAAmB,CAAC,UAAoB;QACpD,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,oBAAS,CAAC,KAAK,EAAE,yCAAyC,UAAU,GAAG,CAAC,CAAC;QAE/F,MAAM,mBAAmB,GAAG,IAAA,8CAAmC,EAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAEvF,OAAO;YACL,YAAY,EAAE,CACZ,MAAM,IAAA,aAAK,EAAC,UAAU,EAAE,EAAE,2BAA2B,EAAE,QAAQ,EAAE,mBAAmB,EAAE,CAAC,CACxF,CAAC,IAAI,EAAE;YACR,WAAW,EAAE,iBAAiB;SAC/B,CAAC;IACJ,CAAC;CACF;AAjMD,4CAiMC;AAED,IAAK,eAKJ;AALD,WAAK,eAAe;IAClB,yEAAc,CAAA;IACd,qDAAI,CAAA;IACJ,+DAAS,CAAA;IACT,uGAA6B,CAAA;AAC/B,CAAC,EALI,eAAe,KAAf,eAAe,QAKnB;AASD,KAAK,UAAU,YAAY,CAAC,EAAa,EAAE,MAAc,EAAE,GAAW;IACpE;;;;;;;;;;;;;;;OAeG;IACH,MAAM,QAAQ,GAAG,MAAM,EAAE,CAAC,aAAa,CAAC;QACtC,MAAM,EAAE,MAAM;QACd,MAAM,EAAE,GAAG;QACX,OAAO,EAAE,CAAC;KACX,CAAC,CAAC;IACH,OAAO,CACL,QAAQ,CAAC,QAAQ,IAAI,IAAI;QACzB,QAAQ,CAAC,QAAQ,CAAC,IAAI,CACpB,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,IAAI,IAAI,IAAI,MAAM,CAAC,IAAI,GAAG,mBAAmB,CAAC,CAC7F,CACF,CAAC;AACJ,CAAC;AAmBD;;;;;;GAMG;AACH,MAAM,iBAAiB;IACd,MAAM,CAAC,GAAG,CAAC,IAAkB;QAClC,MAAM,QAAQ,GAAG,iBAAiB,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACpD,IAAI,QAAQ,EAAE,CAAC;YACb,OAAO,QAAQ,CAAC;QAClB,CAAC;QAED,MAAM,KAAK,GAAG,IAAI,iBAAiB,EAAE,CAAC;QACtC,iBAAiB,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;QAC1C,OAAO,KAAK,CAAC;IACf,CAAC;IAOD;QAHiB,eAAU,GAAG,IAAI,GAAG,EAA2B,CAAC;QAChD,gBAAW,GAAG,IAAI,GAAG,EAA4B,CAAC;IAGnE,CAAC;IAEM,KAAK,CAAC,eAAe,CAC1B,EAAa,EACb,MAAc,EACd,eAAwB;QAExB,OAAO,MAAM,CAAC,IAAI,CAAC,UAAU,EAAE,MAAM,EAAE,KAAK,IAAI,EAAE;YAChD,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,gBAAgB,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;YAE3D,QAAQ,UAAU,EAAE,CAAC;gBACnB,KAAK,eAAe,CAAC,IAAI;oBACvB,IACE,eAAe;wBACf,CAAC,MAAM,IAAI,CAAC,gBAAgB,CAAC,EAAE,EAAE,MAAM,EAAE,eAAe,CAAC,CAAC,KAAK,eAAe,CAAC,SAAS,EACxF,CAAC;wBACD,6EAA6E;wBAC7E,6CAA6C;wBAC7C,OAAO,eAAe,CAAC,6BAA6B,CAAC;oBACvD,CAAC;oBACD,OAAO,eAAe,CAAC,IAAI,CAAC;gBAC9B;oBACE,OAAO,UAAU,CAAC;YACtB,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAEM,KAAK,CAAC,gBAAgB,CAAC,EAAa,EAAE,MAAc;QACzD,OAAO,MAAM,CAAC,IAAI,CAAC,WAAW,EAAE,MAAM,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC,CAAC;IACpF,CAAC;IAEO,KAAK,CAAC,gBAAgB,CAC5B,EAAa,EACb,MAAc,EACd,OAAgB;QAEhB,IAAI,CAAC;YACH,MAAM,EAAE,CAAC,iBAAiB,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,mBAAmB,EAAE,OAAO,EAAE,CAAC,CAAC;YAC7E,OAAO,eAAe,CAAC,IAAI,CAAC;QAC9B,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,IAAI,CAAC,CAAC,IAAI,KAAK,cAAc,EAAE,CAAC;gBAC9B,OAAO,eAAe,CAAC,cAAc,CAAC;YACxC,CAAC;YACD,IAAI,CAAC,cAAc,EAAE,mBAAmB,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC;gBAC3D,OAAO,eAAe,CAAC,SAAS,CAAC;YACnC,CAAC;YACD,MAAM,CAAC,CAAC;QACV,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,iBAAiB,CAAC,EAAa,EAAE,MAAc;QAC3D,IAAI,CAAC;YACH,MAAM,UAAU,GAAG,MAAM,EAAE,CAAC,mBAAmB,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;YACpE,MAAM,CAAC,GAAG,UAAU,EAAE,iCAAiC,EAAE,KAAK,EAAE,MAAM,IAAI,CAAC,CAAC;YAC5E,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;gBACV,MAAM,KAAK,GACT,UAAU,EAAE,iCAAiC,EAAE,KAAK,EAAE,EAAE,CACtD,CAAC,CACF,EAAE,kCAAkC,CAAC;gBACxC,IAAI,OAAO,GAAG,KAAK,EAAE,YAAY,CAAC;gBAClC,IAAI,OAAO,KAAK,QAAQ;oBAAE,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,CAAC;gBACpD,IAAI,OAAO,KAAK,SAAS;oBAAE,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,cAAc,EAAE,CAAC;YACrF,CAAC;YACD,OAAO,EAAE,IAAI,EAAE,eAAe,EAAE,CAAC;QACnC,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,IAAI,CAAC,CAAC,IAAI,KAAK,cAAc,EAAE,CAAC;gBAC9B,OAAO,EAAE,IAAI,EAAE,gBAAgB,EAAE,CAAC;YACpC,CAAC;YACD,IAAI,CAAC,CAAC,IAAI,KAAK,gDAAgD,EAAE,CAAC;gBAChE,OAAO,EAAE,IAAI,EAAE,eAAe,EAAE,CAAC;YACnC,CAAC;YAED,IAAI,CAAC,cAAc,EAAE,mBAAmB,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC;gBAC3D,OAAO,EAAE,IAAI,EAAE,eAAe,EAAE,CAAC;YACnC,CAAC;YACD,OAAO,EAAE,IAAI,EAAE,eAAe,EAAE,CAAC;QACnC,CAAC;IACH,CAAC;;AAnFuB,wBAAM,GAAG,IAAI,OAAO,EAAmC,AAAjD,CAAkD;AAsFlF,KAAK,UAAU,MAAM,CAAO,KAAgB,EAAE,GAAM,EAAE,OAA6B;IACjF,IAAI,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC;QACnB,OAAO,KAAK,CAAC,GAAG,CAAC,GAAG,CAAE,CAAC;IACzB,CAAC;IAED,MAAM,KAAK,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC;IACjC,KAAK,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;IACtB,OAAO,KAAK,CAAC;AACf,CAAC","sourcesContent":["import { createReadStream, promises as fs } from 'fs';\nimport * as path from 'path';\nimport { FileAssetPackaging, FileSource } from '@aws-cdk/cloud-assembly-schema';\nimport * as mime from 'mime';\nimport { destinationToClientOptions } from './client-options';\nimport { FileManifestEntry } from '../../asset-manifest';\nimport { IS3Client } from '../../aws';\nimport { PutObjectCommandInput } from '../../aws-types';\nimport { EventType, shellEventPublisherFromEventEmitter } from '../../progress';\nimport { zipDirectory } from '../archive';\nimport { IAssetHandler, IHandlerHost, type PublishOptions } from '../asset-handler';\nimport { pathExists } from '../fs-extra';\nimport { replaceAwsPlaceholders } from '../placeholders';\nimport { shell } from '../shell';\n\n/**\n * The size of an empty zip file is 22 bytes\n *\n * Ref: https://en.wikipedia.org/wiki/ZIP_(file_format)\n */\nconst EMPTY_ZIP_FILE_SIZE = 22;\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  ) {\n    this.fileCacheRoot = path.join(workDir, '.cache');\n  }\n\n  public async build(): Promise<void> {\n  }\n\n  public async isPublished(): Promise<boolean> {\n    const destination = await replaceAwsPlaceholders(this.asset.destination, this.host.aws);\n    const s3Url = `s3://${destination.bucketName}/${destination.objectKey}`;\n    try {\n      const s3 = await this.host.aws.s3Client({\n        ...destinationToClientOptions(destination),\n        quiet: true,\n      });\n      this.host.emitMessage(EventType.CHECK, `Check ${s3Url}`);\n\n      if (await objectExists(s3, destination.bucketName, destination.objectKey)) {\n        this.host.emitMessage(EventType.FOUND, `Found ${s3Url}`);\n        return true;\n      }\n    } catch (e: any) {\n      this.host.emitMessage(EventType.DEBUG, `${e.message}`);\n    }\n    return false;\n  }\n\n  public async publish(options: PublishOptions = {}): Promise<void> {\n    const destination = await replaceAwsPlaceholders(this.asset.destination, this.host.aws);\n    const s3Url = `s3://${destination.bucketName}/${destination.objectKey}`;\n\n    const clientOptions = destinationToClientOptions(destination);\n    const s3 = await this.host.aws.s3Client(clientOptions);\n    this.host.emitMessage(EventType.CHECK, `Check ${s3Url}`);\n\n    const bucketInfo = BucketInformation.for(this.host);\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 () =>\n      (await this.host.aws.discoverTargetAccount(clientOptions)).accountId;\n\n    const allowCrossAccount = options.allowCrossAccount ?? true;\n    switch (\n      await bucketInfo.bucketOwnership(\n        s3,\n        destination.bucketName,\n        allowCrossAccount ? undefined : await account(),\n      )\n    ) {\n      case BucketOwnership.MINE:\n        break;\n      case BucketOwnership.DOES_NOT_EXIST:\n        throw new Error(\n          `No bucket named '${destination.bucketName}'. Is account ${await account()} bootstrapped?`,\n        );\n      case BucketOwnership.NO_ACCESS:\n        throw new Error(\n          `Bucket named '${destination.bucketName}' exists, but we dont have access to it.`,\n        );\n      case BucketOwnership.SOMEONE_ELSES_AND_HAVE_ACCESS:\n        if (!allowCrossAccount) {\n          throw new Error(\n            `❗❗ UNEXPECTED BUCKET OWNER DETECTED ❗❗\n\n              We've detected that the S3 bucket ${destination.bucketName} was\n              originally created in account ${await account()} as part of the CloudFormation stack CDKToolkit,\n              but now resides in a different AWS account. To prevent cross-account asset bucket access of your\n              deployments, CDK will stop now.\n\n              If this situation is intentional and you own the AWS account that the bucket has moved to, remove the\n              resource named StagingBucket from the template of CloudFormation stack CDKToolkit and try again.\n\n              If this situation is not intentional, we strongly recommend auditing your account to make sure all\n              resources are configured the way you expect them [1]. For questions or concerns, please contact\n              AWS Support [2].\n\n              [1] https://repost.aws/knowledge-center/potential-account-compromise\n\n              [2] https://aws.amazon.com/support`,\n          );\n        }\n        break;\n    }\n\n    if (await objectExists(s3, destination.bucketName, destination.objectKey)) {\n      this.host.emitMessage(EventType.FOUND, `Found ${s3Url}`);\n      return;\n    }\n\n    // Identify the the bucket encryption type to set the header on upload\n    // required for SCP rules denying uploads without encryption header\n    let paramsEncryption: { [index: string]: any } = {};\n    const encryption2 = await bucketInfo.bucketEncryption(s3, destination.bucketName);\n    switch (encryption2.type) {\n      case 'no_encryption':\n        break;\n      case 'aes256':\n        paramsEncryption = { ServerSideEncryption: 'AES256' };\n        break;\n      case 'kms':\n        // We must include the key ID otherwise S3 will encrypt with the default key\n        paramsEncryption = {\n          ServerSideEncryption: 'aws:kms',\n          SSEKMSKeyId: encryption2.kmsKeyId,\n        };\n        break;\n      case 'does_not_exist':\n        this.host.emitMessage(\n          EventType.DEBUG,\n          `No bucket named '${destination.bucketName}'. Is account ${await account()} bootstrapped?`,\n        );\n        break;\n      case 'access_denied':\n        this.host.emitMessage(\n          EventType.DEBUG,\n          `Could not read encryption settings of bucket '${destination.bucketName}': uploading with default settings (\"cdk bootstrap\" to version 9 if your organization's policies prevent a successful upload or to get rid of this message).`,\n        );\n        break;\n    }\n\n    if (this.host.aborted) {\n      return;\n    }\n    const publishFile = this.asset.source.executable\n      ? await this.externalPackageFile(this.asset.source.executable)\n      : await this.packageFile(this.asset.source);\n\n    this.host.emitMessage(EventType.UPLOAD, `Upload ${s3Url}`);\n\n    const params = Object.assign(\n      {},\n      {\n        Bucket: destination.bucketName,\n        Key: destination.objectKey,\n        Body: createReadStream(publishFile.packagedPath),\n        ContentType: publishFile.contentType,\n        ChecksumAlgorithm: 'SHA256',\n      } satisfies PutObjectCommandInput,\n      paramsEncryption,\n    );\n\n    await s3.upload(params);\n  }\n\n  private async packageFile(source: FileSource): Promise<PackagedFileAsset> {\n    if (!source.path) {\n      throw new Error(\n        `'path' is expected in the File asset source, got: ${JSON.stringify(source)}`,\n      );\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 ${packagedPath}`);\n        return { packagedPath, contentType };\n      }\n\n      this.host.emitMessage(EventType.BUILD, `Zip ${fullPath} -> ${packagedPath}`);\n      await zipDirectory(fullPath, packagedPath, (m) => this.host.emitMessage(EventType.DEBUG, m));\n      return { packagedPath, contentType };\n    } else {\n      const contentType = mime.getType(fullPath) ?? 'application/octet-stream';\n      return { packagedPath: fullPath, contentType };\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    const shellEventPublisher = shellEventPublisherFromEventEmitter(this.host.emitMessage);\n\n    return {\n      packagedPath: (\n        await shell(executable, { subprocessOutputDestination: 'ignore', shellEventPublisher })\n      ).trim(),\n      contentType: 'application/zip',\n    };\n  }\n}\n\nenum BucketOwnership {\n  DOES_NOT_EXIST,\n  MINE,\n  NO_ACCESS,\n  SOMEONE_ELSES_AND_HAVE_ACCESS,\n}\n\ntype BucketEncryption =\n  | { readonly type: 'no_encryption' }\n  | { readonly type: 'aes256' }\n  | { readonly type: 'kms'; readonly kmsKeyId?: string }\n  | { readonly type: 'access_denied' }\n  | { readonly type: 'does_not_exist' };\n\nasync function objectExists(s3: IS3Client, 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   * If the file is too small, we discount it as a cache hit. There is an issue\n   * somewhere that sometimes produces empty zip files, and we would otherwise\n   * never retry building those assets without users having to manually clear\n   * their bucket, which is a bad experience.\n   */\n  const response = await s3.listObjectsV2({\n    Bucket: bucket,\n    Prefix: key,\n    MaxKeys: 1,\n  });\n  return (\n    response.Contents != null &&\n    response.Contents.some(\n      (object) => object.Key === key && (object.Size == null || object.Size > EMPTY_ZIP_FILE_SIZE),\n    )\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\n/**\n * Cache for bucket information, so we don't have to keep doing the same calls again and again\n *\n * We scope the lifetime of the cache to the lifetime of the host, so that we don't have to do\n * anything special for tests and yet the cache will live for the entire lifetime of the asset\n * upload session when used by the CLI.\n */\nclass BucketInformation {\n  public static for(host: IHandlerHost) {\n    const existing = BucketInformation.caches.get(host);\n    if (existing) {\n      return existing;\n    }\n\n    const fresh = new BucketInformation();\n    BucketInformation.caches.set(host, fresh);\n    return fresh;\n  }\n\n  private static readonly caches = new WeakMap<IHandlerHost, BucketInformation>();\n\n  private readonly ownerships = new Map<string, BucketOwnership>();\n  private readonly encryptions = new Map<string, BucketEncryption>();\n\n  private constructor() {\n  }\n\n  public async bucketOwnership(\n    s3: IS3Client,\n    bucket: string,\n    expectedAccount?: string,\n  ): Promise<BucketOwnership> {\n    return cached(this.ownerships, bucket, async () => {\n      const anyAccount = await this._bucketOwnership(s3, bucket);\n\n      switch (anyAccount) {\n        case BucketOwnership.MINE:\n          if (\n            expectedAccount &&\n            (await this._bucketOwnership(s3, bucket, expectedAccount)) === BucketOwnership.NO_ACCESS\n          ) {\n            // if the only difference between MINE and NO_ACCESS is the expected account,\n            // then its definitely someone else's bucket.\n            return BucketOwnership.SOMEONE_ELSES_AND_HAVE_ACCESS;\n          }\n          return BucketOwnership.MINE;\n        default:\n          return anyAccount;\n      }\n    });\n  }\n\n  public async bucketEncryption(s3: IS3Client, bucket: string): Promise<BucketEncryption> {\n    return cached(this.encryptions, bucket, () => this._bucketEncryption(s3, bucket));\n  }\n\n  private async _bucketOwnership(\n    s3: IS3Client,\n    bucket: string,\n    account?: string,\n  ): Promise<BucketOwnership.MINE | BucketOwnership.DOES_NOT_EXIST | BucketOwnership.NO_ACCESS> {\n    try {\n      await s3.getBucketLocation({ Bucket: bucket, ExpectedBucketOwner: account });\n      return BucketOwnership.MINE;\n    } catch (e: any) {\n      if (e.name === 'NoSuchBucket') {\n        return BucketOwnership.DOES_NOT_EXIST;\n      }\n      if (['AccessDenied', 'AllAccessDisabled'].includes(e.name)) {\n        return BucketOwnership.NO_ACCESS;\n      }\n      throw e;\n    }\n  }\n\n  private async _bucketEncryption(s3: IS3Client, bucket: string): Promise<BucketEncryption> {\n    try {\n      const encryption = await s3.getBucketEncryption({ Bucket: bucket });\n      const l = encryption?.ServerSideEncryptionConfiguration?.Rules?.length ?? 0;\n      if (l > 0) {\n        const apply =\n          encryption?.ServerSideEncryptionConfiguration?.Rules?.at(\n            0,\n          )?.ApplyServerSideEncryptionByDefault;\n        let ssealgo = apply?.SSEAlgorithm;\n        if (ssealgo === 'AES256') return { type: 'aes256' };\n        if (ssealgo === 'aws:kms') return { type: 'kms', kmsKeyId: apply?.KMSMasterKeyID };\n      }\n      return { type: 'no_encryption' };\n    } catch (e: any) {\n      if (e.name === 'NoSuchBucket') {\n        return { type: 'does_not_exist' };\n      }\n      if (e.name === 'ServerSideEncryptionConfigurationNotFoundError') {\n        return { type: 'no_encryption' };\n      }\n\n      if (['AccessDenied', 'AllAccessDisabled'].includes(e.name)) {\n        return { type: 'access_denied' };\n      }\n      return { type: 'no_encryption' };\n    }\n  }\n}\n\nasync function cached<A, B>(cache: Map<A, B>, key: A, factory: (x: A) => Promise<B>): Promise<B> {\n  if (cache.has(key)) {\n    return cache.get(key)!;\n  }\n\n  const fresh = await factory(key);\n  cache.set(key, fresh);\n  return fresh;\n}\n"]}
|
|
289
|
+
//# 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 * as mime from 'mime';\nimport { destinationToClientOptions } from './client-options';\nimport { FileManifestEntry } from '../../asset-manifest';\nimport { IS3Client } from '../../aws';\nimport { PutObjectCommandInput } from '../../aws-types';\nimport { EventType, shellEventPublisherFromEventEmitter } from '../../progress';\nimport { zipDirectory } from '../archive';\nimport { IAssetHandler, IHandlerHost, type PublishOptions } from '../asset-handler';\nimport { pathExists } from '../fs-extra';\nimport { replaceAwsPlaceholders } from '../placeholders';\nimport { shell } from '../shell';\n\n/**\n * The size of an empty zip file is 22 bytes\n *\n * Ref: https://en.wikipedia.org/wiki/ZIP_(file_format)\n */\nconst EMPTY_ZIP_FILE_SIZE = 22;\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  ) {\n    this.fileCacheRoot = path.join(workDir, '.cache');\n  }\n\n  public async build(): Promise<void> {\n  }\n\n  public async isPublished(): Promise<boolean> {\n    const destination = await replaceAwsPlaceholders(this.asset.destination, this.host.aws);\n    const s3Url = `s3://${destination.bucketName}/${destination.objectKey}`;\n    try {\n      const s3 = await this.host.aws.s3Client({\n        ...destinationToClientOptions(destination),\n        quiet: true,\n      });\n      this.host.emitMessage(EventType.CHECK, `Check ${s3Url}`);\n\n      if (await objectExists(s3, destination.bucketName, destination.objectKey)) {\n        this.host.emitMessage(EventType.FOUND, `Found ${s3Url}`);\n        return true;\n      }\n    } catch (e: any) {\n      this.host.emitMessage(EventType.DEBUG, `${e.message}`);\n    }\n    return false;\n  }\n\n  public async publish(options: PublishOptions = {}): Promise<void> {\n    const destination = await replaceAwsPlaceholders(this.asset.destination, this.host.aws);\n    const s3Url = `s3://${destination.bucketName}/${destination.objectKey}`;\n\n    const clientOptions = destinationToClientOptions(destination);\n    const s3 = await this.host.aws.s3Client(clientOptions);\n    this.host.emitMessage(EventType.CHECK, `Check ${s3Url}`);\n\n    const bucketInfo = BucketInformation.for(this.host);\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 () =>\n      (await this.host.aws.discoverTargetAccount(clientOptions)).accountId;\n\n    const allowCrossAccount = options.allowCrossAccount ?? true;\n    switch (\n      await bucketInfo.bucketOwnership(\n        s3,\n        destination.bucketName,\n        allowCrossAccount ? undefined : await account(),\n      )\n    ) {\n      case BucketOwnership.MINE:\n        break;\n      case BucketOwnership.DOES_NOT_EXIST:\n        throw new Error(\n          `No bucket named '${destination.bucketName}'. Is account ${await account()} bootstrapped?`,\n        );\n      case BucketOwnership.NO_ACCESS:\n        throw new Error(\n          `Bucket named '${destination.bucketName}' exists, but we dont have access to it.`,\n        );\n      case BucketOwnership.SOMEONE_ELSES_AND_HAVE_ACCESS:\n        if (!allowCrossAccount) {\n          throw new Error(\n            `❗❗ UNEXPECTED BUCKET OWNER DETECTED ❗❗\n\n              We've detected that the S3 bucket ${destination.bucketName} was\n              originally created in account ${await account()} as part of the CloudFormation stack CDKToolkit,\n              but now resides in a different AWS account. To prevent cross-account asset bucket access of your\n              deployments, CDK will stop now.\n\n              If this situation is intentional and you own the AWS account that the bucket has moved to, remove the\n              resource named StagingBucket from the template of CloudFormation stack CDKToolkit and try again.\n\n              If this situation is not intentional, we strongly recommend auditing your account to make sure all\n              resources are configured the way you expect them [1]. For questions or concerns, please contact\n              AWS Support [2].\n\n              [1] https://repost.aws/knowledge-center/potential-account-compromise\n\n              [2] https://aws.amazon.com/support`,\n          );\n        }\n        break;\n    }\n\n    if (!options.force && await objectExists(s3, destination.bucketName, destination.objectKey)) {\n      this.host.emitMessage(EventType.FOUND, `Found ${s3Url}`);\n      return;\n    }\n\n    // Identify the the bucket encryption type to set the header on upload\n    // required for SCP rules denying uploads without encryption header\n    let paramsEncryption: { [index: string]: any } = {};\n    const encryption2 = await bucketInfo.bucketEncryption(s3, destination.bucketName);\n    switch (encryption2.type) {\n      case 'no_encryption':\n        break;\n      case 'aes256':\n        paramsEncryption = { ServerSideEncryption: 'AES256' };\n        break;\n      case 'kms':\n        // We must include the key ID otherwise S3 will encrypt with the default key\n        paramsEncryption = {\n          ServerSideEncryption: 'aws:kms',\n          SSEKMSKeyId: encryption2.kmsKeyId,\n        };\n        break;\n      case 'does_not_exist':\n        this.host.emitMessage(\n          EventType.DEBUG,\n          `No bucket named '${destination.bucketName}'. Is account ${await account()} bootstrapped?`,\n        );\n        break;\n      case 'access_denied':\n        this.host.emitMessage(\n          EventType.DEBUG,\n          `Could not read encryption settings of bucket '${destination.bucketName}': uploading with default settings (\"cdk bootstrap\" to version 9 if your organization's policies prevent a successful upload or to get rid of this message).`,\n        );\n        break;\n    }\n\n    if (this.host.aborted) {\n      return;\n    }\n    const publishFile = this.asset.source.executable\n      ? await this.externalPackageFile(this.asset.source.executable)\n      : await this.packageFile(this.asset.source);\n\n    this.host.emitMessage(EventType.UPLOAD, `Upload ${s3Url}`);\n\n    const params = Object.assign(\n      {},\n      {\n        Bucket: destination.bucketName,\n        Key: destination.objectKey,\n        Body: createReadStream(publishFile.packagedPath),\n        ContentType: publishFile.contentType,\n        ChecksumAlgorithm: 'SHA256',\n      } satisfies PutObjectCommandInput,\n      paramsEncryption,\n    );\n\n    await s3.upload(params);\n  }\n\n  private async packageFile(source: FileSource): Promise<PackagedFileAsset> {\n    if (!source.path) {\n      throw new Error(\n        `'path' is expected in the File asset source, got: ${JSON.stringify(source)}`,\n      );\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 ${packagedPath}`);\n        return { packagedPath, contentType };\n      }\n\n      this.host.emitMessage(EventType.BUILD, `Zip ${fullPath} -> ${packagedPath}`);\n      await zipDirectory(fullPath, packagedPath, (m) => this.host.emitMessage(EventType.DEBUG, m));\n      return { packagedPath, contentType };\n    } else {\n      const contentType = mime.getType(fullPath) ?? 'application/octet-stream';\n      return { packagedPath: fullPath, contentType };\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    const shellEventPublisher = shellEventPublisherFromEventEmitter(this.host.emitMessage);\n\n    return {\n      packagedPath: (\n        await shell(executable, { subprocessOutputDestination: 'ignore', shellEventPublisher })\n      ).trim(),\n      contentType: 'application/zip',\n    };\n  }\n}\n\nenum BucketOwnership {\n  DOES_NOT_EXIST,\n  MINE,\n  NO_ACCESS,\n  SOMEONE_ELSES_AND_HAVE_ACCESS,\n}\n\ntype BucketEncryption =\n  | { readonly type: 'no_encryption' }\n  | { readonly type: 'aes256' }\n  | { readonly type: 'kms'; readonly kmsKeyId?: string }\n  | { readonly type: 'access_denied' }\n  | { readonly type: 'does_not_exist' };\n\nasync function objectExists(s3: IS3Client, 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   * If the file is too small, we discount it as a cache hit. There is an issue\n   * somewhere that sometimes produces empty zip files, and we would otherwise\n   * never retry building those assets without users having to manually clear\n   * their bucket, which is a bad experience.\n   */\n  const response = await s3.listObjectsV2({\n    Bucket: bucket,\n    Prefix: key,\n    MaxKeys: 1,\n  });\n  return (\n    response.Contents != null &&\n    response.Contents.some(\n      (object) => object.Key === key && (object.Size == null || object.Size > EMPTY_ZIP_FILE_SIZE),\n    )\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\n/**\n * Cache for bucket information, so we don't have to keep doing the same calls again and again\n *\n * We scope the lifetime of the cache to the lifetime of the host, so that we don't have to do\n * anything special for tests and yet the cache will live for the entire lifetime of the asset\n * upload session when used by the CLI.\n */\nclass BucketInformation {\n  public static for(host: IHandlerHost) {\n    const existing = BucketInformation.caches.get(host);\n    if (existing) {\n      return existing;\n    }\n\n    const fresh = new BucketInformation();\n    BucketInformation.caches.set(host, fresh);\n    return fresh;\n  }\n\n  private static readonly caches = new WeakMap<IHandlerHost, BucketInformation>();\n\n  private readonly ownerships = new Map<string, BucketOwnership>();\n  private readonly encryptions = new Map<string, BucketEncryption>();\n\n  private constructor() {\n  }\n\n  public async bucketOwnership(\n    s3: IS3Client,\n    bucket: string,\n    expectedAccount?: string,\n  ): Promise<BucketOwnership> {\n    return cached(this.ownerships, bucket, async () => {\n      const anyAccount = await this._bucketOwnership(s3, bucket);\n\n      switch (anyAccount) {\n        case BucketOwnership.MINE:\n          if (\n            expectedAccount &&\n            (await this._bucketOwnership(s3, bucket, expectedAccount)) === BucketOwnership.NO_ACCESS\n          ) {\n            // if the only difference between MINE and NO_ACCESS is the expected account,\n            // then its definitely someone else's bucket.\n            return BucketOwnership.SOMEONE_ELSES_AND_HAVE_ACCESS;\n          }\n          return BucketOwnership.MINE;\n        default:\n          return anyAccount;\n      }\n    });\n  }\n\n  public async bucketEncryption(s3: IS3Client, bucket: string): Promise<BucketEncryption> {\n    return cached(this.encryptions, bucket, () => this._bucketEncryption(s3, bucket));\n  }\n\n  private async _bucketOwnership(\n    s3: IS3Client,\n    bucket: string,\n    account?: string,\n  ): Promise<BucketOwnership.MINE | BucketOwnership.DOES_NOT_EXIST | BucketOwnership.NO_ACCESS> {\n    try {\n      await s3.getBucketLocation({ Bucket: bucket, ExpectedBucketOwner: account });\n      return BucketOwnership.MINE;\n    } catch (e: any) {\n      if (e.name === 'NoSuchBucket') {\n        return BucketOwnership.DOES_NOT_EXIST;\n      }\n      if (['AccessDenied', 'AllAccessDisabled'].includes(e.name)) {\n        return BucketOwnership.NO_ACCESS;\n      }\n      throw e;\n    }\n  }\n\n  private async _bucketEncryption(s3: IS3Client, bucket: string): Promise<BucketEncryption> {\n    try {\n      const encryption = await s3.getBucketEncryption({ Bucket: bucket });\n      const l = encryption?.ServerSideEncryptionConfiguration?.Rules?.length ?? 0;\n      if (l > 0) {\n        const apply =\n          encryption?.ServerSideEncryptionConfiguration?.Rules?.at(\n            0,\n          )?.ApplyServerSideEncryptionByDefault;\n        let ssealgo = apply?.SSEAlgorithm;\n        if (ssealgo === 'AES256') return { type: 'aes256' };\n        if (ssealgo === 'aws:kms') return { type: 'kms', kmsKeyId: apply?.KMSMasterKeyID };\n      }\n      return { type: 'no_encryption' };\n    } catch (e: any) {\n      if (e.name === 'NoSuchBucket') {\n        return { type: 'does_not_exist' };\n      }\n      if (e.name === 'ServerSideEncryptionConfigurationNotFoundError') {\n        return { type: 'no_encryption' };\n      }\n\n      if (['AccessDenied', 'AllAccessDisabled'].includes(e.name)) {\n        return { type: 'access_denied' };\n      }\n      return { type: 'no_encryption' };\n    }\n  }\n}\n\nasync function cached<A, B>(cache: Map<A, B>, key: A, factory: (x: A) => Promise<B>): Promise<B> {\n  if (cache.has(key)) {\n    return cache.get(key)!;\n  }\n\n  const fresh = await factory(key);\n  cache.set(key, fresh);\n  return fresh;\n}\n"]}
|
package/lib/publishing.js
CHANGED
|
@@ -59,7 +59,7 @@ class AssetPublishing {
|
|
|
59
59
|
*/
|
|
60
60
|
async buildEntry(asset) {
|
|
61
61
|
try {
|
|
62
|
-
if (this.progressEvent(progress_1.EventType.START, `Building ${asset.
|
|
62
|
+
if (this.progressEvent(progress_1.EventType.START, `Building ${asset.displayName(false)}`)) {
|
|
63
63
|
return false;
|
|
64
64
|
}
|
|
65
65
|
const handler = this.assetHandler(asset);
|
|
@@ -68,7 +68,7 @@ class AssetPublishing {
|
|
|
68
68
|
throw new Error('Aborted');
|
|
69
69
|
}
|
|
70
70
|
this.completedOperations++;
|
|
71
|
-
if (this.progressEvent(progress_1.EventType.SUCCESS, `Built ${asset.
|
|
71
|
+
if (this.progressEvent(progress_1.EventType.SUCCESS, `Built ${asset.displayName(false)}`)) {
|
|
72
72
|
return false;
|
|
73
73
|
}
|
|
74
74
|
}
|
|
@@ -86,7 +86,7 @@ class AssetPublishing {
|
|
|
86
86
|
*/
|
|
87
87
|
async publishEntry(asset, options = {}) {
|
|
88
88
|
try {
|
|
89
|
-
if (this.progressEvent(progress_1.EventType.START, `Publishing ${asset.
|
|
89
|
+
if (this.progressEvent(progress_1.EventType.START, `Publishing ${asset.displayName(true)}`)) {
|
|
90
90
|
return false;
|
|
91
91
|
}
|
|
92
92
|
const handler = this.assetHandler(asset);
|
|
@@ -95,7 +95,7 @@ class AssetPublishing {
|
|
|
95
95
|
throw new Error('Aborted');
|
|
96
96
|
}
|
|
97
97
|
this.completedOperations++;
|
|
98
|
-
if (this.progressEvent(progress_1.EventType.SUCCESS, `Published ${asset.
|
|
98
|
+
if (this.progressEvent(progress_1.EventType.SUCCESS, `Published ${asset.displayName(true)}`)) {
|
|
99
99
|
return false;
|
|
100
100
|
}
|
|
101
101
|
}
|
|
@@ -122,7 +122,7 @@ class AssetPublishing {
|
|
|
122
122
|
*/
|
|
123
123
|
async publishAsset(asset, options = {}) {
|
|
124
124
|
try {
|
|
125
|
-
if (this.progressEvent(progress_1.EventType.START, `Publishing ${asset.
|
|
125
|
+
if (this.progressEvent(progress_1.EventType.START, `Publishing ${asset.displayName(true)}`)) {
|
|
126
126
|
return false;
|
|
127
127
|
}
|
|
128
128
|
const handler = this.assetHandler(asset);
|
|
@@ -136,7 +136,7 @@ class AssetPublishing {
|
|
|
136
136
|
throw new Error('Aborted');
|
|
137
137
|
}
|
|
138
138
|
this.completedOperations++;
|
|
139
|
-
if (this.progressEvent(progress_1.EventType.SUCCESS, `Published ${asset.
|
|
139
|
+
if (this.progressEvent(progress_1.EventType.SUCCESS, `Published ${asset.displayName(true)}`)) {
|
|
140
140
|
return false;
|
|
141
141
|
}
|
|
142
142
|
}
|
|
@@ -190,4 +190,4 @@ class AssetPublishing {
|
|
|
190
190
|
}
|
|
191
191
|
}
|
|
192
192
|
exports.AssetPublishing = AssetPublishing;
|
|
193
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"publishing.js","sourceRoot":"","sources":["publishing.ts"],"names":[],"mappings":"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","sourcesContent":["import { AssetManifest, IManifestEntry } from './asset-manifest';\nimport { IAws } from './aws';\nimport {\n  IAssetHandler,\n  IHandlerHost,\n  type PublishOptions,\n  SubprocessOutputDestination,\n} from './private/asset-handler';\nimport { DockerFactory } from './private/docker';\nimport { makeAssetHandler } from './private/handlers';\nimport { pLimit } from './private/p-limit';\nimport { EventType, IPublishProgress, IPublishProgressListener } from './progress';\n\nexport interface AssetPublishingOptions {\n  /**\n   * Entry point for AWS client\n   */\n  readonly aws: IAws;\n\n  /**\n   * Listener for progress events\n   *\n   * @default No listener\n   */\n  readonly progressListener?: IPublishProgressListener;\n\n  /**\n   * Whether to throw at the end if there were errors\n   *\n   * @default true\n   */\n  readonly throwOnError?: boolean;\n\n  /**\n   * Whether to publish in parallel, when 'publish()' is called\n   *\n   * @default false\n   */\n  readonly publishInParallel?: boolean;\n\n  /**\n   * Whether to build assets, when 'publish()' is called\n   *\n   * @default true\n   */\n  readonly buildAssets?: boolean;\n\n  /**\n   * Whether to publish assets, when 'publish()' is called\n   *\n   * @default true\n   */\n  readonly publishAssets?: boolean;\n\n  /**\n   * @deprecated use {@link #subprocessOutputDestination} instead\n   */\n  readonly quiet?: boolean;\n\n  /**\n   * Where to send output of a subprocesses\n   *\n   * @default 'stdio'\n   */\n  subprocessOutputDestination?: SubprocessOutputDestination;\n}\n\n/**\n * A failure to publish an asset\n */\nexport interface FailedAsset {\n  /**\n   * The asset that failed to publish\n   */\n  readonly asset: IManifestEntry;\n\n  /**\n   * The failure that occurred\n   */\n  readonly error: Error;\n}\n\nexport class AssetPublishing implements IPublishProgress {\n  /**\n   * The message for the IPublishProgress interface\n   */\n  public message: string = 'Starting';\n\n  /**\n   * The current asset for the IPublishProgress interface\n   */\n  public currentAsset?: IManifestEntry;\n  public readonly failures = new Array<FailedAsset>();\n  private readonly assets: IManifestEntry[];\n\n  private readonly totalOperations: number;\n  private completedOperations: number = 0;\n  private aborted = false;\n  private readonly handlerHost: IHandlerHost;\n  private readonly publishInParallel: boolean;\n  private readonly buildAssets: boolean;\n  private readonly publishAssets: boolean;\n  private readonly handlerCache = new Map<IManifestEntry, IAssetHandler>();\n\n  constructor(\n    private readonly manifest: AssetManifest,\n    private readonly options: AssetPublishingOptions,\n  ) {\n    this.assets = manifest.entries;\n    this.totalOperations = this.assets.length;\n    this.publishInParallel = options.publishInParallel ?? false;\n    this.buildAssets = options.buildAssets ?? true;\n    this.publishAssets = options.publishAssets ?? true;\n\n    const self = this;\n    this.handlerHost = {\n      aws: this.options.aws,\n      get aborted() {\n        return self.aborted;\n      },\n      emitMessage(t, m) {\n        self.progressEvent(t, m);\n      },\n      dockerFactory: new DockerFactory(),\n    };\n  }\n\n  /**\n   * Publish all assets from the manifest\n   */\n  public async publish(options: PublishOptions = {}): Promise<void> {\n    if (this.publishInParallel) {\n      const limit = pLimit(20);\n      // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n      await Promise.all(\n        this.assets.map((asset) => limit(async () => this.publishAsset(asset, options))),\n      );\n    } else {\n      for (const asset of this.assets) {\n        if (!(await this.publishAsset(asset, options))) {\n          break;\n        }\n      }\n    }\n\n    if ((this.options.throwOnError ?? true) && this.failures.length > 0) {\n      throw new Error(`Error publishing: ${this.failures.map((e) => e.error.message)}`);\n    }\n  }\n\n  /**\n   * Build a single asset from the manifest\n   */\n  public async buildEntry(asset: IManifestEntry) {\n    try {\n      if (this.progressEvent(EventType.START, `Building ${asset.id}`)) {\n        return false;\n      }\n\n      const handler = this.assetHandler(asset);\n      await handler.build();\n\n      if (this.aborted) {\n        throw new Error('Aborted');\n      }\n\n      this.completedOperations++;\n      if (this.progressEvent(EventType.SUCCESS, `Built ${asset.id}`)) {\n        return false;\n      }\n    } catch (e: any) {\n      this.failures.push({ asset, error: e });\n      this.completedOperations++;\n      if (this.progressEvent(EventType.FAIL, e.message)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  /**\n   * Publish a single asset from the manifest\n   */\n  public async publishEntry(asset: IManifestEntry, options: PublishOptions = {}) {\n    try {\n      if (this.progressEvent(EventType.START, `Publishing ${asset.id}`)) {\n        return false;\n      }\n\n      const handler = this.assetHandler(asset);\n      await handler.publish(options);\n\n      if (this.aborted) {\n        throw new Error('Aborted');\n      }\n\n      this.completedOperations++;\n      if (this.progressEvent(EventType.SUCCESS, `Published ${asset.id}`)) {\n        return false;\n      }\n    } catch (e: any) {\n      this.failures.push({ asset, error: e });\n      this.completedOperations++;\n      if (this.progressEvent(EventType.FAIL, e.message)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  /**\n   * Return whether a single asset is published\n   */\n  public isEntryPublished(asset: IManifestEntry) {\n    const handler = this.assetHandler(asset);\n    return handler.isPublished();\n  }\n\n  /**\n   * publish an asset (used by 'publish()')\n   * @param asset The asset to publish\n   * @returns false when publishing should stop\n   */\n  private async publishAsset(asset: IManifestEntry, options: PublishOptions = {}) {\n    try {\n      if (this.progressEvent(EventType.START, `Publishing ${asset.id}`)) {\n        return false;\n      }\n\n      const handler = this.assetHandler(asset);\n\n      if (this.buildAssets) {\n        await handler.build();\n      }\n\n      if (this.publishAssets) {\n        await handler.publish(options);\n      }\n\n      if (this.aborted) {\n        throw new Error('Aborted');\n      }\n\n      this.completedOperations++;\n      if (this.progressEvent(EventType.SUCCESS, `Published ${asset.id}`)) {\n        return false;\n      }\n    } catch (e: any) {\n      this.failures.push({ asset, error: e });\n      this.completedOperations++;\n      if (this.progressEvent(EventType.FAIL, e.message)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  public get percentComplete() {\n    if (this.totalOperations === 0) {\n      return 100;\n    }\n    return Math.floor((this.completedOperations / this.totalOperations) * 100);\n  }\n\n  public abort(): void {\n    this.aborted = true;\n  }\n\n  public get hasFailures() {\n    return this.failures.length > 0;\n  }\n\n  /**\n   * Publish a progress event to the listener, if present.\n   *\n   * Returns whether an abort is requested. Helper to get rid of repetitive code in publish().\n   */\n  private progressEvent(event: EventType, message: string): boolean {\n    this.message = message;\n    if (this.options.progressListener) {\n      this.options.progressListener.onPublishEvent(event, this);\n    }\n    return this.aborted;\n  }\n\n  private assetHandler(asset: IManifestEntry) {\n    const existing = this.handlerCache.get(asset);\n    if (existing) {\n      return existing;\n    }\n    if (this.options.quiet !== undefined && this.options.subprocessOutputDestination) {\n      throw new Error(\n        'Cannot set both quiet and subprocessOutputDestination. Please use only subprocessOutputDestination',\n      );\n    }\n    const subprocessOutputDestination =\n      this.options.subprocessOutputDestination ?? (this.options.quiet ? 'ignore' : 'stdio');\n    const ret = makeAssetHandler(this.manifest, asset, this.handlerHost, {\n      subprocessOutputDestination,\n    });\n    this.handlerCache.set(asset, ret);\n    return ret;\n  }\n}\n"]}
|
|
193
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"publishing.js","sourceRoot":"","sources":["publishing.ts"],"names":[],"mappings":"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","sourcesContent":["import { AssetManifest, IManifestEntry } from './asset-manifest';\nimport { IAws } from './aws';\nimport {\n  IAssetHandler,\n  IHandlerHost,\n  type PublishOptions,\n  SubprocessOutputDestination,\n} from './private/asset-handler';\nimport { DockerFactory } from './private/docker';\nimport { makeAssetHandler } from './private/handlers';\nimport { pLimit } from './private/p-limit';\nimport { EventType, IPublishProgress, IPublishProgressListener } from './progress';\n\nexport interface AssetPublishingOptions {\n  /**\n   * Entry point for AWS client\n   */\n  readonly aws: IAws;\n\n  /**\n   * Listener for progress events\n   *\n   * @default No listener\n   */\n  readonly progressListener?: IPublishProgressListener;\n\n  /**\n   * Whether to throw at the end if there were errors\n   *\n   * @default true\n   */\n  readonly throwOnError?: boolean;\n\n  /**\n   * Whether to publish in parallel, when 'publish()' is called\n   *\n   * @default false\n   */\n  readonly publishInParallel?: boolean;\n\n  /**\n   * Whether to build assets, when 'publish()' is called\n   *\n   * @default true\n   */\n  readonly buildAssets?: boolean;\n\n  /**\n   * Whether to publish assets, when 'publish()' is called\n   *\n   * @default true\n   */\n  readonly publishAssets?: boolean;\n\n  /**\n   * @deprecated use {@link #subprocessOutputDestination} instead\n   */\n  readonly quiet?: boolean;\n\n  /**\n   * Where to send output of a subprocesses\n   *\n   * @default 'stdio'\n   */\n  subprocessOutputDestination?: SubprocessOutputDestination;\n}\n\n/**\n * A failure to publish an asset\n */\nexport interface FailedAsset {\n  /**\n   * The asset that failed to publish\n   */\n  readonly asset: IManifestEntry;\n\n  /**\n   * The failure that occurred\n   */\n  readonly error: Error;\n}\n\nexport class AssetPublishing implements IPublishProgress {\n  /**\n   * The message for the IPublishProgress interface\n   */\n  public message: string = 'Starting';\n\n  /**\n   * The current asset for the IPublishProgress interface\n   */\n  public currentAsset?: IManifestEntry;\n  public readonly failures = new Array<FailedAsset>();\n  private readonly assets: IManifestEntry[];\n\n  private readonly totalOperations: number;\n  private completedOperations: number = 0;\n  private aborted = false;\n  private readonly handlerHost: IHandlerHost;\n  private readonly publishInParallel: boolean;\n  private readonly buildAssets: boolean;\n  private readonly publishAssets: boolean;\n  private readonly handlerCache = new Map<IManifestEntry, IAssetHandler>();\n\n  constructor(\n    private readonly manifest: AssetManifest,\n    private readonly options: AssetPublishingOptions,\n  ) {\n    this.assets = manifest.entries;\n    this.totalOperations = this.assets.length;\n    this.publishInParallel = options.publishInParallel ?? false;\n    this.buildAssets = options.buildAssets ?? true;\n    this.publishAssets = options.publishAssets ?? true;\n\n    const self = this;\n    this.handlerHost = {\n      aws: this.options.aws,\n      get aborted() {\n        return self.aborted;\n      },\n      emitMessage(t, m) {\n        self.progressEvent(t, m);\n      },\n      dockerFactory: new DockerFactory(),\n    };\n  }\n\n  /**\n   * Publish all assets from the manifest\n   */\n  public async publish(options: PublishOptions = {}): Promise<void> {\n    if (this.publishInParallel) {\n      const limit = pLimit(20);\n      // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n      await Promise.all(\n        this.assets.map((asset) => limit(async () => this.publishAsset(asset, options))),\n      );\n    } else {\n      for (const asset of this.assets) {\n        if (!(await this.publishAsset(asset, options))) {\n          break;\n        }\n      }\n    }\n\n    if ((this.options.throwOnError ?? true) && this.failures.length > 0) {\n      throw new Error(`Error publishing: ${this.failures.map((e) => e.error.message)}`);\n    }\n  }\n\n  /**\n   * Build a single asset from the manifest\n   */\n  public async buildEntry(asset: IManifestEntry) {\n    try {\n      if (this.progressEvent(EventType.START, `Building ${asset.displayName(false)}`)) {\n        return false;\n      }\n\n      const handler = this.assetHandler(asset);\n      await handler.build();\n\n      if (this.aborted) {\n        throw new Error('Aborted');\n      }\n\n      this.completedOperations++;\n      if (this.progressEvent(EventType.SUCCESS, `Built ${asset.displayName(false)}`)) {\n        return false;\n      }\n    } catch (e: any) {\n      this.failures.push({ asset, error: e });\n      this.completedOperations++;\n      if (this.progressEvent(EventType.FAIL, e.message)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  /**\n   * Publish a single asset from the manifest\n   */\n  public async publishEntry(asset: IManifestEntry, options: PublishOptions = {}) {\n    try {\n      if (this.progressEvent(EventType.START, `Publishing ${asset.displayName(true)}`)) {\n        return false;\n      }\n\n      const handler = this.assetHandler(asset);\n      await handler.publish(options);\n\n      if (this.aborted) {\n        throw new Error('Aborted');\n      }\n\n      this.completedOperations++;\n      if (this.progressEvent(EventType.SUCCESS, `Published ${asset.displayName(true)}`)) {\n        return false;\n      }\n    } catch (e: any) {\n      this.failures.push({ asset, error: e });\n      this.completedOperations++;\n      if (this.progressEvent(EventType.FAIL, e.message)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  /**\n   * Return whether a single asset is published\n   */\n  public isEntryPublished(asset: IManifestEntry) {\n    const handler = this.assetHandler(asset);\n    return handler.isPublished();\n  }\n\n  /**\n   * publish an asset (used by 'publish()')\n   * @param asset The asset to publish\n   * @returns false when publishing should stop\n   */\n  private async publishAsset(asset: IManifestEntry, options: PublishOptions = {}) {\n    try {\n      if (this.progressEvent(EventType.START, `Publishing ${asset.displayName(true)}`)) {\n        return false;\n      }\n\n      const handler = this.assetHandler(asset);\n\n      if (this.buildAssets) {\n        await handler.build();\n      }\n\n      if (this.publishAssets) {\n        await handler.publish(options);\n      }\n\n      if (this.aborted) {\n        throw new Error('Aborted');\n      }\n\n      this.completedOperations++;\n      if (this.progressEvent(EventType.SUCCESS, `Published ${asset.displayName(true)}`)) {\n        return false;\n      }\n    } catch (e: any) {\n      this.failures.push({ asset, error: e });\n      this.completedOperations++;\n      if (this.progressEvent(EventType.FAIL, e.message)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  public get percentComplete() {\n    if (this.totalOperations === 0) {\n      return 100;\n    }\n    return Math.floor((this.completedOperations / this.totalOperations) * 100);\n  }\n\n  public abort(): void {\n    this.aborted = true;\n  }\n\n  public get hasFailures() {\n    return this.failures.length > 0;\n  }\n\n  /**\n   * Publish a progress event to the listener, if present.\n   *\n   * Returns whether an abort is requested. Helper to get rid of repetitive code in publish().\n   */\n  private progressEvent(event: EventType, message: string): boolean {\n    this.message = message;\n    if (this.options.progressListener) {\n      this.options.progressListener.onPublishEvent(event, this);\n    }\n    return this.aborted;\n  }\n\n  private assetHandler(asset: IManifestEntry) {\n    const existing = this.handlerCache.get(asset);\n    if (existing) {\n      return existing;\n    }\n    if (this.options.quiet !== undefined && this.options.subprocessOutputDestination) {\n      throw new Error(\n        'Cannot set both quiet and subprocessOutputDestination. Please use only subprocessOutputDestination',\n      );\n    }\n    const subprocessOutputDestination =\n      this.options.subprocessOutputDestination ?? (this.options.quiet ? 'ignore' : 'stdio');\n    const ret = makeAssetHandler(this.manifest, asset, this.handlerHost, {\n      subprocessOutputDestination,\n    });\n    this.handlerCache.set(asset, ret);\n    return ret;\n  }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -41,7 +41,6 @@
|
|
|
41
41
|
"@smithy/util-stream": "^4.1.2",
|
|
42
42
|
"@stylistic/eslint-plugin": "^3.1.0",
|
|
43
43
|
"@types/archiver": "^6.0.3",
|
|
44
|
-
"@types/glob": "^8.1.0",
|
|
45
44
|
"@types/jest": "^29.5.14",
|
|
46
45
|
"@types/mime": "^2",
|
|
47
46
|
"@types/mock-fs": "^4",
|
|
@@ -54,7 +53,7 @@
|
|
|
54
53
|
"commit-and-tag-version": "^12",
|
|
55
54
|
"constructs": "^10.0.0",
|
|
56
55
|
"eslint": "^9",
|
|
57
|
-
"eslint-config-prettier": "^10.0.
|
|
56
|
+
"eslint-config-prettier": "^10.0.2",
|
|
58
57
|
"eslint-import-resolver-typescript": "^3.8.3",
|
|
59
58
|
"eslint-plugin-import": "^2.31.0",
|
|
60
59
|
"eslint-plugin-jest": "^28.11.0",
|
|
@@ -73,8 +72,8 @@
|
|
|
73
72
|
"typescript": "5.6"
|
|
74
73
|
},
|
|
75
74
|
"dependencies": {
|
|
76
|
-
"@aws-cdk/cloud-assembly-schema": "^
|
|
77
|
-
"@aws-cdk/cx-api": "^2.
|
|
75
|
+
"@aws-cdk/cloud-assembly-schema": "^41.1.0",
|
|
76
|
+
"@aws-cdk/cx-api": "^2.181.1",
|
|
78
77
|
"@aws-sdk/client-ecr": "^3",
|
|
79
78
|
"@aws-sdk/client-s3": "^3",
|
|
80
79
|
"@aws-sdk/client-secrets-manager": "^3",
|
|
@@ -101,7 +100,7 @@
|
|
|
101
100
|
"publishConfig": {
|
|
102
101
|
"access": "public"
|
|
103
102
|
},
|
|
104
|
-
"version": "3.
|
|
103
|
+
"version": "3.1.1",
|
|
105
104
|
"types": "lib/index.d.ts",
|
|
106
105
|
"//": "~~ Generated by projen. To modify, edit .projenrc.js and run \"npx projen\"."
|
|
107
106
|
}
|