cdk-assets 3.3.0 → 4.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +4 -4
- package/THIRD_PARTY_LICENSES +18764 -0
- package/bin/cdk-assets.js +109830 -62
- package/bin/docker-credential-cdk-assets.js +4 -4
- package/bin/list.js +3 -3
- package/bin/logging.js +3 -1
- package/bin/publish.js +6 -6
- package/package.json +11 -30
- package/lib/asset-manifest.d.ts +0 -186
- package/lib/asset-manifest.js +0 -270
- package/lib/aws-types.d.ts +0 -1666
- package/lib/aws-types.js +0 -144
- package/lib/aws.d.ts +0 -71
- package/lib/aws.js +0 -131
- package/lib/index.d.ts +0 -4
- package/lib/index.js +0 -21
- package/lib/private/archive.d.ts +0 -3
- package/lib/private/archive.js +0 -87
- package/lib/private/asset-handler.d.ts +0 -60
- package/lib/private/asset-handler.js +0 -3
- package/lib/private/docker-credentials.d.ts +0 -35
- package/lib/private/docker-credentials.js +0 -90
- package/lib/private/docker.d.ts +0 -98
- package/lib/private/docker.js +0 -235
- package/lib/private/fs-extra.d.ts +0 -3
- package/lib/private/fs-extra.js +0 -37
- package/lib/private/handlers/client-options.d.ts +0 -3
- package/lib/private/handlers/client-options.js +0 -12
- package/lib/private/handlers/container-images.d.ts +0 -22
- package/lib/private/handlers/container-images.js +0 -224
- package/lib/private/handlers/files.d.ts +0 -14
- package/lib/private/handlers/files.js +0 -289
- package/lib/private/handlers/index.d.ts +0 -3
- package/lib/private/handlers/index.js +0 -16
- package/lib/private/p-limit.d.ts +0 -10
- package/lib/private/p-limit.js +0 -51
- package/lib/private/placeholders.d.ts +0 -10
- package/lib/private/placeholders.js +0 -34
- package/lib/private/shell.d.ts +0 -24
- package/lib/private/shell.js +0 -131
- package/lib/private/util.d.ts +0 -5
- package/lib/private/util.js +0 -16
- package/lib/progress.d.ts +0 -114
- package/lib/progress.js +0 -104
- package/lib/publishing.d.ts +0 -118
- package/lib/publishing.js +0 -193
- package/scripts/manual-test-manifest.json +0 -12
- package/scripts/manual-test.sh +0 -22
|
@@ -1,289 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.FileAssetHandler = void 0;
|
|
4
|
-
const fs_1 = require("fs");
|
|
5
|
-
const path = require("path");
|
|
6
|
-
const cloud_assembly_schema_1 = require("@aws-cdk/cloud-assembly-schema");
|
|
7
|
-
const mime = require("mime");
|
|
8
|
-
const client_options_1 = require("./client-options");
|
|
9
|
-
const progress_1 = require("../../progress");
|
|
10
|
-
const archive_1 = require("../archive");
|
|
11
|
-
const fs_extra_1 = require("../fs-extra");
|
|
12
|
-
const placeholders_1 = require("../placeholders");
|
|
13
|
-
const shell_1 = require("../shell");
|
|
14
|
-
/**
|
|
15
|
-
* The size of an empty zip file is 22 bytes
|
|
16
|
-
*
|
|
17
|
-
* Ref: https://en.wikipedia.org/wiki/ZIP_(file_format)
|
|
18
|
-
*/
|
|
19
|
-
const EMPTY_ZIP_FILE_SIZE = 22;
|
|
20
|
-
class FileAssetHandler {
|
|
21
|
-
constructor(workDir, asset, host) {
|
|
22
|
-
this.workDir = workDir;
|
|
23
|
-
this.asset = asset;
|
|
24
|
-
this.host = host;
|
|
25
|
-
this.fileCacheRoot = path.join(workDir, '.cache');
|
|
26
|
-
}
|
|
27
|
-
async build() {
|
|
28
|
-
}
|
|
29
|
-
async isPublished() {
|
|
30
|
-
const destination = await (0, placeholders_1.replaceAwsPlaceholders)(this.asset.destination, this.host.aws);
|
|
31
|
-
const s3Url = `s3://${destination.bucketName}/${destination.objectKey}`;
|
|
32
|
-
try {
|
|
33
|
-
const s3 = await this.host.aws.s3Client({
|
|
34
|
-
...(0, client_options_1.destinationToClientOptions)(destination),
|
|
35
|
-
quiet: true,
|
|
36
|
-
});
|
|
37
|
-
this.host.emitMessage(progress_1.EventType.CHECK, `Check ${s3Url}`);
|
|
38
|
-
if (await objectExists(s3, destination.bucketName, destination.objectKey)) {
|
|
39
|
-
this.host.emitMessage(progress_1.EventType.FOUND, `Found ${s3Url}`);
|
|
40
|
-
return true;
|
|
41
|
-
}
|
|
42
|
-
}
|
|
43
|
-
catch (e) {
|
|
44
|
-
this.host.emitMessage(progress_1.EventType.DEBUG, `${e.message}`);
|
|
45
|
-
}
|
|
46
|
-
return false;
|
|
47
|
-
}
|
|
48
|
-
async publish(options = {}) {
|
|
49
|
-
const destination = await (0, placeholders_1.replaceAwsPlaceholders)(this.asset.destination, this.host.aws);
|
|
50
|
-
const s3Url = `s3://${destination.bucketName}/${destination.objectKey}`;
|
|
51
|
-
const clientOptions = (0, client_options_1.destinationToClientOptions)(destination);
|
|
52
|
-
const s3 = await this.host.aws.s3Client(clientOptions);
|
|
53
|
-
this.host.emitMessage(progress_1.EventType.CHECK, `Check ${s3Url}`);
|
|
54
|
-
const bucketInfo = BucketInformation.for(this.host);
|
|
55
|
-
// A thunk for describing the current account. Used when we need to format an error
|
|
56
|
-
// message, not in the success case.
|
|
57
|
-
const account = async () => (await this.host.aws.discoverTargetAccount(clientOptions)).accountId;
|
|
58
|
-
const allowCrossAccount = options.allowCrossAccount ?? true;
|
|
59
|
-
switch (await bucketInfo.bucketOwnership(s3, destination.bucketName, allowCrossAccount ? undefined : await account())) {
|
|
60
|
-
case BucketOwnership.MINE:
|
|
61
|
-
break;
|
|
62
|
-
case BucketOwnership.DOES_NOT_EXIST:
|
|
63
|
-
throw new Error(`No bucket named '${destination.bucketName}'. Is account ${await account()} bootstrapped?`);
|
|
64
|
-
case BucketOwnership.NO_ACCESS:
|
|
65
|
-
throw new Error(`Bucket named '${destination.bucketName}' exists, but we dont have access to it.`);
|
|
66
|
-
case BucketOwnership.SOMEONE_ELSES_AND_HAVE_ACCESS:
|
|
67
|
-
if (!allowCrossAccount) {
|
|
68
|
-
throw new Error(`❗❗ UNEXPECTED BUCKET OWNER DETECTED ❗❗
|
|
69
|
-
|
|
70
|
-
We've detected that the S3 bucket ${destination.bucketName} was
|
|
71
|
-
originally created in account ${await account()} as part of the CloudFormation stack CDKToolkit,
|
|
72
|
-
but now resides in a different AWS account. To prevent cross-account asset bucket access of your
|
|
73
|
-
deployments, CDK will stop now.
|
|
74
|
-
|
|
75
|
-
If this situation is intentional and you own the AWS account that the bucket has moved to, remove the
|
|
76
|
-
resource named StagingBucket from the template of CloudFormation stack CDKToolkit and try again.
|
|
77
|
-
|
|
78
|
-
If this situation is not intentional, we strongly recommend auditing your account to make sure all
|
|
79
|
-
resources are configured the way you expect them [1]. For questions or concerns, please contact
|
|
80
|
-
AWS Support [2].
|
|
81
|
-
|
|
82
|
-
[1] https://repost.aws/knowledge-center/potential-account-compromise
|
|
83
|
-
|
|
84
|
-
[2] https://aws.amazon.com/support`);
|
|
85
|
-
}
|
|
86
|
-
break;
|
|
87
|
-
}
|
|
88
|
-
if (!options.force && await objectExists(s3, destination.bucketName, destination.objectKey)) {
|
|
89
|
-
this.host.emitMessage(progress_1.EventType.FOUND, `Found ${s3Url}`);
|
|
90
|
-
return;
|
|
91
|
-
}
|
|
92
|
-
// Identify the the bucket encryption type to set the header on upload
|
|
93
|
-
// required for SCP rules denying uploads without encryption header
|
|
94
|
-
let paramsEncryption = {};
|
|
95
|
-
const encryption2 = await bucketInfo.bucketEncryption(s3, destination.bucketName);
|
|
96
|
-
switch (encryption2.type) {
|
|
97
|
-
case 'no_encryption':
|
|
98
|
-
break;
|
|
99
|
-
case 'aes256':
|
|
100
|
-
paramsEncryption = { ServerSideEncryption: 'AES256' };
|
|
101
|
-
break;
|
|
102
|
-
case 'kms':
|
|
103
|
-
// We must include the key ID otherwise S3 will encrypt with the default key
|
|
104
|
-
paramsEncryption = {
|
|
105
|
-
ServerSideEncryption: 'aws:kms',
|
|
106
|
-
SSEKMSKeyId: encryption2.kmsKeyId,
|
|
107
|
-
};
|
|
108
|
-
break;
|
|
109
|
-
case 'does_not_exist':
|
|
110
|
-
this.host.emitMessage(progress_1.EventType.DEBUG, `No bucket named '${destination.bucketName}'. Is account ${await account()} bootstrapped?`);
|
|
111
|
-
break;
|
|
112
|
-
case 'access_denied':
|
|
113
|
-
this.host.emitMessage(progress_1.EventType.DEBUG, `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).`);
|
|
114
|
-
break;
|
|
115
|
-
}
|
|
116
|
-
if (this.host.aborted) {
|
|
117
|
-
return;
|
|
118
|
-
}
|
|
119
|
-
const publishFile = this.asset.source.executable
|
|
120
|
-
? await this.externalPackageFile(this.asset.source.executable)
|
|
121
|
-
: await this.packageFile(this.asset.source);
|
|
122
|
-
this.host.emitMessage(progress_1.EventType.UPLOAD, `Upload ${s3Url}`);
|
|
123
|
-
const params = Object.assign({}, {
|
|
124
|
-
Bucket: destination.bucketName,
|
|
125
|
-
Key: destination.objectKey,
|
|
126
|
-
Body: (0, fs_1.createReadStream)(publishFile.packagedPath),
|
|
127
|
-
ContentType: publishFile.contentType,
|
|
128
|
-
ChecksumAlgorithm: 'SHA256',
|
|
129
|
-
}, paramsEncryption);
|
|
130
|
-
await s3.upload(params);
|
|
131
|
-
}
|
|
132
|
-
async packageFile(source) {
|
|
133
|
-
if (!source.path) {
|
|
134
|
-
throw new Error(`'path' is expected in the File asset source, got: ${JSON.stringify(source)}`);
|
|
135
|
-
}
|
|
136
|
-
const fullPath = path.resolve(this.workDir, source.path);
|
|
137
|
-
if (source.packaging === cloud_assembly_schema_1.FileAssetPackaging.ZIP_DIRECTORY) {
|
|
138
|
-
const contentType = 'application/zip';
|
|
139
|
-
await fs_1.promises.mkdir(this.fileCacheRoot, { recursive: true });
|
|
140
|
-
const packagedPath = path.join(this.fileCacheRoot, `${this.asset.id.assetId}.zip`);
|
|
141
|
-
if (await (0, fs_extra_1.pathExists)(packagedPath)) {
|
|
142
|
-
this.host.emitMessage(progress_1.EventType.CACHED, `From cache ${packagedPath}`);
|
|
143
|
-
return { packagedPath, contentType };
|
|
144
|
-
}
|
|
145
|
-
this.host.emitMessage(progress_1.EventType.BUILD, `Zip ${fullPath} -> ${packagedPath}`);
|
|
146
|
-
await (0, archive_1.zipDirectory)(fullPath, packagedPath, (m) => this.host.emitMessage(progress_1.EventType.DEBUG, m));
|
|
147
|
-
return { packagedPath, contentType };
|
|
148
|
-
}
|
|
149
|
-
else {
|
|
150
|
-
const contentType = mime.getType(fullPath) ?? 'application/octet-stream';
|
|
151
|
-
return { packagedPath: fullPath, contentType };
|
|
152
|
-
}
|
|
153
|
-
}
|
|
154
|
-
async externalPackageFile(executable) {
|
|
155
|
-
this.host.emitMessage(progress_1.EventType.BUILD, `Building asset source using command: '${executable}'`);
|
|
156
|
-
const shellEventPublisher = (0, progress_1.shellEventPublisherFromEventEmitter)(this.host.emitMessage);
|
|
157
|
-
return {
|
|
158
|
-
packagedPath: (await (0, shell_1.shell)(executable, { subprocessOutputDestination: 'ignore', shellEventPublisher })).trim(),
|
|
159
|
-
contentType: 'application/zip',
|
|
160
|
-
};
|
|
161
|
-
}
|
|
162
|
-
}
|
|
163
|
-
exports.FileAssetHandler = FileAssetHandler;
|
|
164
|
-
var BucketOwnership;
|
|
165
|
-
(function (BucketOwnership) {
|
|
166
|
-
BucketOwnership[BucketOwnership["DOES_NOT_EXIST"] = 0] = "DOES_NOT_EXIST";
|
|
167
|
-
BucketOwnership[BucketOwnership["MINE"] = 1] = "MINE";
|
|
168
|
-
BucketOwnership[BucketOwnership["NO_ACCESS"] = 2] = "NO_ACCESS";
|
|
169
|
-
BucketOwnership[BucketOwnership["SOMEONE_ELSES_AND_HAVE_ACCESS"] = 3] = "SOMEONE_ELSES_AND_HAVE_ACCESS";
|
|
170
|
-
})(BucketOwnership || (BucketOwnership = {}));
|
|
171
|
-
async function objectExists(s3, bucket, key) {
|
|
172
|
-
/*
|
|
173
|
-
* The object existence check here refrains from using the `headObject` operation because this
|
|
174
|
-
* would create a negative cache entry, making GET-after-PUT eventually consistent. This has been
|
|
175
|
-
* observed to result in CloudFormation issuing "ValidationError: S3 error: Access Denied", for
|
|
176
|
-
* example in https://github.com/aws/aws-cdk/issues/6430.
|
|
177
|
-
*
|
|
178
|
-
* To prevent this, we are instead using the listObjectsV2 call, using the looked up key as the
|
|
179
|
-
* prefix, and limiting results to 1. Since the list operation returns keys ordered by binary
|
|
180
|
-
* UTF-8 representation, the key we are looking for is guaranteed to always be the first match
|
|
181
|
-
* returned if it exists.
|
|
182
|
-
*
|
|
183
|
-
* If the file is too small, we discount it as a cache hit. There is an issue
|
|
184
|
-
* somewhere that sometimes produces empty zip files, and we would otherwise
|
|
185
|
-
* never retry building those assets without users having to manually clear
|
|
186
|
-
* their bucket, which is a bad experience.
|
|
187
|
-
*/
|
|
188
|
-
const response = await s3.listObjectsV2({
|
|
189
|
-
Bucket: bucket,
|
|
190
|
-
Prefix: key,
|
|
191
|
-
MaxKeys: 1,
|
|
192
|
-
});
|
|
193
|
-
return (response.Contents != null &&
|
|
194
|
-
response.Contents.some((object) => object.Key === key && (object.Size == null || object.Size > EMPTY_ZIP_FILE_SIZE)));
|
|
195
|
-
}
|
|
196
|
-
/**
|
|
197
|
-
* Cache for bucket information, so we don't have to keep doing the same calls again and again
|
|
198
|
-
*
|
|
199
|
-
* We scope the lifetime of the cache to the lifetime of the host, so that we don't have to do
|
|
200
|
-
* anything special for tests and yet the cache will live for the entire lifetime of the asset
|
|
201
|
-
* upload session when used by the CLI.
|
|
202
|
-
*/
|
|
203
|
-
class BucketInformation {
|
|
204
|
-
static for(host) {
|
|
205
|
-
const existing = BucketInformation.caches.get(host);
|
|
206
|
-
if (existing) {
|
|
207
|
-
return existing;
|
|
208
|
-
}
|
|
209
|
-
const fresh = new BucketInformation();
|
|
210
|
-
BucketInformation.caches.set(host, fresh);
|
|
211
|
-
return fresh;
|
|
212
|
-
}
|
|
213
|
-
constructor() {
|
|
214
|
-
this.ownerships = new Map();
|
|
215
|
-
this.encryptions = new Map();
|
|
216
|
-
}
|
|
217
|
-
async bucketOwnership(s3, bucket, expectedAccount) {
|
|
218
|
-
return cached(this.ownerships, bucket, async () => {
|
|
219
|
-
const anyAccount = await this._bucketOwnership(s3, bucket);
|
|
220
|
-
switch (anyAccount) {
|
|
221
|
-
case BucketOwnership.MINE:
|
|
222
|
-
if (expectedAccount &&
|
|
223
|
-
(await this._bucketOwnership(s3, bucket, expectedAccount)) === BucketOwnership.NO_ACCESS) {
|
|
224
|
-
// if the only difference between MINE and NO_ACCESS is the expected account,
|
|
225
|
-
// then its definitely someone else's bucket.
|
|
226
|
-
return BucketOwnership.SOMEONE_ELSES_AND_HAVE_ACCESS;
|
|
227
|
-
}
|
|
228
|
-
return BucketOwnership.MINE;
|
|
229
|
-
default:
|
|
230
|
-
return anyAccount;
|
|
231
|
-
}
|
|
232
|
-
});
|
|
233
|
-
}
|
|
234
|
-
async bucketEncryption(s3, bucket) {
|
|
235
|
-
return cached(this.encryptions, bucket, () => this._bucketEncryption(s3, bucket));
|
|
236
|
-
}
|
|
237
|
-
async _bucketOwnership(s3, bucket, account) {
|
|
238
|
-
try {
|
|
239
|
-
await s3.getBucketLocation({ Bucket: bucket, ExpectedBucketOwner: account });
|
|
240
|
-
return BucketOwnership.MINE;
|
|
241
|
-
}
|
|
242
|
-
catch (e) {
|
|
243
|
-
if (e.name === 'NoSuchBucket') {
|
|
244
|
-
return BucketOwnership.DOES_NOT_EXIST;
|
|
245
|
-
}
|
|
246
|
-
if (['AccessDenied', 'AllAccessDisabled'].includes(e.name)) {
|
|
247
|
-
return BucketOwnership.NO_ACCESS;
|
|
248
|
-
}
|
|
249
|
-
throw e;
|
|
250
|
-
}
|
|
251
|
-
}
|
|
252
|
-
async _bucketEncryption(s3, bucket) {
|
|
253
|
-
try {
|
|
254
|
-
const encryption = await s3.getBucketEncryption({ Bucket: bucket });
|
|
255
|
-
const l = encryption?.ServerSideEncryptionConfiguration?.Rules?.length ?? 0;
|
|
256
|
-
if (l > 0) {
|
|
257
|
-
const apply = encryption?.ServerSideEncryptionConfiguration?.Rules?.at(0)?.ApplyServerSideEncryptionByDefault;
|
|
258
|
-
let ssealgo = apply?.SSEAlgorithm;
|
|
259
|
-
if (ssealgo === 'AES256')
|
|
260
|
-
return { type: 'aes256' };
|
|
261
|
-
if (ssealgo === 'aws:kms')
|
|
262
|
-
return { type: 'kms', kmsKeyId: apply?.KMSMasterKeyID };
|
|
263
|
-
}
|
|
264
|
-
return { type: 'no_encryption' };
|
|
265
|
-
}
|
|
266
|
-
catch (e) {
|
|
267
|
-
if (e.name === 'NoSuchBucket') {
|
|
268
|
-
return { type: 'does_not_exist' };
|
|
269
|
-
}
|
|
270
|
-
if (e.name === 'ServerSideEncryptionConfigurationNotFoundError') {
|
|
271
|
-
return { type: 'no_encryption' };
|
|
272
|
-
}
|
|
273
|
-
if (['AccessDenied', 'AllAccessDisabled'].includes(e.name)) {
|
|
274
|
-
return { type: 'access_denied' };
|
|
275
|
-
}
|
|
276
|
-
return { type: 'no_encryption' };
|
|
277
|
-
}
|
|
278
|
-
}
|
|
279
|
-
}
|
|
280
|
-
BucketInformation.caches = new WeakMap();
|
|
281
|
-
async function cached(cache, key, factory) {
|
|
282
|
-
if (cache.has(key)) {
|
|
283
|
-
return cache.get(key);
|
|
284
|
-
}
|
|
285
|
-
const fresh = await factory(key);
|
|
286
|
-
cache.set(key, fresh);
|
|
287
|
-
return fresh;
|
|
288
|
-
}
|
|
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 type { FileSource } from '@aws-cdk/cloud-assembly-schema';\nimport { FileAssetPackaging } from '@aws-cdk/cloud-assembly-schema';\nimport * as mime from 'mime';\nimport { destinationToClientOptions } from './client-options';\nimport type { FileManifestEntry } from '../../asset-manifest';\nimport type { IS3Client } from '../../aws';\nimport type { PutObjectCommandInput } from '../../aws-types';\nimport { EventType, shellEventPublisherFromEventEmitter } from '../../progress';\nimport { zipDirectory } from '../archive';\nimport type { IAssetHandler, IHandlerHost, 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"]}
|
|
@@ -1,3 +0,0 @@
|
|
|
1
|
-
import { type AssetManifest, type IManifestEntry } from '../../asset-manifest';
|
|
2
|
-
import type { IAssetHandler, IHandlerHost, IHandlerOptions } from '../asset-handler';
|
|
3
|
-
export declare function makeAssetHandler(manifest: AssetManifest, asset: IManifestEntry, host: IHandlerHost, options: IHandlerOptions): IAssetHandler;
|
|
@@ -1,16 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.makeAssetHandler = makeAssetHandler;
|
|
4
|
-
const container_images_1 = require("./container-images");
|
|
5
|
-
const files_1 = require("./files");
|
|
6
|
-
const asset_manifest_1 = require("../../asset-manifest");
|
|
7
|
-
function makeAssetHandler(manifest, asset, host, options) {
|
|
8
|
-
if (asset instanceof asset_manifest_1.FileManifestEntry) {
|
|
9
|
-
return new files_1.FileAssetHandler(manifest.directory, asset, host);
|
|
10
|
-
}
|
|
11
|
-
if (asset instanceof asset_manifest_1.DockerImageManifestEntry) {
|
|
12
|
-
return new container_images_1.ContainerImageAssetHandler(manifest.directory, asset, host, options);
|
|
13
|
-
}
|
|
14
|
-
throw new Error(`Unrecognized asset type: '${asset}'`);
|
|
15
|
-
}
|
|
16
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQVVBLDRDQWNDO0FBeEJELHlEQUFnRTtBQUNoRSxtQ0FBMkM7QUFDM0MseURBSzhCO0FBRzlCLFNBQWdCLGdCQUFnQixDQUM5QixRQUF1QixFQUN2QixLQUFxQixFQUNyQixJQUFrQixFQUNsQixPQUF3QjtJQUV4QixJQUFJLEtBQUssWUFBWSxrQ0FBaUIsRUFBRSxDQUFDO1FBQ3ZDLE9BQU8sSUFBSSx3QkFBZ0IsQ0FBQyxRQUFRLENBQUMsU0FBUyxFQUFFLEtBQUssRUFBRSxJQUFJLENBQUMsQ0FBQztJQUMvRCxDQUFDO0lBQ0QsSUFBSSxLQUFLLFlBQVkseUNBQXdCLEVBQUUsQ0FBQztRQUM5QyxPQUFPLElBQUksNkNBQTBCLENBQUMsUUFBUSxDQUFDLFNBQVMsRUFBRSxLQUFLLEVBQUUsSUFBSSxFQUFFLE9BQU8sQ0FBQyxDQUFDO0lBQ2xGLENBQUM7SUFFRCxNQUFNLElBQUksS0FBSyxDQUFDLDZCQUE2QixLQUFLLEdBQUcsQ0FBQyxDQUFDO0FBQ3pELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb250YWluZXJJbWFnZUFzc2V0SGFuZGxlciB9IGZyb20gJy4vY29udGFpbmVyLWltYWdlcyc7XG5pbXBvcnQgeyBGaWxlQXNzZXRIYW5kbGVyIH0gZnJvbSAnLi9maWxlcyc7XG5pbXBvcnQge1xuICB0eXBlIEFzc2V0TWFuaWZlc3QsXG4gIERvY2tlckltYWdlTWFuaWZlc3RFbnRyeSxcbiAgRmlsZU1hbmlmZXN0RW50cnksXG4gIHR5cGUgSU1hbmlmZXN0RW50cnksXG59IGZyb20gJy4uLy4uL2Fzc2V0LW1hbmlmZXN0JztcbmltcG9ydCB0eXBlIHsgSUFzc2V0SGFuZGxlciwgSUhhbmRsZXJIb3N0LCBJSGFuZGxlck9wdGlvbnMgfSBmcm9tICcuLi9hc3NldC1oYW5kbGVyJztcblxuZXhwb3J0IGZ1bmN0aW9uIG1ha2VBc3NldEhhbmRsZXIoXG4gIG1hbmlmZXN0OiBBc3NldE1hbmlmZXN0LFxuICBhc3NldDogSU1hbmlmZXN0RW50cnksXG4gIGhvc3Q6IElIYW5kbGVySG9zdCxcbiAgb3B0aW9uczogSUhhbmRsZXJPcHRpb25zLFxuKTogSUFzc2V0SGFuZGxlciB7XG4gIGlmIChhc3NldCBpbnN0YW5jZW9mIEZpbGVNYW5pZmVzdEVudHJ5KSB7XG4gICAgcmV0dXJuIG5ldyBGaWxlQXNzZXRIYW5kbGVyKG1hbmlmZXN0LmRpcmVjdG9yeSwgYXNzZXQsIGhvc3QpO1xuICB9XG4gIGlmIChhc3NldCBpbnN0YW5jZW9mIERvY2tlckltYWdlTWFuaWZlc3RFbnRyeSkge1xuICAgIHJldHVybiBuZXcgQ29udGFpbmVySW1hZ2VBc3NldEhhbmRsZXIobWFuaWZlc3QuZGlyZWN0b3J5LCBhc3NldCwgaG9zdCwgb3B0aW9ucyk7XG4gIH1cblxuICB0aHJvdyBuZXcgRXJyb3IoYFVucmVjb2duaXplZCBhc3NldCB0eXBlOiAnJHthc3NldH0nYCk7XG59XG4iXX0=
|
package/lib/private/p-limit.d.ts
DELETED
|
@@ -1,10 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* A minimal of p-limit that does not bring in new dependencies, and is not ESM.
|
|
3
|
-
*/
|
|
4
|
-
type PromiseFactory<A> = () => Promise<A>;
|
|
5
|
-
export declare function pLimit(concurrency: number): PLimit;
|
|
6
|
-
interface PLimit {
|
|
7
|
-
dispose(): void;
|
|
8
|
-
<A>(promiseFactory: PromiseFactory<A>): Promise<A>;
|
|
9
|
-
}
|
|
10
|
-
export {};
|
package/lib/private/p-limit.js
DELETED
|
@@ -1,51 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
/**
|
|
3
|
-
* A minimal of p-limit that does not bring in new dependencies, and is not ESM.
|
|
4
|
-
*/
|
|
5
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
-
exports.pLimit = pLimit;
|
|
7
|
-
function pLimit(concurrency) {
|
|
8
|
-
const queue = [];
|
|
9
|
-
let activeCount = 0;
|
|
10
|
-
let stopped = false;
|
|
11
|
-
function dispatch() {
|
|
12
|
-
if (activeCount < concurrency && queue.length > 0) {
|
|
13
|
-
const [fac, resolve, reject] = queue.shift();
|
|
14
|
-
activeCount++;
|
|
15
|
-
fac().then((r) => {
|
|
16
|
-
// Start a new job before reporting back on the previous one
|
|
17
|
-
resumeNext();
|
|
18
|
-
resolve(r);
|
|
19
|
-
}, (e) => {
|
|
20
|
-
// Start a new job before reporting back on the previous one
|
|
21
|
-
resumeNext();
|
|
22
|
-
reject(e);
|
|
23
|
-
});
|
|
24
|
-
}
|
|
25
|
-
}
|
|
26
|
-
function resumeNext() {
|
|
27
|
-
activeCount--;
|
|
28
|
-
if (stopped) {
|
|
29
|
-
for (const [_, __, reject] of queue) {
|
|
30
|
-
reject(new Error('Task has been cancelled'));
|
|
31
|
-
}
|
|
32
|
-
queue.splice(0, queue.length);
|
|
33
|
-
}
|
|
34
|
-
dispatch();
|
|
35
|
-
}
|
|
36
|
-
const ret = (promiseFactory) => {
|
|
37
|
-
return new Promise((resolve, reject) => {
|
|
38
|
-
queue.push([promiseFactory, resolve, reject]);
|
|
39
|
-
dispatch();
|
|
40
|
-
});
|
|
41
|
-
};
|
|
42
|
-
Object.defineProperties(ret, {
|
|
43
|
-
dispose: {
|
|
44
|
-
value: () => {
|
|
45
|
-
stopped = true;
|
|
46
|
-
},
|
|
47
|
-
},
|
|
48
|
-
});
|
|
49
|
-
return ret;
|
|
50
|
-
}
|
|
51
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicC1saW1pdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInAtbGltaXQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOztHQUVHOztBQUlILHdCQWtEQztBQWxERCxTQUFnQixNQUFNLENBQUMsV0FBbUI7SUFDeEMsTUFBTSxLQUFLLEdBQTJFLEVBQUUsQ0FBQztJQUN6RixJQUFJLFdBQVcsR0FBRyxDQUFDLENBQUM7SUFDcEIsSUFBSSxPQUFPLEdBQUcsS0FBSyxDQUFDO0lBRXBCLFNBQVMsUUFBUTtRQUNmLElBQUksV0FBVyxHQUFHLFdBQVcsSUFBSSxLQUFLLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRSxDQUFDO1lBQ2xELE1BQU0sQ0FBQyxHQUFHLEVBQUUsT0FBTyxFQUFFLE1BQU0sQ0FBQyxHQUFHLEtBQUssQ0FBQyxLQUFLLEVBQUcsQ0FBQztZQUM5QyxXQUFXLEVBQUUsQ0FBQztZQUNkLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FDUixDQUFDLENBQUMsRUFBRSxFQUFFO2dCQUNKLDREQUE0RDtnQkFDNUQsVUFBVSxFQUFFLENBQUM7Z0JBQ2IsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQ2IsQ0FBQyxFQUNELENBQUMsQ0FBQyxFQUFFLEVBQUU7Z0JBQ0osNERBQTREO2dCQUM1RCxVQUFVLEVBQUUsQ0FBQztnQkFDYixNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDWixDQUFDLENBQ0YsQ0FBQztRQUNKLENBQUM7SUFDSCxDQUFDO0lBRUQsU0FBUyxVQUFVO1FBQ2pCLFdBQVcsRUFBRSxDQUFDO1FBQ2QsSUFBSSxPQUFPLEVBQUUsQ0FBQztZQUNaLEtBQUssTUFBTSxDQUFDLENBQUMsRUFBRSxFQUFFLEVBQUUsTUFBTSxDQUFDLElBQUksS0FBSyxFQUFFLENBQUM7Z0JBQ3BDLE1BQU0sQ0FBQyxJQUFJLEtBQUssQ0FBQyx5QkFBeUIsQ0FBQyxDQUFDLENBQUM7WUFDL0MsQ0FBQztZQUNELEtBQUssQ0FBQyxNQUFNLENBQUMsQ0FBQyxFQUFFLEtBQUssQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUNoQyxDQUFDO1FBQ0QsUUFBUSxFQUFFLENBQUM7SUFDYixDQUFDO0lBRUQsTUFBTSxHQUFHLEdBQUcsQ0FBSSxjQUFpQyxFQUFFLEVBQUU7UUFDbkQsT0FBTyxJQUFJLE9BQU8sQ0FBSSxDQUFDLE9BQU8sRUFBRSxNQUFNLEVBQUUsRUFBRTtZQUN4QyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUMsY0FBYyxFQUFFLE9BQU8sRUFBRSxNQUFNLENBQUMsQ0FBQyxDQUFDO1lBQzlDLFFBQVEsRUFBRSxDQUFDO1FBQ2IsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDLENBQUM7SUFDRixNQUFNLENBQUMsZ0JBQWdCLENBQUMsR0FBRyxFQUFFO1FBQzNCLE9BQU8sRUFBRTtZQUNQLEtBQUssRUFBRSxHQUFHLEVBQUU7Z0JBQ1YsT0FBTyxHQUFHLElBQUksQ0FBQztZQUNqQixDQUFDO1NBQ0Y7S0FDRixDQUFDLENBQUM7SUFFSCxPQUFPLEdBQWEsQ0FBQztBQUN2QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBBIG1pbmltYWwgIG9mIHAtbGltaXQgdGhhdCBkb2VzIG5vdCBicmluZyBpbiBuZXcgZGVwZW5kZW5jaWVzLCBhbmQgaXMgbm90IEVTTS5cbiAqL1xuXG50eXBlIFByb21pc2VGYWN0b3J5PEE+ID0gKCkgPT4gUHJvbWlzZTxBPjtcblxuZXhwb3J0IGZ1bmN0aW9uIHBMaW1pdChjb25jdXJyZW5jeTogbnVtYmVyKTogUExpbWl0IHtcbiAgY29uc3QgcXVldWU6IEFycmF5PFtQcm9taXNlRmFjdG9yeTxhbnk+LCAoeDogYW55KSA9PiB2b2lkLCAocmVhc29uPzogYW55KSA9PiB2b2lkXT4gPSBbXTtcbiAgbGV0IGFjdGl2ZUNvdW50ID0gMDtcbiAgbGV0IHN0b3BwZWQgPSBmYWxzZTtcblxuICBmdW5jdGlvbiBkaXNwYXRjaCgpIHtcbiAgICBpZiAoYWN0aXZlQ291bnQgPCBjb25jdXJyZW5jeSAmJiBxdWV1ZS5sZW5ndGggPiAwKSB7XG4gICAgICBjb25zdCBbZmFjLCByZXNvbHZlLCByZWplY3RdID0gcXVldWUuc2hpZnQoKSE7XG4gICAgICBhY3RpdmVDb3VudCsrO1xuICAgICAgZmFjKCkudGhlbihcbiAgICAgICAgKHIpID0+IHtcbiAgICAgICAgICAvLyBTdGFydCBhIG5ldyBqb2IgYmVmb3JlIHJlcG9ydGluZyBiYWNrIG9uIHRoZSBwcmV2aW91cyBvbmVcbiAgICAgICAgICByZXN1bWVOZXh0KCk7XG4gICAgICAgICAgcmVzb2x2ZShyKTtcbiAgICAgICAgfSxcbiAgICAgICAgKGUpID0+IHtcbiAgICAgICAgICAvLyBTdGFydCBhIG5ldyBqb2IgYmVmb3JlIHJlcG9ydGluZyBiYWNrIG9uIHRoZSBwcmV2aW91cyBvbmVcbiAgICAgICAgICByZXN1bWVOZXh0KCk7XG4gICAgICAgICAgcmVqZWN0KGUpO1xuICAgICAgICB9LFxuICAgICAgKTtcbiAgICB9XG4gIH1cblxuICBmdW5jdGlvbiByZXN1bWVOZXh0KCkge1xuICAgIGFjdGl2ZUNvdW50LS07XG4gICAgaWYgKHN0b3BwZWQpIHtcbiAgICAgIGZvciAoY29uc3QgW18sIF9fLCByZWplY3RdIG9mIHF1ZXVlKSB7XG4gICAgICAgIHJlamVjdChuZXcgRXJyb3IoJ1Rhc2sgaGFzIGJlZW4gY2FuY2VsbGVkJykpO1xuICAgICAgfVxuICAgICAgcXVldWUuc3BsaWNlKDAsIHF1ZXVlLmxlbmd0aCk7XG4gICAgfVxuICAgIGRpc3BhdGNoKCk7XG4gIH1cblxuICBjb25zdCByZXQgPSA8QT4ocHJvbWlzZUZhY3Rvcnk6IFByb21pc2VGYWN0b3J5PEE+KSA9PiB7XG4gICAgcmV0dXJuIG5ldyBQcm9taXNlPEE+KChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICAgIHF1ZXVlLnB1c2goW3Byb21pc2VGYWN0b3J5LCByZXNvbHZlLCByZWplY3RdKTtcbiAgICAgIGRpc3BhdGNoKCk7XG4gICAgfSk7XG4gIH07XG4gIE9iamVjdC5kZWZpbmVQcm9wZXJ0aWVzKHJldCwge1xuICAgIGRpc3Bvc2U6IHtcbiAgICAgIHZhbHVlOiAoKSA9PiB7XG4gICAgICAgIHN0b3BwZWQgPSB0cnVlO1xuICAgICAgfSxcbiAgICB9LFxuICB9KTtcblxuICByZXR1cm4gcmV0IGFzIFBMaW1pdDtcbn1cblxuaW50ZXJmYWNlIFBMaW1pdCB7XG4gIGRpc3Bvc2UoKTogdm9pZDtcbiAgPEE+KHByb21pc2VGYWN0b3J5OiBQcm9taXNlRmFjdG9yeTxBPik6IFByb21pc2U8QT47XG59XG4iXX0=
|
|
@@ -1,10 +0,0 @@
|
|
|
1
|
-
import type { IAws } from '../aws';
|
|
2
|
-
/**
|
|
3
|
-
* Replace the {ACCOUNT} and {REGION} placeholders in all strings found in a complex object.
|
|
4
|
-
*
|
|
5
|
-
* Duplicated between cdk-assets and aws-cdk CLI because we don't have a good single place to put it
|
|
6
|
-
* (they're nominally independent tools).
|
|
7
|
-
*/
|
|
8
|
-
export declare function replaceAwsPlaceholders<A extends {
|
|
9
|
-
region?: string;
|
|
10
|
-
}>(object: A, aws: IAws): Promise<A>;
|
|
@@ -1,34 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.replaceAwsPlaceholders = replaceAwsPlaceholders;
|
|
4
|
-
const cx_api_1 = require("@aws-cdk/cx-api");
|
|
5
|
-
/**
|
|
6
|
-
* Replace the {ACCOUNT} and {REGION} placeholders in all strings found in a complex object.
|
|
7
|
-
*
|
|
8
|
-
* Duplicated between cdk-assets and aws-cdk CLI because we don't have a good single place to put it
|
|
9
|
-
* (they're nominally independent tools).
|
|
10
|
-
*/
|
|
11
|
-
async function replaceAwsPlaceholders(object, aws) {
|
|
12
|
-
let partition = async () => {
|
|
13
|
-
const p = await aws.discoverPartition();
|
|
14
|
-
partition = () => Promise.resolve(p);
|
|
15
|
-
return p;
|
|
16
|
-
};
|
|
17
|
-
let account = async () => {
|
|
18
|
-
const a = await aws.discoverCurrentAccount();
|
|
19
|
-
account = () => Promise.resolve(a);
|
|
20
|
-
return a;
|
|
21
|
-
};
|
|
22
|
-
return cx_api_1.EnvironmentPlaceholders.replaceAsync(object, {
|
|
23
|
-
async region() {
|
|
24
|
-
return object.region ?? aws.discoverDefaultRegion();
|
|
25
|
-
},
|
|
26
|
-
async accountId() {
|
|
27
|
-
return (await account()).accountId;
|
|
28
|
-
},
|
|
29
|
-
async partition() {
|
|
30
|
-
return partition();
|
|
31
|
-
},
|
|
32
|
-
});
|
|
33
|
-
}
|
|
34
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGxhY2Vob2xkZXJzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsicGxhY2Vob2xkZXJzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBU0Esd0RBMkJDO0FBcENELDRDQUEwRDtBQUcxRDs7Ozs7R0FLRztBQUNJLEtBQUssVUFBVSxzQkFBc0IsQ0FDMUMsTUFBUyxFQUNULEdBQVM7SUFFVCxJQUFJLFNBQVMsR0FBRyxLQUFLLElBQUksRUFBRTtRQUN6QixNQUFNLENBQUMsR0FBRyxNQUFNLEdBQUcsQ0FBQyxpQkFBaUIsRUFBRSxDQUFDO1FBQ3hDLFNBQVMsR0FBRyxHQUFHLEVBQUUsQ0FBQyxPQUFPLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ3JDLE9BQU8sQ0FBQyxDQUFDO0lBQ1gsQ0FBQyxDQUFDO0lBRUYsSUFBSSxPQUFPLEdBQUcsS0FBSyxJQUFJLEVBQUU7UUFDdkIsTUFBTSxDQUFDLEdBQUcsTUFBTSxHQUFHLENBQUMsc0JBQXNCLEVBQUUsQ0FBQztRQUM3QyxPQUFPLEdBQUcsR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNuQyxPQUFPLENBQUMsQ0FBQztJQUNYLENBQUMsQ0FBQztJQUVGLE9BQU8sZ0NBQXVCLENBQUMsWUFBWSxDQUFDLE1BQU0sRUFBRTtRQUNsRCxLQUFLLENBQUMsTUFBTTtZQUNWLE9BQU8sTUFBTSxDQUFDLE1BQU0sSUFBSSxHQUFHLENBQUMscUJBQXFCLEVBQUUsQ0FBQztRQUN0RCxDQUFDO1FBQ0QsS0FBSyxDQUFDLFNBQVM7WUFDYixPQUFPLENBQUMsTUFBTSxPQUFPLEVBQUUsQ0FBQyxDQUFDLFNBQVMsQ0FBQztRQUNyQyxDQUFDO1FBQ0QsS0FBSyxDQUFDLFNBQVM7WUFDYixPQUFPLFNBQVMsRUFBRSxDQUFDO1FBQ3JCLENBQUM7S0FDRixDQUFDLENBQUM7QUFDTCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRW52aXJvbm1lbnRQbGFjZWhvbGRlcnMgfSBmcm9tICdAYXdzLWNkay9jeC1hcGknO1xuaW1wb3J0IHR5cGUgeyBJQXdzIH0gZnJvbSAnLi4vYXdzJztcblxuLyoqXG4gKiBSZXBsYWNlIHRoZSB7QUNDT1VOVH0gYW5kIHtSRUdJT059IHBsYWNlaG9sZGVycyBpbiBhbGwgc3RyaW5ncyBmb3VuZCBpbiBhIGNvbXBsZXggb2JqZWN0LlxuICpcbiAqIER1cGxpY2F0ZWQgYmV0d2VlbiBjZGstYXNzZXRzIGFuZCBhd3MtY2RrIENMSSBiZWNhdXNlIHdlIGRvbid0IGhhdmUgYSBnb29kIHNpbmdsZSBwbGFjZSB0byBwdXQgaXRcbiAqICh0aGV5J3JlIG5vbWluYWxseSBpbmRlcGVuZGVudCB0b29scykuXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiByZXBsYWNlQXdzUGxhY2Vob2xkZXJzPEEgZXh0ZW5kcyB7IHJlZ2lvbj86IHN0cmluZyB9PihcbiAgb2JqZWN0OiBBLFxuICBhd3M6IElBd3MsXG4pOiBQcm9taXNlPEE+IHtcbiAgbGV0IHBhcnRpdGlvbiA9IGFzeW5jICgpID0+IHtcbiAgICBjb25zdCBwID0gYXdhaXQgYXdzLmRpc2NvdmVyUGFydGl0aW9uKCk7XG4gICAgcGFydGl0aW9uID0gKCkgPT4gUHJvbWlzZS5yZXNvbHZlKHApO1xuICAgIHJldHVybiBwO1xuICB9O1xuXG4gIGxldCBhY2NvdW50ID0gYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IGEgPSBhd2FpdCBhd3MuZGlzY292ZXJDdXJyZW50QWNjb3VudCgpO1xuICAgIGFjY291bnQgPSAoKSA9PiBQcm9taXNlLnJlc29sdmUoYSk7XG4gICAgcmV0dXJuIGE7XG4gIH07XG5cbiAgcmV0dXJuIEVudmlyb25tZW50UGxhY2Vob2xkZXJzLnJlcGxhY2VBc3luYyhvYmplY3QsIHtcbiAgICBhc3luYyByZWdpb24oKSB7XG4gICAgICByZXR1cm4gb2JqZWN0LnJlZ2lvbiA/PyBhd3MuZGlzY292ZXJEZWZhdWx0UmVnaW9uKCk7XG4gICAgfSxcbiAgICBhc3luYyBhY2NvdW50SWQoKSB7XG4gICAgICByZXR1cm4gKGF3YWl0IGFjY291bnQoKSkuYWNjb3VudElkO1xuICAgIH0sXG4gICAgYXN5bmMgcGFydGl0aW9uKCkge1xuICAgICAgcmV0dXJuIHBhcnRpdGlvbigpO1xuICAgIH0sXG4gIH0pO1xufVxuIl19
|
package/lib/private/shell.d.ts
DELETED
|
@@ -1,24 +0,0 @@
|
|
|
1
|
-
import * as child_process from 'child_process';
|
|
2
|
-
import type { SubprocessOutputDestination } from './asset-handler';
|
|
3
|
-
export type ShellEventType = 'open' | 'data_stdout' | 'data_stderr' | 'close';
|
|
4
|
-
export type ShellEventPublisher = (event: ShellEventType, message: string) => void;
|
|
5
|
-
export interface ShellOptions extends child_process.SpawnOptions {
|
|
6
|
-
readonly shellEventPublisher: ShellEventPublisher;
|
|
7
|
-
readonly input?: string;
|
|
8
|
-
readonly subprocessOutputDestination?: SubprocessOutputDestination;
|
|
9
|
-
}
|
|
10
|
-
/**
|
|
11
|
-
* OS helpers
|
|
12
|
-
*
|
|
13
|
-
* Shell function which both prints to stdout and collects the output into a
|
|
14
|
-
* string.
|
|
15
|
-
*/
|
|
16
|
-
export declare function shell(command: string[], options: ShellOptions): Promise<string>;
|
|
17
|
-
export type ProcessFailedError = ProcessFailed;
|
|
18
|
-
declare class ProcessFailed extends Error {
|
|
19
|
-
readonly exitCode: number | null;
|
|
20
|
-
readonly signal: NodeJS.Signals | null;
|
|
21
|
-
readonly code = "PROCESS_FAILED";
|
|
22
|
-
constructor(exitCode: number | null, signal: NodeJS.Signals | null, message: string);
|
|
23
|
-
}
|
|
24
|
-
export {};
|