aws-cdk 2.1000.0 → 2.1000.2
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/THIRD_PARTY_LICENSES +2 -2
- package/build-info.json +2 -2
- package/lib/api/bootstrap/bootstrap-environment.d.ts +3 -1
- package/lib/api/bootstrap/bootstrap-environment.js +23 -19
- package/lib/api/bootstrap/deploy-bootstrap.d.ts +4 -2
- package/lib/api/bootstrap/deploy-bootstrap.js +13 -12
- package/lib/api/deployments/asset-publishing.d.ts +11 -3
- package/lib/api/deployments/asset-publishing.js +38 -25
- package/lib/api/deployments/assets.d.ts +3 -2
- package/lib/api/deployments/assets.js +10 -11
- package/lib/api/deployments/checks.d.ts +2 -1
- package/lib/api/deployments/checks.js +5 -5
- package/lib/api/deployments/cloudformation.d.ts +7 -7
- package/lib/api/deployments/cloudformation.js +29 -29
- package/lib/api/deployments/deploy-stack.d.ts +4 -3
- package/lib/api/deployments/deploy-stack.js +54 -51
- package/lib/api/deployments/deployments.d.ts +7 -3
- package/lib/api/deployments/deployments.js +26 -23
- package/lib/api/deployments/hotswap-deployments.d.ts +2 -1
- package/lib/api/deployments/hotswap-deployments.js +24 -18
- package/lib/api/{environment-access.d.ts → environment/environment-access.d.ts} +7 -4
- package/lib/api/environment/environment-access.js +205 -0
- package/lib/api/{environment-resources.d.ts → environment/environment-resources.d.ts} +7 -5
- package/lib/api/environment/environment-resources.js +209 -0
- package/lib/api/environment/index.d.ts +2 -0
- package/lib/api/environment/index.js +19 -0
- package/lib/api/garbage-collection/garbage-collector.d.ts +7 -0
- package/lib/api/garbage-collection/garbage-collector.js +39 -36
- package/lib/api/garbage-collection/index.d.ts +1 -0
- package/lib/api/garbage-collection/index.js +18 -0
- package/lib/api/garbage-collection/progress-printer.d.ts +4 -1
- package/lib/api/garbage-collection/progress-printer.js +7 -5
- package/lib/api/garbage-collection/stack-refresh.d.ts +6 -1
- package/lib/api/garbage-collection/stack-refresh.js +8 -8
- package/lib/api/logs/find-cloudwatch-logs.d.ts +2 -1
- package/lib/api/logs/find-cloudwatch-logs.js +4 -4
- package/lib/api/resource-import/importer.d.ts +1 -1
- package/lib/api/resource-import/importer.js +1 -1
- package/lib/api/resource-import/migrator.d.ts +1 -1
- package/lib/api/resource-import/migrator.js +1 -1
- package/lib/api/toolkit-info.d.ts +2 -1
- package/lib/api/toolkit-info.js +7 -6
- package/lib/api/util/template-body-parameter.d.ts +1 -1
- package/lib/api/util/template-body-parameter.js +1 -1
- package/lib/cli/cdk-toolkit.d.ts +1 -1
- package/lib/cli/cdk-toolkit.js +8 -4
- package/lib/cli/cli.js +8 -3
- package/lib/cli/messages.d.ts +10 -1
- package/lib/cli/messages.js +2 -1
- package/lib/index.js +17683 -17592
- package/lib/index_bg.wasm +0 -0
- package/lib/toolkit/cli-io-host.d.ts +7 -0
- package/lib/toolkit/cli-io-host.js +1 -1
- package/package.json +2 -2
- package/release.txt +2 -0
- package/CONTRIBUTING.md +0 -276
- package/generate.sh +0 -25
- package/images/garbage-collection.png +0 -0
- package/lib/api/environment-access.js +0 -203
- package/lib/api/environment-resources.js +0 -208
|
@@ -0,0 +1,209 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.NoBootstrapStackEnvironmentResources = exports.EnvironmentResources = exports.EnvironmentResourcesRegistry = void 0;
|
|
4
|
+
const messages_1 = require("../../cli/messages");
|
|
5
|
+
const notices_1 = require("../../notices");
|
|
6
|
+
const error_1 = require("../../toolkit/error");
|
|
7
|
+
const error_2 = require("../../util/error");
|
|
8
|
+
const toolkit_info_1 = require("../toolkit-info");
|
|
9
|
+
/**
|
|
10
|
+
* Registry class for `EnvironmentResources`.
|
|
11
|
+
*
|
|
12
|
+
* The state management of this class is a bit non-standard. We want to cache
|
|
13
|
+
* data related to toolkit stacks and SSM parameters, but we are not in charge
|
|
14
|
+
* of ensuring caching of SDKs. Since `EnvironmentResources` needs an SDK to
|
|
15
|
+
* function, we treat it as an ephemeral class, and store the actual cached data
|
|
16
|
+
* in `EnvironmentResourcesRegistry`.
|
|
17
|
+
*/
|
|
18
|
+
class EnvironmentResourcesRegistry {
|
|
19
|
+
constructor(toolkitStackName) {
|
|
20
|
+
this.toolkitStackName = toolkitStackName;
|
|
21
|
+
this.cache = new Map();
|
|
22
|
+
}
|
|
23
|
+
for(resolvedEnvironment, sdk, msg) {
|
|
24
|
+
const key = `${resolvedEnvironment.account}:${resolvedEnvironment.region}`;
|
|
25
|
+
let envCache = this.cache.get(key);
|
|
26
|
+
if (!envCache) {
|
|
27
|
+
envCache = emptyCache();
|
|
28
|
+
this.cache.set(key, envCache);
|
|
29
|
+
}
|
|
30
|
+
return new EnvironmentResources(resolvedEnvironment, sdk, msg, envCache, this.toolkitStackName);
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
exports.EnvironmentResourcesRegistry = EnvironmentResourcesRegistry;
|
|
34
|
+
/**
|
|
35
|
+
* Interface with the account and region we're deploying into
|
|
36
|
+
*
|
|
37
|
+
* Manages lookups for bootstrapped resources, falling back to the legacy "CDK Toolkit"
|
|
38
|
+
* original bootstrap stack if necessary.
|
|
39
|
+
*
|
|
40
|
+
* The state management of this class is a bit non-standard. We want to cache
|
|
41
|
+
* data related to toolkit stacks and SSM parameters, but we are not in charge
|
|
42
|
+
* of ensuring caching of SDKs. Since `EnvironmentResources` needs an SDK to
|
|
43
|
+
* function, we treat it as an ephemeral class, and store the actual cached data
|
|
44
|
+
* in `EnvironmentResourcesRegistry`.
|
|
45
|
+
*/
|
|
46
|
+
class EnvironmentResources {
|
|
47
|
+
constructor(environment, sdk, msg, cache, toolkitStackName) {
|
|
48
|
+
this.environment = environment;
|
|
49
|
+
this.sdk = sdk;
|
|
50
|
+
this.msg = msg;
|
|
51
|
+
this.cache = cache;
|
|
52
|
+
this.toolkitStackName = toolkitStackName;
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* Look up the toolkit for a given environment, using a given SDK
|
|
56
|
+
*/
|
|
57
|
+
async lookupToolkit() {
|
|
58
|
+
if (!this.cache.toolkitInfo) {
|
|
59
|
+
this.cache.toolkitInfo = await toolkit_info_1.ToolkitInfo.lookup(this.environment, this.sdk, this.msg, this.toolkitStackName);
|
|
60
|
+
}
|
|
61
|
+
return this.cache.toolkitInfo;
|
|
62
|
+
}
|
|
63
|
+
/**
|
|
64
|
+
* Validate that the bootstrap stack version matches or exceeds the expected version
|
|
65
|
+
*
|
|
66
|
+
* Use the SSM parameter name to read the version number if given, otherwise use the version
|
|
67
|
+
* discovered on the bootstrap stack.
|
|
68
|
+
*
|
|
69
|
+
* Pass in the SSM parameter name so we can cache the lookups an don't need to do the same
|
|
70
|
+
* lookup again and again for every artifact.
|
|
71
|
+
*/
|
|
72
|
+
async validateVersion(expectedVersion, ssmParameterName) {
|
|
73
|
+
if (expectedVersion === undefined) {
|
|
74
|
+
// No requirement
|
|
75
|
+
return;
|
|
76
|
+
}
|
|
77
|
+
const defExpectedVersion = expectedVersion;
|
|
78
|
+
if (ssmParameterName !== undefined) {
|
|
79
|
+
try {
|
|
80
|
+
doValidate(await this.versionFromSsmParameter(ssmParameterName), this.environment);
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
catch (e) {
|
|
84
|
+
if (e.name !== 'AccessDeniedException') {
|
|
85
|
+
throw e;
|
|
86
|
+
}
|
|
87
|
+
// This is a fallback! The bootstrap template that goes along with this change introduces
|
|
88
|
+
// a new 'ssm:GetParameter' permission, but when run using the previous bootstrap template we
|
|
89
|
+
// won't have the permissions yet to read the version, so we won't be able to show the
|
|
90
|
+
// message telling the user they need to update! When we see an AccessDeniedException, fall
|
|
91
|
+
// back to the version we read from Stack Outputs; but ONLY if the version we discovered via
|
|
92
|
+
// outputs is legitimately an old version. If it's newer than that, something else must be broken,
|
|
93
|
+
// so let it fail as it would if we didn't have this fallback.
|
|
94
|
+
const bootstrapStack = await this.lookupToolkit();
|
|
95
|
+
if (bootstrapStack.found && bootstrapStack.version < BOOTSTRAP_TEMPLATE_VERSION_INTRODUCING_GETPARAMETER) {
|
|
96
|
+
await this.msg.ioHost.notify((0, messages_1.warn)(this.msg.action, `Could not read SSM parameter ${ssmParameterName}: ${(0, error_2.formatErrorMessage)(e)}, falling back to version from ${bootstrapStack}`));
|
|
97
|
+
doValidate(bootstrapStack.version, this.environment);
|
|
98
|
+
return;
|
|
99
|
+
}
|
|
100
|
+
throw new error_1.ToolkitError(`This CDK deployment requires bootstrap stack version '${expectedVersion}', but during the confirmation via SSM parameter ${ssmParameterName} the following error occurred: ${e}`);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
// No SSM parameter
|
|
104
|
+
const bootstrapStack = await this.lookupToolkit();
|
|
105
|
+
doValidate(bootstrapStack.version, this.environment);
|
|
106
|
+
function doValidate(version, environment) {
|
|
107
|
+
const notices = notices_1.Notices.get();
|
|
108
|
+
if (notices) {
|
|
109
|
+
// if `Notices` hasn't been initialized there is probably a good
|
|
110
|
+
// reason for it. handle gracefully.
|
|
111
|
+
notices.addBootstrappedEnvironment({ bootstrapStackVersion: version, environment });
|
|
112
|
+
}
|
|
113
|
+
if (defExpectedVersion > version) {
|
|
114
|
+
throw new error_1.ToolkitError(`This CDK deployment requires bootstrap stack version '${expectedVersion}', found '${version}'. Please run 'cdk bootstrap'.`);
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* Read a version from an SSM parameter, cached
|
|
120
|
+
*/
|
|
121
|
+
async versionFromSsmParameter(parameterName) {
|
|
122
|
+
var _a, _b;
|
|
123
|
+
const existing = this.cache.ssmParameters.get(parameterName);
|
|
124
|
+
if (existing !== undefined) {
|
|
125
|
+
return existing;
|
|
126
|
+
}
|
|
127
|
+
const ssm = this.sdk.ssm();
|
|
128
|
+
try {
|
|
129
|
+
const result = await ssm.getParameter({ Name: parameterName });
|
|
130
|
+
const asNumber = parseInt(`${(_a = result.Parameter) === null || _a === void 0 ? void 0 : _a.Value}`, 10);
|
|
131
|
+
if (isNaN(asNumber)) {
|
|
132
|
+
throw new error_1.ToolkitError(`SSM parameter ${parameterName} not a number: ${(_b = result.Parameter) === null || _b === void 0 ? void 0 : _b.Value}`);
|
|
133
|
+
}
|
|
134
|
+
this.cache.ssmParameters.set(parameterName, asNumber);
|
|
135
|
+
return asNumber;
|
|
136
|
+
}
|
|
137
|
+
catch (e) {
|
|
138
|
+
if (e.name === 'ParameterNotFound') {
|
|
139
|
+
throw new error_1.ToolkitError(`SSM parameter ${parameterName} not found. Has the environment been bootstrapped? Please run \'cdk bootstrap\' (see https://docs.aws.amazon.com/cdk/latest/guide/bootstrapping.html)`);
|
|
140
|
+
}
|
|
141
|
+
throw e;
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
async prepareEcrRepository(repositoryName) {
|
|
145
|
+
var _a, _b;
|
|
146
|
+
if (!this.sdk) {
|
|
147
|
+
throw new error_1.ToolkitError('ToolkitInfo needs to have been initialized with an sdk to call prepareEcrRepository');
|
|
148
|
+
}
|
|
149
|
+
const ecr = this.sdk.ecr();
|
|
150
|
+
// check if repo already exists
|
|
151
|
+
try {
|
|
152
|
+
await this.msg.ioHost.notify((0, messages_1.debug)(this.msg.action, `${repositoryName}: checking if ECR repository already exists`));
|
|
153
|
+
const describeResponse = await ecr.describeRepositories({
|
|
154
|
+
repositoryNames: [repositoryName],
|
|
155
|
+
});
|
|
156
|
+
const existingRepositoryUri = (_a = describeResponse.repositories[0]) === null || _a === void 0 ? void 0 : _a.repositoryUri;
|
|
157
|
+
if (existingRepositoryUri) {
|
|
158
|
+
return { repositoryUri: existingRepositoryUri };
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
catch (e) {
|
|
162
|
+
if (e.name !== 'RepositoryNotFoundException') {
|
|
163
|
+
throw e;
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
// create the repo (tag it so it will be easier to garbage collect in the future)
|
|
167
|
+
await this.msg.ioHost.notify((0, messages_1.debug)(this.msg.action, `${repositoryName}: creating ECR repository`));
|
|
168
|
+
const assetTag = { Key: 'awscdk:asset', Value: 'true' };
|
|
169
|
+
const response = await ecr.createRepository({
|
|
170
|
+
repositoryName,
|
|
171
|
+
tags: [assetTag],
|
|
172
|
+
});
|
|
173
|
+
const repositoryUri = (_b = response.repository) === null || _b === void 0 ? void 0 : _b.repositoryUri;
|
|
174
|
+
if (!repositoryUri) {
|
|
175
|
+
throw new error_1.ToolkitError(`CreateRepository did not return a repository URI for ${repositoryUri}`);
|
|
176
|
+
}
|
|
177
|
+
// configure image scanning on push (helps in identifying software vulnerabilities, no additional charge)
|
|
178
|
+
await this.msg.ioHost.notify((0, messages_1.debug)(this.msg.action, `${repositoryName}: enable image scanning`));
|
|
179
|
+
await ecr.putImageScanningConfiguration({
|
|
180
|
+
repositoryName,
|
|
181
|
+
imageScanningConfiguration: { scanOnPush: true },
|
|
182
|
+
});
|
|
183
|
+
return { repositoryUri };
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
exports.EnvironmentResources = EnvironmentResources;
|
|
187
|
+
class NoBootstrapStackEnvironmentResources extends EnvironmentResources {
|
|
188
|
+
constructor(environment, sdk, msg) {
|
|
189
|
+
super(environment, sdk, msg, emptyCache());
|
|
190
|
+
}
|
|
191
|
+
/**
|
|
192
|
+
* Look up the toolkit for a given environment, using a given SDK
|
|
193
|
+
*/
|
|
194
|
+
async lookupToolkit() {
|
|
195
|
+
throw new error_1.ToolkitError('Trying to perform an operation that requires a bootstrap stack; you should not see this error, this is a bug in the CDK CLI.');
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
exports.NoBootstrapStackEnvironmentResources = NoBootstrapStackEnvironmentResources;
|
|
199
|
+
function emptyCache() {
|
|
200
|
+
return {
|
|
201
|
+
ssmParameters: new Map(),
|
|
202
|
+
toolkitInfo: undefined,
|
|
203
|
+
};
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* The bootstrap template version that introduced ssm:GetParameter
|
|
207
|
+
*/
|
|
208
|
+
const BOOTSTRAP_TEMPLATE_VERSION_INTRODUCING_GETPARAMETER = 5;
|
|
209
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"environment-resources.js","sourceRoot":"","sources":["environment-resources.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Environment } from '@aws-cdk/cx-api';\nimport { debug, warn } from '../../cli/messages';\nimport { Notices } from '../../notices';\nimport { IoMessaging } from '../../toolkit/cli-io-host';\nimport { ToolkitError } from '../../toolkit/error';\nimport { formatErrorMessage } from '../../util/error';\nimport type { SDK } from '../aws-auth';\nimport { type EcrRepositoryInfo, ToolkitInfo } from '../toolkit-info';\n\n/**\n * Registry class for `EnvironmentResources`.\n *\n * The state management of this class is a bit non-standard. We want to cache\n * data related to toolkit stacks and SSM parameters, but we are not in charge\n * of ensuring caching of SDKs. Since `EnvironmentResources` needs an SDK to\n * function, we treat it as an ephemeral class, and store the actual cached data\n * in `EnvironmentResourcesRegistry`.\n */\nexport class EnvironmentResourcesRegistry {\n  private readonly cache = new Map<string, EnvironmentCache>();\n\n  constructor(private readonly toolkitStackName?: string) {}\n\n  public for(resolvedEnvironment: Environment, sdk: SDK, msg: IoMessaging) {\n    const key = `${resolvedEnvironment.account}:${resolvedEnvironment.region}`;\n    let envCache = this.cache.get(key);\n    if (!envCache) {\n      envCache = emptyCache();\n      this.cache.set(key, envCache);\n    }\n    return new EnvironmentResources(resolvedEnvironment, sdk, msg, envCache, this.toolkitStackName);\n  }\n}\n\n/**\n * Interface with the account and region we're deploying into\n *\n * Manages lookups for bootstrapped resources, falling back to the legacy \"CDK Toolkit\"\n * original bootstrap stack if necessary.\n *\n * The state management of this class is a bit non-standard. We want to cache\n * data related to toolkit stacks and SSM parameters, but we are not in charge\n * of ensuring caching of SDKs. Since `EnvironmentResources` needs an SDK to\n * function, we treat it as an ephemeral class, and store the actual cached data\n * in `EnvironmentResourcesRegistry`.\n */\nexport class EnvironmentResources {\n  constructor(\n    public readonly environment: Environment,\n    private readonly sdk: SDK,\n    private readonly msg: IoMessaging,\n    private readonly cache: EnvironmentCache,\n    private readonly toolkitStackName?: string,\n  ) {}\n\n  /**\n   * Look up the toolkit for a given environment, using a given SDK\n   */\n  public async lookupToolkit() {\n    if (!this.cache.toolkitInfo) {\n      this.cache.toolkitInfo = await ToolkitInfo.lookup(this.environment, this.sdk, this.msg, this.toolkitStackName);\n    }\n    return this.cache.toolkitInfo;\n  }\n\n  /**\n   * Validate that the bootstrap stack version matches or exceeds the expected version\n   *\n   * Use the SSM parameter name to read the version number if given, otherwise use the version\n   * discovered on the bootstrap stack.\n   *\n   * Pass in the SSM parameter name so we can cache the lookups an don't need to do the same\n   * lookup again and again for every artifact.\n   */\n  public async validateVersion(expectedVersion: number | undefined, ssmParameterName: string | undefined) {\n    if (expectedVersion === undefined) {\n      // No requirement\n      return;\n    }\n    const defExpectedVersion = expectedVersion;\n\n    if (ssmParameterName !== undefined) {\n      try {\n        doValidate(await this.versionFromSsmParameter(ssmParameterName), this.environment);\n        return;\n      } catch (e: any) {\n        if (e.name !== 'AccessDeniedException') {\n          throw e;\n        }\n\n        // This is a fallback! The bootstrap template that goes along with this change introduces\n        // a new 'ssm:GetParameter' permission, but when run using the previous bootstrap template we\n        // won't have the permissions yet to read the version, so we won't be able to show the\n        // message telling the user they need to update! When we see an AccessDeniedException, fall\n        // back to the version we read from Stack Outputs; but ONLY if the version we discovered via\n        // outputs is legitimately an old version. If it's newer than that, something else must be broken,\n        // so let it fail as it would if we didn't have this fallback.\n        const bootstrapStack = await this.lookupToolkit();\n        if (bootstrapStack.found && bootstrapStack.version < BOOTSTRAP_TEMPLATE_VERSION_INTRODUCING_GETPARAMETER) {\n          await this.msg.ioHost.notify(warn(\n            this.msg.action,\n            `Could not read SSM parameter ${ssmParameterName}: ${formatErrorMessage(e)}, falling back to version from ${bootstrapStack}`,\n          ));\n          doValidate(bootstrapStack.version, this.environment);\n          return;\n        }\n\n        throw new ToolkitError(\n          `This CDK deployment requires bootstrap stack version '${expectedVersion}', but during the confirmation via SSM parameter ${ssmParameterName} the following error occurred: ${e}`,\n        );\n      }\n    }\n\n    // No SSM parameter\n    const bootstrapStack = await this.lookupToolkit();\n    doValidate(bootstrapStack.version, this.environment);\n\n    function doValidate(version: number, environment: Environment) {\n      const notices = Notices.get();\n      if (notices) {\n        // if `Notices` hasn't been initialized there is probably a good\n        // reason for it. handle gracefully.\n        notices.addBootstrappedEnvironment({ bootstrapStackVersion: version, environment });\n      }\n      if (defExpectedVersion > version) {\n        throw new ToolkitError(\n          `This CDK deployment requires bootstrap stack version '${expectedVersion}', found '${version}'. Please run 'cdk bootstrap'.`,\n        );\n      }\n    }\n  }\n\n  /**\n   * Read a version from an SSM parameter, cached\n   */\n  public async versionFromSsmParameter(parameterName: string): Promise<number> {\n    const existing = this.cache.ssmParameters.get(parameterName);\n    if (existing !== undefined) {\n      return existing;\n    }\n\n    const ssm = this.sdk.ssm();\n\n    try {\n      const result = await ssm.getParameter({ Name: parameterName });\n\n      const asNumber = parseInt(`${result.Parameter?.Value}`, 10);\n      if (isNaN(asNumber)) {\n        throw new ToolkitError(`SSM parameter ${parameterName} not a number: ${result.Parameter?.Value}`);\n      }\n\n      this.cache.ssmParameters.set(parameterName, asNumber);\n      return asNumber;\n    } catch (e: any) {\n      if (e.name === 'ParameterNotFound') {\n        throw new ToolkitError(\n          `SSM parameter ${parameterName} not found. Has the environment been bootstrapped? Please run \\'cdk bootstrap\\' (see https://docs.aws.amazon.com/cdk/latest/guide/bootstrapping.html)`,\n        );\n      }\n      throw e;\n    }\n  }\n\n  public async prepareEcrRepository(repositoryName: string): Promise<EcrRepositoryInfo> {\n    if (!this.sdk) {\n      throw new ToolkitError('ToolkitInfo needs to have been initialized with an sdk to call prepareEcrRepository');\n    }\n    const ecr = this.sdk.ecr();\n\n    // check if repo already exists\n    try {\n      await this.msg.ioHost.notify(debug(this.msg.action, `${repositoryName}: checking if ECR repository already exists`));\n      const describeResponse = await ecr.describeRepositories({\n        repositoryNames: [repositoryName],\n      });\n      const existingRepositoryUri = describeResponse.repositories![0]?.repositoryUri;\n      if (existingRepositoryUri) {\n        return { repositoryUri: existingRepositoryUri };\n      }\n    } catch (e: any) {\n      if (e.name !== 'RepositoryNotFoundException') {\n        throw e;\n      }\n    }\n\n    // create the repo (tag it so it will be easier to garbage collect in the future)\n    await this.msg.ioHost.notify(debug(this.msg.action, `${repositoryName}: creating ECR repository`));\n    const assetTag = { Key: 'awscdk:asset', Value: 'true' };\n    const response = await ecr.createRepository({\n      repositoryName,\n      tags: [assetTag],\n    });\n    const repositoryUri = response.repository?.repositoryUri;\n    if (!repositoryUri) {\n      throw new ToolkitError(`CreateRepository did not return a repository URI for ${repositoryUri}`);\n    }\n\n    // configure image scanning on push (helps in identifying software vulnerabilities, no additional charge)\n    await this.msg.ioHost.notify(debug(this.msg.action, `${repositoryName}: enable image scanning`));\n    await ecr.putImageScanningConfiguration({\n      repositoryName,\n      imageScanningConfiguration: { scanOnPush: true },\n    });\n\n    return { repositoryUri };\n  }\n}\n\nexport class NoBootstrapStackEnvironmentResources extends EnvironmentResources {\n  constructor(environment: Environment, sdk: SDK, msg: IoMessaging) {\n    super(environment, sdk, msg, emptyCache());\n  }\n\n  /**\n   * Look up the toolkit for a given environment, using a given SDK\n   */\n  public async lookupToolkit(): Promise<ToolkitInfo> {\n    throw new ToolkitError(\n      'Trying to perform an operation that requires a bootstrap stack; you should not see this error, this is a bug in the CDK CLI.',\n    );\n  }\n}\n\n/**\n * Data that is cached on a per-environment level\n *\n * This cache may be shared between different instances of the `EnvironmentResources` class.\n */\ninterface EnvironmentCache {\n  readonly ssmParameters: Map<string, number>;\n  toolkitInfo?: ToolkitInfo;\n}\n\nfunction emptyCache(): EnvironmentCache {\n  return {\n    ssmParameters: new Map(),\n    toolkitInfo: undefined,\n  };\n}\n\n/**\n * The bootstrap template version that introduced ssm:GetParameter\n */\nconst BOOTSTRAP_TEMPLATE_VERSION_INTRODUCING_GETPARAMETER = 5;\n"]}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./environment-access"), exports);
|
|
18
|
+
__exportStar(require("./environment-resources"), exports);
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsdURBQXFDO0FBQ3JDLDBEQUF3QyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vZW52aXJvbm1lbnQtYWNjZXNzJztcbmV4cG9ydCAqIGZyb20gJy4vZW52aXJvbm1lbnQtcmVzb3VyY2VzJztcbiJdfQ==
|
|
@@ -2,6 +2,7 @@ import * as cxapi from '@aws-cdk/cx-api';
|
|
|
2
2
|
import { Tag } from '@aws-sdk/client-s3';
|
|
3
3
|
import { IS3Client, SDK, SdkProvider } from '../aws-auth';
|
|
4
4
|
import { ActiveAssetCache, BackgroundStackRefresh } from './stack-refresh';
|
|
5
|
+
import { IoMessaging } from '../../toolkit/cli-io-host';
|
|
5
6
|
export declare const S3_ISOLATED_TAG = "aws-cdk:isolated";
|
|
6
7
|
export declare const ECR_ISOLATED_TAG = "aws-cdk.isolated";
|
|
7
8
|
export type GcAsset = ImageAsset | ObjectAsset;
|
|
@@ -72,6 +73,10 @@ interface GarbageCollectorProps {
|
|
|
72
73
|
* Will be used to make SDK calls to CloudFormation, S3, and ECR.
|
|
73
74
|
*/
|
|
74
75
|
readonly sdkProvider: SdkProvider;
|
|
76
|
+
/**
|
|
77
|
+
* Used to send messages.
|
|
78
|
+
*/
|
|
79
|
+
readonly msg: IoMessaging;
|
|
75
80
|
/**
|
|
76
81
|
* The name of the bootstrap stack to look for.
|
|
77
82
|
*
|
|
@@ -96,6 +101,8 @@ export declare class GarbageCollector {
|
|
|
96
101
|
private permissionToTag;
|
|
97
102
|
private bootstrapStackName;
|
|
98
103
|
private confirm;
|
|
104
|
+
private ioHost;
|
|
105
|
+
private action;
|
|
99
106
|
constructor(props: GarbageCollectorProps);
|
|
100
107
|
/**
|
|
101
108
|
* Perform garbage collection on the resolved environment.
|