aws-cdk 2.174.1 → 2.175.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/build-info.json +2 -2
- package/lib/api/aws-auth/provider-caching.js +2 -2
- package/lib/api/cxapp/cloud-executable.d.ts +0 -4
- package/lib/api/cxapp/cloud-executable.js +1 -113
- package/lib/cdk-toolkit.d.ts +1 -0
- package/lib/cdk-toolkit.js +44 -1
- package/lib/cli-arguments.d.ts +55 -3
- package/lib/cli-arguments.js +1 -1
- package/lib/convert-to-cli-args.js +19 -4
- package/lib/index.js +40 -27
- package/lib/init-templates/.init-version.json +1 -1
- package/lib/init-templates/.recommended-feature-flags.json +2 -0
- package/lib/util/yargs-helpers.js +2 -2
- package/package.json +9 -9
- package/test/api/cloud-executable.test.js +1 -52
- package/test/cdk-toolkit.test.js +46 -1
- package/test/cli-arguments.test.js +25 -2
- package/test/util/yargs-helpers.test.d.ts +1 -0
- package/test/util/yargs-helpers.test.js +16 -0
package/build-info.json
CHANGED
|
@@ -14,11 +14,11 @@ const property_provider_1 = require("@smithy/property-provider");
|
|
|
14
14
|
* in caches which will return the cached value until it expires.
|
|
15
15
|
*/
|
|
16
16
|
function makeCachingProvider(provider) {
|
|
17
|
-
return (0, property_provider_1.memoize)(provider, credentialsAboutToExpire, (token) => token.expiration
|
|
17
|
+
return (0, property_provider_1.memoize)(provider, credentialsAboutToExpire, (token) => !!token.expiration);
|
|
18
18
|
}
|
|
19
19
|
function credentialsAboutToExpire(token) {
|
|
20
20
|
const expiryMarginSecs = 5;
|
|
21
21
|
// token.expiration is sometimes null
|
|
22
22
|
return !!token.expiration && token.expiration.getTime() - Date.now() < expiryMarginSecs * 1000;
|
|
23
23
|
}
|
|
24
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
24
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvdmlkZXItY2FjaGluZy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInByb3ZpZGVyLWNhY2hpbmcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFhQSxrREFNQztBQUVELDREQUlDO0FBekJELGlFQUFvRDtBQUdwRDs7Ozs7Ozs7O0dBU0c7QUFDSCxTQUFnQixtQkFBbUIsQ0FBQyxRQUF1QztJQUN6RSxPQUFPLElBQUEsMkJBQU8sRUFDWixRQUFRLEVBQ1Isd0JBQXdCLEVBQ3hCLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLFVBQVUsQ0FDOUIsQ0FBQztBQUNKLENBQUM7QUFFRCxTQUFnQix3QkFBd0IsQ0FBQyxLQUE0QjtJQUNuRSxNQUFNLGdCQUFnQixHQUFHLENBQUMsQ0FBQztJQUMzQixxQ0FBcUM7SUFDckMsT0FBTyxDQUFDLENBQUMsS0FBSyxDQUFDLFVBQVUsSUFBSSxLQUFLLENBQUMsVUFBVSxDQUFDLE9BQU8sRUFBRSxHQUFHLElBQUksQ0FBQyxHQUFHLEVBQUUsR0FBRyxnQkFBZ0IsR0FBRyxJQUFJLENBQUM7QUFDakcsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IG1lbW9pemUgfSBmcm9tICdAc21pdGh5L3Byb3BlcnR5LXByb3ZpZGVyJztcbmltcG9ydCB7IEF3c0NyZWRlbnRpYWxJZGVudGl0eSwgQXdzQ3JlZGVudGlhbElkZW50aXR5UHJvdmlkZXIgfSBmcm9tICdAc21pdGh5L3R5cGVzJztcblxuLyoqXG4gKiBXcmFwIGEgY3JlZGVudGlhbCBwcm92aWRlciBpbiBhIGNhY2hlXG4gKlxuICogU29tZSBjcmVkZW50aWFsIHByb3ZpZGVycyBpbiB0aGUgU0RLdjMgYXJlIGNhY2hlZCAodGhlIGRlZmF1bHQgTm9kZVxuICogY2hhaW4sIHNwZWNpZmljYWxseSkgYnV0IG1vc3Qgb3RoZXJzIGFyZSBub3QuXG4gKlxuICogU2luY2Ugd2Ugd2FudCB0byBhdm9pZCBkdXBsaWNhdGUgY2FsbHMgdG8gYEFzc3VtZVJvbGVgLCBvciBkdXBsaWNhdGVcbiAqIE1GQSBwcm9tcHRzIG9yIHdoYXQgaGF2ZSB5b3UsIHdlIGFyZSBnb2luZyB0byBsaWJlcmFsbHkgd3JhcCBwcm92aWRlcnNcbiAqIGluIGNhY2hlcyB3aGljaCB3aWxsIHJldHVybiB0aGUgY2FjaGVkIHZhbHVlIHVudGlsIGl0IGV4cGlyZXMuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBtYWtlQ2FjaGluZ1Byb3ZpZGVyKHByb3ZpZGVyOiBBd3NDcmVkZW50aWFsSWRlbnRpdHlQcm92aWRlcik6IEF3c0NyZWRlbnRpYWxJZGVudGl0eVByb3ZpZGVyIHtcbiAgcmV0dXJuIG1lbW9pemUoXG4gICAgcHJvdmlkZXIsXG4gICAgY3JlZGVudGlhbHNBYm91dFRvRXhwaXJlLFxuICAgICh0b2tlbikgPT4gISF0b2tlbi5leHBpcmF0aW9uLFxuICApO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gY3JlZGVudGlhbHNBYm91dFRvRXhwaXJlKHRva2VuOiBBd3NDcmVkZW50aWFsSWRlbnRpdHkpIHtcbiAgY29uc3QgZXhwaXJ5TWFyZ2luU2VjcyA9IDU7XG4gIC8vIHRva2VuLmV4cGlyYXRpb24gaXMgc29tZXRpbWVzIG51bGxcbiAgcmV0dXJuICEhdG9rZW4uZXhwaXJhdGlvbiAmJiB0b2tlbi5leHBpcmF0aW9uLmdldFRpbWUoKSAtIERhdGUubm93KCkgPCBleHBpcnlNYXJnaW5TZWNzICogMTAwMDtcbn1cbiJdfQ==
|
|
@@ -40,9 +40,5 @@ export declare class CloudExecutable {
|
|
|
40
40
|
*/
|
|
41
41
|
synthesize(cacheCloudAssembly?: boolean): Promise<CloudAssembly>;
|
|
42
42
|
private doSynthesize;
|
|
43
|
-
/**
|
|
44
|
-
* Modify the templates in the assembly in-place to add metadata resource declarations
|
|
45
|
-
*/
|
|
46
|
-
private addMetadataResource;
|
|
47
43
|
private get canLookup();
|
|
48
44
|
}
|
|
@@ -1,20 +1,10 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.CloudExecutable = void 0;
|
|
4
|
-
const fs_1 = require("fs");
|
|
5
|
-
const cxapi = require("@aws-cdk/cx-api");
|
|
6
|
-
const region_info_1 = require("@aws-cdk/region-info");
|
|
7
|
-
const semver = require("semver");
|
|
8
4
|
const cloud_assembly_1 = require("./cloud-assembly");
|
|
9
5
|
const contextproviders = require("../../context-providers");
|
|
10
6
|
const logging_1 = require("../../logging");
|
|
11
7
|
const error_1 = require("../../toolkit/error");
|
|
12
|
-
/**
|
|
13
|
-
* The Cloud Assembly schema version where the framework started to generate analytics itself
|
|
14
|
-
*
|
|
15
|
-
* See https://github.com/aws/aws-cdk/pull/10306
|
|
16
|
-
*/
|
|
17
|
-
const TEMPLATE_INCLUDES_ANALYTICS_SCHEMA_VERSION = '6.0.0';
|
|
18
8
|
/**
|
|
19
9
|
* Represent the Cloud Executable and the synthesis we can do on it
|
|
20
10
|
*/
|
|
@@ -42,7 +32,6 @@ class CloudExecutable {
|
|
|
42
32
|
return this._cloudAssembly;
|
|
43
33
|
}
|
|
44
34
|
async doSynthesize() {
|
|
45
|
-
const trackVersions = this.props.configuration.settings.get(['versionReporting']);
|
|
46
35
|
// We may need to run the cloud executable multiple times in order to satisfy all missing context
|
|
47
36
|
// (When the executable runs, it will tell us about context it wants to use
|
|
48
37
|
// but it missing. We'll then look up the context and run the executable again, and
|
|
@@ -72,71 +61,9 @@ class CloudExecutable {
|
|
|
72
61
|
continue;
|
|
73
62
|
}
|
|
74
63
|
}
|
|
75
|
-
if (trackVersions && !semver.gte(assembly.version, TEMPLATE_INCLUDES_ANALYTICS_SCHEMA_VERSION)) {
|
|
76
|
-
// @deprecate(v2): the framework now manages its own analytics. For
|
|
77
|
-
// Cloud Assemblies *older* than when we introduced this feature, have
|
|
78
|
-
// the CLI add it. Otherwise, do nothing.
|
|
79
|
-
await this.addMetadataResource(assembly);
|
|
80
|
-
}
|
|
81
64
|
return new cloud_assembly_1.CloudAssembly(assembly);
|
|
82
65
|
}
|
|
83
66
|
}
|
|
84
|
-
/**
|
|
85
|
-
* Modify the templates in the assembly in-place to add metadata resource declarations
|
|
86
|
-
*/
|
|
87
|
-
async addMetadataResource(rootAssembly) {
|
|
88
|
-
if (!rootAssembly.runtime) {
|
|
89
|
-
return;
|
|
90
|
-
}
|
|
91
|
-
const modules = formatModules(rootAssembly.runtime);
|
|
92
|
-
await processAssembly(rootAssembly);
|
|
93
|
-
async function processAssembly(assembly) {
|
|
94
|
-
for (const stack of assembly.stacks) {
|
|
95
|
-
await processStack(stack);
|
|
96
|
-
}
|
|
97
|
-
for (const nested of assembly.nestedAssemblies) {
|
|
98
|
-
await processAssembly(nested.nestedAssembly);
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
async function processStack(stack) {
|
|
102
|
-
const resourcePresent = stack.environment.region === cxapi.UNKNOWN_REGION
|
|
103
|
-
|| region_info_1.RegionInfo.get(stack.environment.region).cdkMetadataResourceAvailable;
|
|
104
|
-
if (!resourcePresent) {
|
|
105
|
-
return;
|
|
106
|
-
}
|
|
107
|
-
if (!stack.template.Resources) {
|
|
108
|
-
stack.template.Resources = {};
|
|
109
|
-
}
|
|
110
|
-
if (stack.template.Resources.CDKMetadata) {
|
|
111
|
-
// Already added by framework, this is expected.
|
|
112
|
-
return;
|
|
113
|
-
}
|
|
114
|
-
stack.template.Resources.CDKMetadata = {
|
|
115
|
-
Type: 'AWS::CDK::Metadata',
|
|
116
|
-
Properties: {
|
|
117
|
-
Modules: modules,
|
|
118
|
-
},
|
|
119
|
-
};
|
|
120
|
-
if (stack.environment.region === cxapi.UNKNOWN_REGION) {
|
|
121
|
-
stack.template.Conditions = stack.template.Conditions || {};
|
|
122
|
-
const condName = 'CDKMetadataAvailable';
|
|
123
|
-
if (!stack.template.Conditions[condName]) {
|
|
124
|
-
stack.template.Conditions[condName] = _makeCdkMetadataAvailableCondition();
|
|
125
|
-
stack.template.Resources.CDKMetadata.Condition = condName;
|
|
126
|
-
}
|
|
127
|
-
else {
|
|
128
|
-
(0, logging_1.warning)(`The stack ${stack.id} already includes a ${condName} condition`);
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
// The template has changed in-memory, but the file on disk remains unchanged so far.
|
|
132
|
-
// The CLI *might* later on deploy the in-memory version (if it's <50kB) or use the
|
|
133
|
-
// on-disk version (if it's >50kB).
|
|
134
|
-
//
|
|
135
|
-
// Be sure to flush the changes we just made back to disk. The on-disk format is always
|
|
136
|
-
// JSON.
|
|
137
|
-
await fs_1.promises.writeFile(stack.templateFullPath, JSON.stringify(stack.template, undefined, 2), { encoding: 'utf-8' });
|
|
138
|
-
}
|
|
139
|
-
}
|
|
140
67
|
get canLookup() {
|
|
141
68
|
return !!(this.props.configuration.settings.get(['lookups']) ?? true);
|
|
142
69
|
}
|
|
@@ -159,43 +86,4 @@ function setsEqual(a, b) {
|
|
|
159
86
|
}
|
|
160
87
|
return true;
|
|
161
88
|
}
|
|
162
|
-
|
|
163
|
-
return _fnOr(region_info_1.RegionInfo.regions
|
|
164
|
-
.filter(ri => ri.cdkMetadataResourceAvailable)
|
|
165
|
-
.map(ri => ({ 'Fn::Equals': [{ Ref: 'AWS::Region' }, ri.name] })));
|
|
166
|
-
}
|
|
167
|
-
/**
|
|
168
|
-
* This takes a bunch of operands and crafts an `Fn::Or` for those. Funny thing is `Fn::Or` requires
|
|
169
|
-
* at least 2 operands and at most 10 operands, so we have to... do this.
|
|
170
|
-
*/
|
|
171
|
-
function _fnOr(operands) {
|
|
172
|
-
if (operands.length === 0) {
|
|
173
|
-
throw new error_1.ToolkitError('Cannot build `Fn::Or` with zero operands!');
|
|
174
|
-
}
|
|
175
|
-
if (operands.length === 1) {
|
|
176
|
-
return operands[0];
|
|
177
|
-
}
|
|
178
|
-
if (operands.length <= 10) {
|
|
179
|
-
return { 'Fn::Or': operands };
|
|
180
|
-
}
|
|
181
|
-
return _fnOr(_inGroupsOf(operands, 10).map(group => _fnOr(group)));
|
|
182
|
-
}
|
|
183
|
-
function _inGroupsOf(array, maxGroup) {
|
|
184
|
-
const result = new Array();
|
|
185
|
-
for (let i = 0; i < array.length; i += maxGroup) {
|
|
186
|
-
result.push(array.slice(i, i + maxGroup));
|
|
187
|
-
}
|
|
188
|
-
return result;
|
|
189
|
-
}
|
|
190
|
-
function formatModules(runtime) {
|
|
191
|
-
const modules = new Array();
|
|
192
|
-
// inject toolkit version to list of modules
|
|
193
|
-
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
|
194
|
-
const toolkitVersion = require('../../../package.json').version;
|
|
195
|
-
modules.push(`aws-cdk=${toolkitVersion}`);
|
|
196
|
-
for (const key of Object.keys(runtime.libraries).sort()) {
|
|
197
|
-
modules.push(`${key}=${runtime.libraries[key]}`);
|
|
198
|
-
}
|
|
199
|
-
return modules.join(',');
|
|
200
|
-
}
|
|
201
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-executable.js","sourceRoot":"","sources":["cloud-executable.ts"],"names":[],"mappings":"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","sourcesContent":["import { promises as fs } from 'fs';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { RegionInfo } from '@aws-cdk/region-info';\nimport * as semver from 'semver';\nimport { CloudAssembly } from './cloud-assembly';\nimport * as contextproviders from '../../context-providers';\nimport { debug, warning } from '../../logging';\nimport { Configuration } from '../../settings';\nimport { ToolkitError } from '../../toolkit/error';\nimport { SdkProvider } from '../aws-auth';\n\n/**\n * @returns output directory\n */\nexport type Synthesizer = (aws: SdkProvider, config: Configuration) => Promise<cxapi.CloudAssembly>;\n\n/**\n * The Cloud Assembly schema version where the framework started to generate analytics itself\n *\n * See https://github.com/aws/aws-cdk/pull/10306\n */\nconst TEMPLATE_INCLUDES_ANALYTICS_SCHEMA_VERSION = '6.0.0';\n\nexport interface CloudExecutableProps {\n  /**\n   * Application configuration (settings and context)\n   */\n  configuration: Configuration;\n\n  /**\n   * AWS object (used by synthesizer and contextprovider)\n   */\n  sdkProvider: SdkProvider;\n\n  /**\n   * Callback invoked to synthesize the actual stacks\n   */\n  synthesizer: Synthesizer;\n}\n\n/**\n * Represent the Cloud Executable and the synthesis we can do on it\n */\nexport class CloudExecutable {\n  private _cloudAssembly?: CloudAssembly;\n\n  constructor(private readonly props: CloudExecutableProps) {\n  }\n\n  /**\n   * Return whether there is an app command from the configuration\n   */\n  public get hasApp() {\n    return !!this.props.configuration.settings.get(['app']);\n  }\n\n  /**\n   * Synthesize a set of stacks.\n   *\n   * @param cacheCloudAssembly whether to cache the Cloud Assembly after it has been first synthesized.\n   *   This is 'true' by default, and only set to 'false' for 'cdk watch',\n   *   which needs to re-synthesize the Assembly each time it detects a change to the project files\n   */\n  public async synthesize(cacheCloudAssembly: boolean = true): Promise<CloudAssembly> {\n    if (!this._cloudAssembly || !cacheCloudAssembly) {\n      this._cloudAssembly = await this.doSynthesize();\n    }\n    return this._cloudAssembly;\n  }\n\n  private async doSynthesize(): Promise<CloudAssembly> {\n    const trackVersions: boolean = this.props.configuration.settings.get(['versionReporting']);\n\n    // We may need to run the cloud executable multiple times in order to satisfy all missing context\n    // (When the executable runs, it will tell us about context it wants to use\n    // but it missing. We'll then look up the context and run the executable again, and\n    // again, until it doesn't complain anymore or we've stopped making progress).\n    let previouslyMissingKeys: Set<string> | undefined;\n    while (true) {\n      const assembly = await this.props.synthesizer(this.props.sdkProvider, this.props.configuration);\n\n      if (assembly.manifest.missing && assembly.manifest.missing.length > 0) {\n        const missingKeys = missingContextKeys(assembly.manifest.missing);\n\n        if (!this.canLookup) {\n          throw new ToolkitError(\n            'Context lookups have been disabled. '\n            + 'Make sure all necessary context is already in \\'cdk.context.json\\' by running \\'cdk synth\\' on a machine with sufficient AWS credentials and committing the result. '\n            + `Missing context keys: '${Array.from(missingKeys).join(', ')}'`);\n        }\n\n        let tryLookup = true;\n        if (previouslyMissingKeys && setsEqual(missingKeys, previouslyMissingKeys)) {\n          debug('Not making progress trying to resolve environmental context. Giving up.');\n          tryLookup = false;\n        }\n\n        previouslyMissingKeys = missingKeys;\n\n        if (tryLookup) {\n          debug('Some context information is missing. Fetching...');\n\n          await contextproviders.provideContextValues(\n            assembly.manifest.missing,\n            this.props.configuration.context,\n            this.props.sdkProvider);\n\n          // Cache the new context to disk\n          await this.props.configuration.saveContext();\n\n          // Execute again\n          continue;\n        }\n      }\n\n      if (trackVersions && !semver.gte(assembly.version, TEMPLATE_INCLUDES_ANALYTICS_SCHEMA_VERSION)) {\n        // @deprecate(v2): the framework now manages its own analytics. For\n        // Cloud Assemblies *older* than when we introduced this feature, have\n        // the CLI add it. Otherwise, do nothing.\n        await this.addMetadataResource(assembly);\n      }\n\n      return new CloudAssembly(assembly);\n    }\n  }\n\n  /**\n   * Modify the templates in the assembly in-place to add metadata resource declarations\n   */\n  private async addMetadataResource(rootAssembly: cxapi.CloudAssembly) {\n    if (!rootAssembly.runtime) { return; }\n\n    const modules = formatModules(rootAssembly.runtime);\n    await processAssembly(rootAssembly);\n\n    async function processAssembly(assembly: cxapi.CloudAssembly) {\n      for (const stack of assembly.stacks) {\n        await processStack(stack);\n      }\n      for (const nested of assembly.nestedAssemblies) {\n        await processAssembly(nested.nestedAssembly);\n      }\n    }\n\n    async function processStack(stack: cxapi.CloudFormationStackArtifact) {\n      const resourcePresent = stack.environment.region === cxapi.UNKNOWN_REGION\n        || RegionInfo.get(stack.environment.region).cdkMetadataResourceAvailable;\n      if (!resourcePresent) { return; }\n\n      if (!stack.template.Resources) {\n        stack.template.Resources = {};\n      }\n      if (stack.template.Resources.CDKMetadata) {\n        // Already added by framework, this is expected.\n        return;\n      }\n\n      stack.template.Resources.CDKMetadata = {\n        Type: 'AWS::CDK::Metadata',\n        Properties: {\n          Modules: modules,\n        },\n      };\n\n      if (stack.environment.region === cxapi.UNKNOWN_REGION) {\n        stack.template.Conditions = stack.template.Conditions || {};\n        const condName = 'CDKMetadataAvailable';\n        if (!stack.template.Conditions[condName]) {\n          stack.template.Conditions[condName] = _makeCdkMetadataAvailableCondition();\n          stack.template.Resources.CDKMetadata.Condition = condName;\n        } else {\n          warning(`The stack ${stack.id} already includes a ${condName} condition`);\n        }\n      }\n\n      // The template has changed in-memory, but the file on disk remains unchanged so far.\n      // The CLI *might* later on deploy the in-memory version (if it's <50kB) or use the\n      // on-disk version (if it's >50kB).\n      //\n      // Be sure to flush the changes we just made back to disk. The on-disk format is always\n      // JSON.\n      await fs.writeFile(stack.templateFullPath, JSON.stringify(stack.template, undefined, 2), { encoding: 'utf-8' });\n    }\n  }\n\n  private get canLookup() {\n    return !!(this.props.configuration.settings.get(['lookups']) ?? true);\n  }\n}\n\n/**\n * Return all keys of missing context items\n */\nfunction missingContextKeys(missing?: cxapi.MissingContext[]): Set<string> {\n  return new Set((missing || []).map(m => m.key));\n}\n\nfunction setsEqual<A>(a: Set<A>, b: Set<A>) {\n  if (a.size !== b.size) { return false; }\n  for (const x of a) {\n    if (!b.has(x)) { return false; }\n  }\n  return true;\n}\n\nfunction _makeCdkMetadataAvailableCondition() {\n  return _fnOr(RegionInfo.regions\n    .filter(ri => ri.cdkMetadataResourceAvailable)\n    .map(ri => ({ 'Fn::Equals': [{ Ref: 'AWS::Region' }, ri.name] })));\n}\n\n/**\n * This takes a bunch of operands and crafts an `Fn::Or` for those. Funny thing is `Fn::Or` requires\n * at least 2 operands and at most 10 operands, so we have to... do this.\n */\nfunction _fnOr(operands: any[]): any {\n  if (operands.length === 0) {\n    throw new ToolkitError('Cannot build `Fn::Or` with zero operands!');\n  }\n  if (operands.length === 1) {\n    return operands[0];\n  }\n  if (operands.length <= 10) {\n    return { 'Fn::Or': operands };\n  }\n  return _fnOr(_inGroupsOf(operands, 10).map(group => _fnOr(group)));\n}\n\nfunction _inGroupsOf<T>(array: T[], maxGroup: number): T[][] {\n  const result = new Array<T[]>();\n  for (let i = 0; i < array.length; i += maxGroup) {\n    result.push(array.slice(i, i + maxGroup));\n  }\n  return result;\n}\n\nfunction formatModules(runtime: cxapi.RuntimeInfo): string {\n  const modules = new Array<string>();\n\n  // inject toolkit version to list of modules\n  // eslint-disable-next-line @typescript-eslint/no-require-imports\n  const toolkitVersion = require('../../../package.json').version;\n  modules.push(`aws-cdk=${toolkitVersion}`);\n\n  for (const key of Object.keys(runtime.libraries).sort()) {\n    modules.push(`${key}=${runtime.libraries[key]}`);\n  }\n  return modules.join(',');\n}\n"]}
|
|
89
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-executable.js","sourceRoot":"","sources":["cloud-executable.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport { CloudAssembly } from './cloud-assembly';\nimport * as contextproviders from '../../context-providers';\nimport { debug } from '../../logging';\nimport { Configuration } from '../../settings';\nimport { ToolkitError } from '../../toolkit/error';\nimport { SdkProvider } from '../aws-auth';\n\n/**\n * @returns output directory\n */\nexport type Synthesizer = (aws: SdkProvider, config: Configuration) => Promise<cxapi.CloudAssembly>;\n\nexport interface CloudExecutableProps {\n  /**\n   * Application configuration (settings and context)\n   */\n  configuration: Configuration;\n\n  /**\n   * AWS object (used by synthesizer and contextprovider)\n   */\n  sdkProvider: SdkProvider;\n\n  /**\n   * Callback invoked to synthesize the actual stacks\n   */\n  synthesizer: Synthesizer;\n}\n\n/**\n * Represent the Cloud Executable and the synthesis we can do on it\n */\nexport class CloudExecutable {\n  private _cloudAssembly?: CloudAssembly;\n\n  constructor(private readonly props: CloudExecutableProps) {\n  }\n\n  /**\n   * Return whether there is an app command from the configuration\n   */\n  public get hasApp() {\n    return !!this.props.configuration.settings.get(['app']);\n  }\n\n  /**\n   * Synthesize a set of stacks.\n   *\n   * @param cacheCloudAssembly whether to cache the Cloud Assembly after it has been first synthesized.\n   *   This is 'true' by default, and only set to 'false' for 'cdk watch',\n   *   which needs to re-synthesize the Assembly each time it detects a change to the project files\n   */\n  public async synthesize(cacheCloudAssembly: boolean = true): Promise<CloudAssembly> {\n    if (!this._cloudAssembly || !cacheCloudAssembly) {\n      this._cloudAssembly = await this.doSynthesize();\n    }\n    return this._cloudAssembly;\n  }\n\n  private async doSynthesize(): Promise<CloudAssembly> {\n    // We may need to run the cloud executable multiple times in order to satisfy all missing context\n    // (When the executable runs, it will tell us about context it wants to use\n    // but it missing. We'll then look up the context and run the executable again, and\n    // again, until it doesn't complain anymore or we've stopped making progress).\n    let previouslyMissingKeys: Set<string> | undefined;\n    while (true) {\n      const assembly = await this.props.synthesizer(this.props.sdkProvider, this.props.configuration);\n\n      if (assembly.manifest.missing && assembly.manifest.missing.length > 0) {\n        const missingKeys = missingContextKeys(assembly.manifest.missing);\n\n        if (!this.canLookup) {\n          throw new ToolkitError(\n            'Context lookups have been disabled. '\n            + 'Make sure all necessary context is already in \\'cdk.context.json\\' by running \\'cdk synth\\' on a machine with sufficient AWS credentials and committing the result. '\n            + `Missing context keys: '${Array.from(missingKeys).join(', ')}'`);\n        }\n\n        let tryLookup = true;\n        if (previouslyMissingKeys && setsEqual(missingKeys, previouslyMissingKeys)) {\n          debug('Not making progress trying to resolve environmental context. Giving up.');\n          tryLookup = false;\n        }\n\n        previouslyMissingKeys = missingKeys;\n\n        if (tryLookup) {\n          debug('Some context information is missing. Fetching...');\n\n          await contextproviders.provideContextValues(\n            assembly.manifest.missing,\n            this.props.configuration.context,\n            this.props.sdkProvider);\n\n          // Cache the new context to disk\n          await this.props.configuration.saveContext();\n\n          // Execute again\n          continue;\n        }\n      }\n\n      return new CloudAssembly(assembly);\n    }\n  }\n\n  private get canLookup() {\n    return !!(this.props.configuration.settings.get(['lookups']) ?? true);\n  }\n}\n\n/**\n * Return all keys of missing context items\n */\nfunction missingContextKeys(missing?: cxapi.MissingContext[]): Set<string> {\n  return new Set((missing || []).map(m => m.key));\n}\n\nfunction setsEqual<A>(a: Set<A>, b: Set<A>) {\n  if (a.size !== b.size) { return false; }\n  for (const x of a) {\n    if (!b.has(x)) { return false; }\n  }\n  return true;\n}\n"]}
|
package/lib/cdk-toolkit.d.ts
CHANGED
|
@@ -121,6 +121,7 @@ export declare class CdkToolkit {
|
|
|
121
121
|
private selectStacksForDeploy;
|
|
122
122
|
private selectStacksForDiff;
|
|
123
123
|
private selectStacksForDestroy;
|
|
124
|
+
private suggestStacks;
|
|
124
125
|
/**
|
|
125
126
|
* Validate the stacks for errors and warnings according to the CLI's current settings
|
|
126
127
|
*/
|