aws-cdk 2.44.0 → 2.46.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 +7 -5
- package/THIRD_PARTY_LICENSES +4 -4
- package/build-info.json +2 -2
- package/lib/api/cxapp/exec.js +45 -4
- package/lib/api/hotswap/ecs-services.js +3 -1
- package/lib/api/hotswap/stepfunctions-state-machines.js +4 -1
- package/lib/commands/doctor.js +2 -2
- package/lib/index.js +17104 -6037
- package/lib/init-templates/app/go/%name%.template.go +1 -1
- package/lib/notices.d.ts +1 -26
- package/lib/notices.js +4 -26
- package/lib/settings.js +3 -3
- package/lib/tree.d.ts +31 -0
- package/lib/tree.js +39 -0
- package/lib/util/objects.d.ts +7 -0
- package/lib/util/objects.js +34 -2
- package/package.json +13 -12
- package/test/api/hotswap/state-machine-hotswap-deployments.test.js +35 -1
- package/test/integ/cli/app/app.js +5 -1
- package/test/integ/common/jest-test.bash +1 -1
- package/test/settings.test.js +5 -5
- package/test/tree.test.d.ts +1 -0
- package/test/tree.test.js +110 -0
- package/test/util/objects.test.js +14 -1
package/README.md
CHANGED
|
@@ -288,7 +288,7 @@ are written to the same output file where each stack artifact ID is a key in the
|
|
|
288
288
|
|
|
289
289
|
|
|
290
290
|
```console
|
|
291
|
-
$ cdk deploy '
|
|
291
|
+
$ cdk deploy '**' --outputs-file "/Users/code/myproject/outputs.json"
|
|
292
292
|
```
|
|
293
293
|
|
|
294
294
|
Example `outputs.json` after deployment of multiple stacks
|
|
@@ -336,19 +336,19 @@ The `progress` key can also be specified as a user setting (`~/.cdk.json`)
|
|
|
336
336
|
|
|
337
337
|
#### CloudFormation Change Sets vs direct stack updates
|
|
338
338
|
|
|
339
|
-
By default CDK
|
|
340
|
-
be deployed
|
|
339
|
+
By default, CDK creates a CloudFormation change set with the changes that will
|
|
340
|
+
be deployed and then executes it. This behavior can be controlled with the
|
|
341
341
|
`--method` parameter:
|
|
342
342
|
|
|
343
343
|
- `--method=change-set` (default): create and execute the change set.
|
|
344
|
-
- `--method=prepare-change-set`: create
|
|
344
|
+
- `--method=prepare-change-set`: create the change set but don't execute it.
|
|
345
345
|
This is useful if you have external tools that will inspect the change set or
|
|
346
346
|
you have an approval process for change sets.
|
|
347
347
|
- `--method=direct`: do not create a change set but apply the change immediately.
|
|
348
348
|
This is typically a bit faster than creating a change set, but it loses
|
|
349
349
|
the progress information.
|
|
350
350
|
|
|
351
|
-
To
|
|
351
|
+
To deploy faster without using change sets:
|
|
352
352
|
|
|
353
353
|
```console
|
|
354
354
|
$ cdk deploy --method=direct
|
|
@@ -410,6 +410,8 @@ For this reason, only use it for development purposes.
|
|
|
410
410
|
**⚠ Note #2**: This command is considered experimental,
|
|
411
411
|
and might have breaking changes in the future.
|
|
412
412
|
|
|
413
|
+
**⚠ Note #3**: Expected defaults for certain parameters may be different with the hotswap parameter. For example, an ECS service's minimum healthy percentage will currently be set to 0. Please review the source accordingly if this occurs.
|
|
414
|
+
|
|
413
415
|
### `cdk watch`
|
|
414
416
|
|
|
415
417
|
The `watch` command is similar to `deploy`,
|
package/THIRD_PARTY_LICENSES
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
The aws-cdk package includes the following third-party software/licensing:
|
|
2
2
|
|
|
3
|
-
** @jsii/check-node@1.
|
|
3
|
+
** @jsii/check-node@1.69.0 - https://www.npmjs.com/package/@jsii/check-node/v/1.69.0 | Apache-2.0
|
|
4
4
|
jsii
|
|
5
5
|
Copyright 2018 Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
6
6
|
|
|
@@ -268,7 +268,7 @@ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH RE
|
|
|
268
268
|
|
|
269
269
|
----------------
|
|
270
270
|
|
|
271
|
-
** aws-sdk@2.
|
|
271
|
+
** aws-sdk@2.1231.0 - https://www.npmjs.com/package/aws-sdk/v/2.1231.0 | Apache-2.0
|
|
272
272
|
AWS SDK for JavaScript
|
|
273
273
|
Copyright 2012-2017 Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
274
274
|
|
|
@@ -3031,7 +3031,7 @@ License, as follows:
|
|
|
3031
3031
|
|
|
3032
3032
|
----------------
|
|
3033
3033
|
|
|
3034
|
-
** semver@7.3.
|
|
3034
|
+
** semver@7.3.8 - https://www.npmjs.com/package/semver/v/7.3.8 | ISC
|
|
3035
3035
|
The ISC License
|
|
3036
3036
|
|
|
3037
3037
|
Copyright (c) Isaac Z. Schlueter and Contributors
|
|
@@ -3113,7 +3113,7 @@ CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
3113
3113
|
|
|
3114
3114
|
----------------
|
|
3115
3115
|
|
|
3116
|
-
** socks@2.7.
|
|
3116
|
+
** socks@2.7.1 - https://www.npmjs.com/package/socks/v/2.7.1 | MIT
|
|
3117
3117
|
The MIT License (MIT)
|
|
3118
3118
|
|
|
3119
3119
|
Copyright (c) 2013 Josh Glazebrook
|
package/build-info.json
CHANGED
package/lib/api/cxapp/exec.js
CHANGED
|
@@ -2,12 +2,16 @@
|
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.execProgram = void 0;
|
|
4
4
|
const childProcess = require("child_process");
|
|
5
|
+
const os = require("os");
|
|
5
6
|
const path = require("path");
|
|
6
7
|
const cxschema = require("@aws-cdk/cloud-assembly-schema");
|
|
7
8
|
const cxapi = require("@aws-cdk/cx-api");
|
|
8
9
|
const fs = require("fs-extra");
|
|
10
|
+
const semver = require("semver");
|
|
9
11
|
const logging_1 = require("../../logging");
|
|
10
12
|
const settings_1 = require("../../settings");
|
|
13
|
+
const tree_1 = require("../../tree");
|
|
14
|
+
const objects_1 = require("../../util/objects");
|
|
11
15
|
const version_1 = require("../../version");
|
|
12
16
|
/** Invokes the cloud executable and returns JSON output */
|
|
13
17
|
async function execProgram(aws, config) {
|
|
@@ -38,10 +42,9 @@ async function execProgram(aws, config) {
|
|
|
38
42
|
if (!stagingEnabled) {
|
|
39
43
|
context[cxapi.DISABLE_ASSET_STAGING_CONTEXT] = true;
|
|
40
44
|
}
|
|
41
|
-
const bundlingStacks = config.settings.get(['bundlingStacks']) ?? ['
|
|
45
|
+
const bundlingStacks = config.settings.get(['bundlingStacks']) ?? ['**'];
|
|
42
46
|
context[cxapi.BUNDLING_STACKS] = bundlingStacks;
|
|
43
47
|
logging_1.debug('context:', context);
|
|
44
|
-
env[cxapi.CONTEXT_ENV] = JSON.stringify(context);
|
|
45
48
|
const build = config.settings.get(['build']);
|
|
46
49
|
if (build) {
|
|
47
50
|
await exec(build);
|
|
@@ -72,8 +75,22 @@ async function execProgram(aws, config) {
|
|
|
72
75
|
env[cxapi.CLI_ASM_VERSION_ENV] = cxschema.Manifest.version();
|
|
73
76
|
env[cxapi.CLI_VERSION_ENV] = version_1.versionNumber();
|
|
74
77
|
logging_1.debug('env:', env);
|
|
78
|
+
const envVariableSizeLimit = os.platform() === 'win32' ? 32760 : 131072;
|
|
79
|
+
const [smallContext, overflow] = objects_1.splitBySize(context, spaceAvailableForContext(env, envVariableSizeLimit));
|
|
80
|
+
// Store the safe part in the environment variable
|
|
81
|
+
env[cxapi.CONTEXT_ENV] = JSON.stringify(smallContext);
|
|
82
|
+
// If there was any overflow, write it to a temporary file
|
|
83
|
+
let contextOverflowLocation;
|
|
84
|
+
if (Object.keys(overflow ?? {}).length > 0) {
|
|
85
|
+
const contextDir = await fs.mkdtemp(path.join(os.tmpdir(), 'cdk-context'));
|
|
86
|
+
contextOverflowLocation = path.join(contextDir, 'context-overflow.json');
|
|
87
|
+
fs.writeJSONSync(contextOverflowLocation, overflow);
|
|
88
|
+
env[cxapi.CONTEXT_OVERFLOW_LOCATION_ENV] = contextOverflowLocation;
|
|
89
|
+
}
|
|
75
90
|
await exec(commandLine.join(' '));
|
|
76
|
-
|
|
91
|
+
const assembly = createAssembly(outdir);
|
|
92
|
+
contextOverflowCleanup(contextOverflowLocation, assembly);
|
|
93
|
+
return assembly;
|
|
77
94
|
function createAssembly(appDir) {
|
|
78
95
|
try {
|
|
79
96
|
return new cxapi.CloudAssembly(appDir);
|
|
@@ -192,4 +209,28 @@ async function guessExecutable(commandLine) {
|
|
|
192
209
|
}
|
|
193
210
|
return commandLine;
|
|
194
211
|
}
|
|
195
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exec.js","sourceRoot":"","sources":["exec.ts"],"names":[],"mappings":"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","sourcesContent":["import * as childProcess from 'child_process';\nimport * as path from 'path';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as fs from 'fs-extra';\nimport { debug } from '../../logging';\nimport { Configuration, PROJECT_CONFIG, USER_DEFAULTS } from '../../settings';\nimport { versionNumber } from '../../version';\nimport { SdkProvider } from '../aws-auth';\n\n/** Invokes the cloud executable and returns JSON output */\nexport async function execProgram(aws: SdkProvider, config: Configuration): Promise<cxapi.CloudAssembly> {\n  const env: { [key: string]: string } = { };\n\n  const context = config.context.all;\n  await populateDefaultEnvironmentIfNeeded(aws, env);\n\n  const debugMode: boolean = config.settings.get(['debug']) ?? true;\n  if (debugMode) {\n    env.CDK_DEBUG = 'true';\n  }\n\n  const pathMetadata: boolean = config.settings.get(['pathMetadata']) ?? true;\n  if (pathMetadata) {\n    context[cxapi.PATH_METADATA_ENABLE_CONTEXT] = true;\n  }\n\n  const assetMetadata: boolean = config.settings.get(['assetMetadata']) ?? true;\n  if (assetMetadata) {\n    context[cxapi.ASSET_RESOURCE_METADATA_ENABLED_CONTEXT] = true;\n  }\n\n  const versionReporting: boolean = config.settings.get(['versionReporting']) ?? true;\n  if (versionReporting) { context[cxapi.ANALYTICS_REPORTING_ENABLED_CONTEXT] = true; }\n  // We need to keep on doing this for framework version from before this flag was deprecated.\n  if (!versionReporting) { context['aws:cdk:disable-version-reporting'] = true; }\n\n  const stagingEnabled = config.settings.get(['staging']) ?? true;\n  if (!stagingEnabled) {\n    context[cxapi.DISABLE_ASSET_STAGING_CONTEXT] = true;\n  }\n\n  const bundlingStacks = config.settings.get(['bundlingStacks']) ?? ['*'];\n  context[cxapi.BUNDLING_STACKS] = bundlingStacks;\n\n  debug('context:', context);\n  env[cxapi.CONTEXT_ENV] = JSON.stringify(context);\n\n  const build = config.settings.get(['build']);\n  if (build) {\n    await exec(build);\n  }\n\n  const app = config.settings.get(['app']);\n  if (!app) {\n    throw new Error(`--app is required either in command-line, in ${PROJECT_CONFIG} or in ${USER_DEFAULTS}`);\n  }\n\n  // bypass \"synth\" if app points to a cloud assembly\n  if (await fs.pathExists(app) && (await fs.stat(app)).isDirectory()) {\n    debug('--app points to a cloud assembly, so we bypass synth');\n    return createAssembly(app);\n  }\n\n  const commandLine = await guessExecutable(appToArray(app));\n\n  const outdir = config.settings.get(['output']);\n  if (!outdir) {\n    throw new Error('unexpected: --output is required');\n  }\n  try {\n    await fs.mkdirp(outdir);\n  } catch (error) {\n    throw new Error(`Could not create output directory ${outdir} (${error.message})`);\n  }\n\n  debug('outdir:', outdir);\n  env[cxapi.OUTDIR_ENV] = outdir;\n\n  // Send version information\n  env[cxapi.CLI_ASM_VERSION_ENV] = cxschema.Manifest.version();\n  env[cxapi.CLI_VERSION_ENV] = versionNumber();\n\n  debug('env:', env);\n\n  await exec(commandLine.join(' '));\n\n  return createAssembly(outdir);\n\n  function createAssembly(appDir: string) {\n    try {\n      return new cxapi.CloudAssembly(appDir);\n    } catch (error) {\n      if (error.message.includes(cxschema.VERSION_MISMATCH)) {\n        // this means the CLI version is too old.\n        // we instruct the user to upgrade.\n        throw new Error(`This CDK CLI is not compatible with the CDK library used by your application. Please upgrade the CLI to the latest version.\\n(${error.message})`);\n      }\n      throw error;\n    }\n  }\n\n  async function exec(commandAndArgs: string) {\n    return new Promise<void>((ok, fail) => {\n      // We use a slightly lower-level interface to:\n      //\n      // - Pass arguments in an array instead of a string, to get around a\n      //   number of quoting issues introduced by the intermediate shell layer\n      //   (which would be different between Linux and Windows).\n      //\n      // - Inherit stderr from controlling terminal. We don't use the captured value\n      //   anyway, and if the subprocess is printing to it for debugging purposes the\n      //   user gets to see it sooner. Plus, capturing doesn't interact nicely with some\n      //   processes like Maven.\n      const proc = childProcess.spawn(commandAndArgs, {\n        stdio: ['ignore', 'inherit', 'inherit'],\n        detached: false,\n        shell: true,\n        env: {\n          ...process.env,\n          ...env,\n        },\n      });\n\n      proc.on('error', fail);\n\n      proc.on('exit', code => {\n        if (code === 0) {\n          return ok();\n        } else {\n          return fail(new Error(`Subprocess exited with error ${code}`));\n        }\n      });\n    });\n  }\n}\n\n/**\n * If we don't have region/account defined in context, we fall back to the default SDK behavior\n * where region is retrieved from ~/.aws/config and account is based on default credentials provider\n * chain and then STS is queried.\n *\n * This is done opportunistically: for example, if we can't access STS for some reason or the region\n * is not configured, the context value will be 'null' and there could failures down the line. In\n * some cases, synthesis does not require region/account information at all, so that might be perfectly\n * fine in certain scenarios.\n *\n * @param context The context key/value bash.\n */\nasync function populateDefaultEnvironmentIfNeeded(aws: SdkProvider, env: { [key: string]: string | undefined}) {\n  env[cxapi.DEFAULT_REGION_ENV] = aws.defaultRegion;\n  debug(`Setting \"${cxapi.DEFAULT_REGION_ENV}\" environment variable to`, env[cxapi.DEFAULT_REGION_ENV]);\n\n  const accountId = (await aws.defaultAccount())?.accountId;\n  if (accountId) {\n    env[cxapi.DEFAULT_ACCOUNT_ENV] = accountId;\n    debug(`Setting \"${cxapi.DEFAULT_ACCOUNT_ENV}\" environment variable to`, env[cxapi.DEFAULT_ACCOUNT_ENV]);\n  }\n}\n\n/**\n * Make sure the 'app' is an array\n *\n * If it's a string, split on spaces as a trivial way of tokenizing the command line.\n */\nfunction appToArray(app: any) {\n  return typeof app === 'string' ? app.split(' ') : app;\n}\n\ntype CommandGenerator = (file: string) => string[];\n\n/**\n * Execute the given file with the same 'node' process as is running the current process\n */\nfunction executeNode(scriptFile: string): string[] {\n  return [process.execPath, scriptFile];\n}\n\n/**\n * Mapping of extensions to command-line generators\n */\nconst EXTENSION_MAP = new Map<string, CommandGenerator>([\n  ['.js', executeNode],\n]);\n\n/**\n * Guess the executable from the command-line argument\n *\n * Only do this if the file is NOT marked as executable. If it is,\n * we'll defer to the shebang inside the file itself.\n *\n * If we're on Windows, we ALWAYS take the handler, since it's hard to\n * verify if registry associations have or have not been set up for this\n * file type, so we'll assume the worst and take control.\n */\nasync function guessExecutable(commandLine: string[]) {\n  if (commandLine.length === 1) {\n    let fstat;\n\n    try {\n      fstat = await fs.stat(commandLine[0]);\n    } catch (error) {\n      debug(`Not a file: '${commandLine[0]}'. Using '${commandLine}' as command-line`);\n      return commandLine;\n    }\n\n    // eslint-disable-next-line no-bitwise\n    const isExecutable = (fstat.mode & fs.constants.X_OK) !== 0;\n    const isWindows = process.platform === 'win32';\n\n    const handler = EXTENSION_MAP.get(path.extname(commandLine[0]));\n    if (handler && (!isExecutable || isWindows)) {\n      return handler(commandLine[0]);\n    }\n  }\n  return commandLine;\n}\n"]}
|
|
212
|
+
function contextOverflowCleanup(location, assembly) {
|
|
213
|
+
if (location) {
|
|
214
|
+
fs.removeSync(path.dirname(location));
|
|
215
|
+
const tree = tree_1.loadTree(assembly);
|
|
216
|
+
const frameworkDoesNotSupportContextOverflow = tree_1.some(tree, node => {
|
|
217
|
+
const fqn = node.constructInfo?.fqn;
|
|
218
|
+
const version = node.constructInfo?.version;
|
|
219
|
+
return (fqn === 'aws-cdk-lib.App' && version != null && semver.lte(version, '2.38.0'))
|
|
220
|
+
|| fqn === '@aws-cdk/core.App'; // v1
|
|
221
|
+
});
|
|
222
|
+
// We're dealing with an old version of the framework here. It is unaware of the temporary
|
|
223
|
+
// file, which means that it will ignore the context overflow.
|
|
224
|
+
if (frameworkDoesNotSupportContextOverflow) {
|
|
225
|
+
logging_1.warning('Part of the context could not be sent to the application. Please update the AWS CDK library to the latest version.');
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
function spaceAvailableForContext(env, limit) {
|
|
230
|
+
const size = (value) => value != null ? Buffer.byteLength(value) : 0;
|
|
231
|
+
const usedSpace = Object.entries(env)
|
|
232
|
+
.map(([k, v]) => k === cxapi.CONTEXT_ENV ? size(k) : size(k) + size(v))
|
|
233
|
+
.reduce((a, b) => a + b, 0);
|
|
234
|
+
return Math.max(0, limit - usedSpace);
|
|
235
|
+
}
|
|
236
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exec.js","sourceRoot":"","sources":["exec.ts"],"names":[],"mappings":"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","sourcesContent":["import * as childProcess from 'child_process';\nimport * as os from 'os';\nimport * as path from 'path';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as fs from 'fs-extra';\nimport * as semver from 'semver';\nimport { debug, warning } from '../../logging';\nimport { Configuration, PROJECT_CONFIG, USER_DEFAULTS } from '../../settings';\nimport { loadTree, some } from '../../tree';\nimport { splitBySize } from '../../util/objects';\nimport { versionNumber } from '../../version';\nimport { SdkProvider } from '../aws-auth';\n\n/** Invokes the cloud executable and returns JSON output */\nexport async function execProgram(aws: SdkProvider, config: Configuration): Promise<cxapi.CloudAssembly> {\n  const env: { [key: string]: string } = { };\n\n  const context = config.context.all;\n  await populateDefaultEnvironmentIfNeeded(aws, env);\n\n  const debugMode: boolean = config.settings.get(['debug']) ?? true;\n  if (debugMode) {\n    env.CDK_DEBUG = 'true';\n  }\n\n  const pathMetadata: boolean = config.settings.get(['pathMetadata']) ?? true;\n  if (pathMetadata) {\n    context[cxapi.PATH_METADATA_ENABLE_CONTEXT] = true;\n  }\n\n  const assetMetadata: boolean = config.settings.get(['assetMetadata']) ?? true;\n  if (assetMetadata) {\n    context[cxapi.ASSET_RESOURCE_METADATA_ENABLED_CONTEXT] = true;\n  }\n\n  const versionReporting: boolean = config.settings.get(['versionReporting']) ?? true;\n  if (versionReporting) { context[cxapi.ANALYTICS_REPORTING_ENABLED_CONTEXT] = true; }\n  // We need to keep on doing this for framework version from before this flag was deprecated.\n  if (!versionReporting) { context['aws:cdk:disable-version-reporting'] = true; }\n\n  const stagingEnabled = config.settings.get(['staging']) ?? true;\n  if (!stagingEnabled) {\n    context[cxapi.DISABLE_ASSET_STAGING_CONTEXT] = true;\n  }\n\n  const bundlingStacks = config.settings.get(['bundlingStacks']) ?? ['**'];\n  context[cxapi.BUNDLING_STACKS] = bundlingStacks;\n\n  debug('context:', context);\n\n  const build = config.settings.get(['build']);\n  if (build) {\n    await exec(build);\n  }\n\n  const app = config.settings.get(['app']);\n  if (!app) {\n    throw new Error(`--app is required either in command-line, in ${PROJECT_CONFIG} or in ${USER_DEFAULTS}`);\n  }\n\n  // bypass \"synth\" if app points to a cloud assembly\n  if (await fs.pathExists(app) && (await fs.stat(app)).isDirectory()) {\n    debug('--app points to a cloud assembly, so we bypass synth');\n    return createAssembly(app);\n  }\n\n  const commandLine = await guessExecutable(appToArray(app));\n\n  const outdir = config.settings.get(['output']);\n  if (!outdir) {\n    throw new Error('unexpected: --output is required');\n  }\n  try {\n    await fs.mkdirp(outdir);\n  } catch (error) {\n    throw new Error(`Could not create output directory ${outdir} (${error.message})`);\n  }\n\n  debug('outdir:', outdir);\n  env[cxapi.OUTDIR_ENV] = outdir;\n\n  // Send version information\n  env[cxapi.CLI_ASM_VERSION_ENV] = cxschema.Manifest.version();\n  env[cxapi.CLI_VERSION_ENV] = versionNumber();\n\n  debug('env:', env);\n\n  const envVariableSizeLimit = os.platform() === 'win32' ? 32760 : 131072;\n  const [smallContext, overflow] = splitBySize(context, spaceAvailableForContext(env, envVariableSizeLimit));\n\n  // Store the safe part in the environment variable\n  env[cxapi.CONTEXT_ENV] = JSON.stringify(smallContext);\n\n  // If there was any overflow, write it to a temporary file\n  let contextOverflowLocation;\n  if (Object.keys(overflow ?? {}).length > 0) {\n    const contextDir = await fs.mkdtemp(path.join(os.tmpdir(), 'cdk-context'));\n    contextOverflowLocation = path.join(contextDir, 'context-overflow.json');\n    fs.writeJSONSync(contextOverflowLocation, overflow);\n    env[cxapi.CONTEXT_OVERFLOW_LOCATION_ENV] = contextOverflowLocation;\n  }\n\n  await exec(commandLine.join(' '));\n\n  const assembly = createAssembly(outdir);\n\n  contextOverflowCleanup(contextOverflowLocation, assembly);\n\n  return assembly;\n\n  function createAssembly(appDir: string) {\n    try {\n      return new cxapi.CloudAssembly(appDir);\n    } catch (error) {\n      if (error.message.includes(cxschema.VERSION_MISMATCH)) {\n        // this means the CLI version is too old.\n        // we instruct the user to upgrade.\n        throw new Error(`This CDK CLI is not compatible with the CDK library used by your application. Please upgrade the CLI to the latest version.\\n(${error.message})`);\n      }\n      throw error;\n    }\n  }\n\n  async function exec(commandAndArgs: string) {\n    return new Promise<void>((ok, fail) => {\n      // We use a slightly lower-level interface to:\n      //\n      // - Pass arguments in an array instead of a string, to get around a\n      //   number of quoting issues introduced by the intermediate shell layer\n      //   (which would be different between Linux and Windows).\n      //\n      // - Inherit stderr from controlling terminal. We don't use the captured value\n      //   anyway, and if the subprocess is printing to it for debugging purposes the\n      //   user gets to see it sooner. Plus, capturing doesn't interact nicely with some\n      //   processes like Maven.\n      const proc = childProcess.spawn(commandAndArgs, {\n        stdio: ['ignore', 'inherit', 'inherit'],\n        detached: false,\n        shell: true,\n        env: {\n          ...process.env,\n          ...env,\n        },\n      });\n\n      proc.on('error', fail);\n\n      proc.on('exit', code => {\n        if (code === 0) {\n          return ok();\n        } else {\n          return fail(new Error(`Subprocess exited with error ${code}`));\n        }\n      });\n    });\n  }\n}\n\n/**\n * If we don't have region/account defined in context, we fall back to the default SDK behavior\n * where region is retrieved from ~/.aws/config and account is based on default credentials provider\n * chain and then STS is queried.\n *\n * This is done opportunistically: for example, if we can't access STS for some reason or the region\n * is not configured, the context value will be 'null' and there could failures down the line. In\n * some cases, synthesis does not require region/account information at all, so that might be perfectly\n * fine in certain scenarios.\n *\n * @param context The context key/value bash.\n */\nasync function populateDefaultEnvironmentIfNeeded(aws: SdkProvider, env: { [key: string]: string | undefined}) {\n  env[cxapi.DEFAULT_REGION_ENV] = aws.defaultRegion;\n  debug(`Setting \"${cxapi.DEFAULT_REGION_ENV}\" environment variable to`, env[cxapi.DEFAULT_REGION_ENV]);\n\n  const accountId = (await aws.defaultAccount())?.accountId;\n  if (accountId) {\n    env[cxapi.DEFAULT_ACCOUNT_ENV] = accountId;\n    debug(`Setting \"${cxapi.DEFAULT_ACCOUNT_ENV}\" environment variable to`, env[cxapi.DEFAULT_ACCOUNT_ENV]);\n  }\n}\n\n/**\n * Make sure the 'app' is an array\n *\n * If it's a string, split on spaces as a trivial way of tokenizing the command line.\n */\nfunction appToArray(app: any) {\n  return typeof app === 'string' ? app.split(' ') : app;\n}\n\ntype CommandGenerator = (file: string) => string[];\n\n/**\n * Execute the given file with the same 'node' process as is running the current process\n */\nfunction executeNode(scriptFile: string): string[] {\n  return [process.execPath, scriptFile];\n}\n\n/**\n * Mapping of extensions to command-line generators\n */\nconst EXTENSION_MAP = new Map<string, CommandGenerator>([\n  ['.js', executeNode],\n]);\n\n/**\n * Guess the executable from the command-line argument\n *\n * Only do this if the file is NOT marked as executable. If it is,\n * we'll defer to the shebang inside the file itself.\n *\n * If we're on Windows, we ALWAYS take the handler, since it's hard to\n * verify if registry associations have or have not been set up for this\n * file type, so we'll assume the worst and take control.\n */\nasync function guessExecutable(commandLine: string[]) {\n  if (commandLine.length === 1) {\n    let fstat;\n\n    try {\n      fstat = await fs.stat(commandLine[0]);\n    } catch (error) {\n      debug(`Not a file: '${commandLine[0]}'. Using '${commandLine}' as command-line`);\n      return commandLine;\n    }\n\n    // eslint-disable-next-line no-bitwise\n    const isExecutable = (fstat.mode & fs.constants.X_OK) !== 0;\n    const isWindows = process.platform === 'win32';\n\n    const handler = EXTENSION_MAP.get(path.extname(commandLine[0]));\n    if (handler && (!isExecutable || isWindows)) {\n      return handler(commandLine[0]);\n    }\n  }\n  return commandLine;\n}\n\nfunction contextOverflowCleanup(location: string | undefined, assembly: cxapi.CloudAssembly) {\n  if (location) {\n    fs.removeSync(path.dirname(location));\n\n    const tree = loadTree(assembly);\n    const frameworkDoesNotSupportContextOverflow = some(tree, node => {\n      const fqn = node.constructInfo?.fqn;\n      const version = node.constructInfo?.version;\n      return (fqn === 'aws-cdk-lib.App' && version != null && semver.lte(version, '2.38.0'))\n        || fqn === '@aws-cdk/core.App'; // v1\n    });\n\n    // We're dealing with an old version of the framework here. It is unaware of the temporary\n    // file, which means that it will ignore the context overflow.\n    if (frameworkDoesNotSupportContextOverflow) {\n      warning('Part of the context could not be sent to the application. Please update the AWS CDK library to the latest version.');\n    }\n  }\n}\n\nfunction spaceAvailableForContext(env: { [key: string]: string }, limit: number) {\n  const size = (value: string) => value != null ? Buffer.byteLength(value) : 0;\n\n  const usedSpace = Object.entries(env)\n    .map(([k, v]) => k === cxapi.CONTEXT_ENV ? size(k) : size(k) + size(v))\n    .reduce((a, b) => a + b, 0);\n\n  return Math.max(0, limit - usedSpace);\n}"]}
|
|
@@ -112,6 +112,8 @@ class EcsServiceHotswapOperation {
|
|
|
112
112
|
clusterPromises = [];
|
|
113
113
|
servicePerClusterUpdates[clusterName] = clusterPromises;
|
|
114
114
|
}
|
|
115
|
+
// Forcing New Deployment and setting Minimum Healthy Percent to 0.
|
|
116
|
+
// As CDK HotSwap is development only, this seems the most efficient way to ensure all tasks are replaced immediately, regardless of original amount.
|
|
115
117
|
clusterPromises.push({
|
|
116
118
|
promise: sdk.ecs().updateService({
|
|
117
119
|
service: ecsService.serviceArn,
|
|
@@ -174,4 +176,4 @@ class EcsServiceHotswapOperation {
|
|
|
174
176
|
}));
|
|
175
177
|
}
|
|
176
178
|
}
|
|
177
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ecs-services.js","sourceRoot":"","sources":["ecs-services.ts"],"names":[],"mappings":"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","sourcesContent":["import * as AWS from 'aws-sdk';\nimport { ISDK } from '../aws-auth';\nimport { EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';\nimport { ChangeHotswapImpact, ChangeHotswapResult, HotswapOperation, HotswappableChangeCandidate, lowerCaseFirstCharacter, transformObjectKeys } from './common';\n\nexport async function isHotswappableEcsServiceChange(\n  logicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<ChangeHotswapResult> {\n  // the only resource change we should allow is an ECS TaskDefinition\n  if (change.newValue.Type !== 'AWS::ECS::TaskDefinition') {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  for (const updatedPropName in change.propertyUpdates) {\n    // We only allow a change in the ContainerDefinitions of the TaskDefinition for now -\n    // it contains the image and environment variables, so seems like a safe bet for now.\n    // We might revisit this decision in the future though!\n    if (updatedPropName !== 'ContainerDefinitions') {\n      return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n    }\n    const containerDefinitionsDifference = (change.propertyUpdates)[updatedPropName];\n    if (containerDefinitionsDifference.newValue === undefined) {\n      return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n    }\n  }\n  // at this point, we know the TaskDefinition can be hotswapped\n\n  // find all ECS Services that reference the TaskDefinition that changed\n  const resourcesReferencingTaskDef = evaluateCfnTemplate.findReferencesTo(logicalId);\n  const ecsServiceResourcesReferencingTaskDef = resourcesReferencingTaskDef.filter(r => r.Type === 'AWS::ECS::Service');\n  const ecsServicesReferencingTaskDef = new Array<EcsService>();\n  for (const ecsServiceResource of ecsServiceResourcesReferencingTaskDef) {\n    const serviceArn = await evaluateCfnTemplate.findPhysicalNameFor(ecsServiceResource.LogicalId);\n    if (serviceArn) {\n      ecsServicesReferencingTaskDef.push({ serviceArn });\n    }\n  }\n  if (ecsServicesReferencingTaskDef.length === 0 ||\n      resourcesReferencingTaskDef.length > ecsServicesReferencingTaskDef.length) {\n    // if there are either no resources referencing the TaskDefinition,\n    // or something besides an ECS Service is referencing it,\n    // hotswap is not possible\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  const taskDefinitionResource = change.newValue.Properties;\n  // first, let's get the name of the family\n  const familyNameOrArn = await evaluateCfnTemplate.establishResourcePhysicalName(logicalId, taskDefinitionResource?.Family);\n  if (!familyNameOrArn) {\n    // if the Family property has not bee provided, and we can't find it in the current Stack,\n    // this means hotswapping is not possible\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n  // the physical name of the Task Definition in CloudFormation includes its current revision number at the end,\n  // remove it if needed\n  const familyNameOrArnParts = familyNameOrArn.split(':');\n  const family = familyNameOrArnParts.length > 1\n    // familyNameOrArn is actually an ARN, of the format 'arn:aws:ecs:region:account:task-definition/<family-name>:<revision-nr>'\n    // so, take the 6th element, at index 5, and split it on '/'\n    ? familyNameOrArnParts[5].split('/')[1]\n    // otherwise, familyNameOrArn is just the simple name evaluated from the CloudFormation template\n    : familyNameOrArn;\n  // then, let's evaluate the body of the remainder of the TaskDef (without the Family property)\n  const evaluatedTaskDef = {\n    ...await evaluateCfnTemplate.evaluateCfnExpression({\n      ...(taskDefinitionResource ?? {}),\n      Family: undefined,\n    }),\n    Family: family,\n  };\n  return new EcsServiceHotswapOperation(evaluatedTaskDef, ecsServicesReferencingTaskDef);\n}\n\ninterface EcsService {\n  readonly serviceArn: string;\n}\n\nclass EcsServiceHotswapOperation implements HotswapOperation {\n  public readonly service = 'ecs-service';\n  public readonly resourceNames: string[] = [];\n\n  constructor(\n    private readonly taskDefinitionResource: any,\n    private readonly servicesReferencingTaskDef: EcsService[],\n  ) {\n    this.resourceNames = servicesReferencingTaskDef.map(ecsService =>\n      `ECS Service '${ecsService.serviceArn.split('/')[2]}'`);\n  }\n\n  public async apply(sdk: ISDK): Promise<any> {\n    // Step 1 - update the changed TaskDefinition, creating a new TaskDefinition Revision\n    // we need to lowercase the evaluated TaskDef from CloudFormation,\n    // as the AWS SDK uses lowercase property names for these\n    const lowercasedTaskDef = transformObjectKeys(this.taskDefinitionResource, lowerCaseFirstCharacter, {\n      // All the properties that take arbitrary string as keys i.e. { \"string\" : \"string\" }\n      // https://docs.aws.amazon.com/AmazonECS/latest/APIReference/API_RegisterTaskDefinition.html#API_RegisterTaskDefinition_RequestSyntax\n      ContainerDefinitions: {\n        DockerLabels: true,\n        FirelensConfiguration: {\n          Options: true,\n        },\n        LogConfiguration: {\n          Options: true,\n        },\n      },\n      Volumes: {\n        DockerVolumeConfiguration: {\n          DriverOpts: true,\n          Labels: true,\n        },\n      },\n    });\n    const registerTaskDefResponse = await sdk.ecs().registerTaskDefinition(lowercasedTaskDef).promise();\n    const taskDefRevArn = registerTaskDefResponse.taskDefinition?.taskDefinitionArn;\n\n    // Step 2 - update the services using that TaskDefinition to point to the new TaskDefinition Revision\n    const servicePerClusterUpdates: { [cluster: string]: Array<{ promise: Promise<any>, ecsService: EcsService }> } = {};\n    for (const ecsService of this.servicesReferencingTaskDef) {\n      const clusterName = ecsService.serviceArn.split('/')[1];\n\n      const existingClusterPromises = servicePerClusterUpdates[clusterName];\n      let clusterPromises: Array<{ promise: Promise<any>, ecsService: EcsService }>;\n      if (existingClusterPromises) {\n        clusterPromises = existingClusterPromises;\n      } else {\n        clusterPromises = [];\n        servicePerClusterUpdates[clusterName] = clusterPromises;\n      }\n\n      clusterPromises.push({\n        promise: sdk.ecs().updateService({\n          service: ecsService.serviceArn,\n          taskDefinition: taskDefRevArn,\n          cluster: clusterName,\n          forceNewDeployment: true,\n          deploymentConfiguration: {\n            minimumHealthyPercent: 0,\n          },\n        }).promise(),\n        ecsService: ecsService,\n      });\n    }\n    await Promise.all(Object.values(servicePerClusterUpdates)\n      .map(clusterUpdates => {\n        return Promise.all(clusterUpdates.map(serviceUpdate => serviceUpdate.promise));\n      }),\n    );\n\n    // Step 3 - wait for the service deployments triggered in Step 2 to finish\n    // configure a custom Waiter\n    (sdk.ecs() as any).api.waiters.deploymentToFinish = {\n      name: 'DeploymentToFinish',\n      operation: 'describeServices',\n      delay: 10,\n      maxAttempts: 60,\n      acceptors: [\n        {\n          matcher: 'pathAny',\n          argument: 'failures[].reason',\n          expected: 'MISSING',\n          state: 'failure',\n        },\n        {\n          matcher: 'pathAny',\n          argument: 'services[].status',\n          expected: 'DRAINING',\n          state: 'failure',\n        },\n        {\n          matcher: 'pathAny',\n          argument: 'services[].status',\n          expected: 'INACTIVE',\n          state: 'failure',\n        },\n        {\n          matcher: 'path',\n          argument: \"length(services[].deployments[? status == 'PRIMARY' && runningCount < desiredCount][]) == `0`\",\n          expected: true,\n          state: 'success',\n        },\n      ],\n    };\n    // create a custom Waiter that uses the deploymentToFinish configuration added above\n    const deploymentWaiter = new (AWS as any).ResourceWaiter(sdk.ecs(), 'deploymentToFinish');\n    // wait for all of the waiters to finish\n    return Promise.all(Object.entries(servicePerClusterUpdates).map(([clusterName, serviceUpdates]) => {\n      return deploymentWaiter.wait({\n        cluster: clusterName,\n        services: serviceUpdates.map(serviceUpdate => serviceUpdate.ecsService.serviceArn),\n      }).promise();\n    }));\n  }\n}\n"]}
|
|
179
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ecs-services.js","sourceRoot":"","sources":["ecs-services.ts"],"names":[],"mappings":"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","sourcesContent":["import * as AWS from 'aws-sdk';\nimport { ISDK } from '../aws-auth';\nimport { EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';\nimport { ChangeHotswapImpact, ChangeHotswapResult, HotswapOperation, HotswappableChangeCandidate, lowerCaseFirstCharacter, transformObjectKeys } from './common';\n\nexport async function isHotswappableEcsServiceChange(\n  logicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<ChangeHotswapResult> {\n  // the only resource change we should allow is an ECS TaskDefinition\n  if (change.newValue.Type !== 'AWS::ECS::TaskDefinition') {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  for (const updatedPropName in change.propertyUpdates) {\n    // We only allow a change in the ContainerDefinitions of the TaskDefinition for now -\n    // it contains the image and environment variables, so seems like a safe bet for now.\n    // We might revisit this decision in the future though!\n    if (updatedPropName !== 'ContainerDefinitions') {\n      return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n    }\n    const containerDefinitionsDifference = (change.propertyUpdates)[updatedPropName];\n    if (containerDefinitionsDifference.newValue === undefined) {\n      return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n    }\n  }\n  // at this point, we know the TaskDefinition can be hotswapped\n\n  // find all ECS Services that reference the TaskDefinition that changed\n  const resourcesReferencingTaskDef = evaluateCfnTemplate.findReferencesTo(logicalId);\n  const ecsServiceResourcesReferencingTaskDef = resourcesReferencingTaskDef.filter(r => r.Type === 'AWS::ECS::Service');\n  const ecsServicesReferencingTaskDef = new Array<EcsService>();\n  for (const ecsServiceResource of ecsServiceResourcesReferencingTaskDef) {\n    const serviceArn = await evaluateCfnTemplate.findPhysicalNameFor(ecsServiceResource.LogicalId);\n    if (serviceArn) {\n      ecsServicesReferencingTaskDef.push({ serviceArn });\n    }\n  }\n  if (ecsServicesReferencingTaskDef.length === 0 ||\n      resourcesReferencingTaskDef.length > ecsServicesReferencingTaskDef.length) {\n    // if there are either no resources referencing the TaskDefinition,\n    // or something besides an ECS Service is referencing it,\n    // hotswap is not possible\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  const taskDefinitionResource = change.newValue.Properties;\n  // first, let's get the name of the family\n  const familyNameOrArn = await evaluateCfnTemplate.establishResourcePhysicalName(logicalId, taskDefinitionResource?.Family);\n  if (!familyNameOrArn) {\n    // if the Family property has not bee provided, and we can't find it in the current Stack,\n    // this means hotswapping is not possible\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n  // the physical name of the Task Definition in CloudFormation includes its current revision number at the end,\n  // remove it if needed\n  const familyNameOrArnParts = familyNameOrArn.split(':');\n  const family = familyNameOrArnParts.length > 1\n    // familyNameOrArn is actually an ARN, of the format 'arn:aws:ecs:region:account:task-definition/<family-name>:<revision-nr>'\n    // so, take the 6th element, at index 5, and split it on '/'\n    ? familyNameOrArnParts[5].split('/')[1]\n    // otherwise, familyNameOrArn is just the simple name evaluated from the CloudFormation template\n    : familyNameOrArn;\n  // then, let's evaluate the body of the remainder of the TaskDef (without the Family property)\n  const evaluatedTaskDef = {\n    ...await evaluateCfnTemplate.evaluateCfnExpression({\n      ...(taskDefinitionResource ?? {}),\n      Family: undefined,\n    }),\n    Family: family,\n  };\n  return new EcsServiceHotswapOperation(evaluatedTaskDef, ecsServicesReferencingTaskDef);\n}\n\ninterface EcsService {\n  readonly serviceArn: string;\n}\n\nclass EcsServiceHotswapOperation implements HotswapOperation {\n  public readonly service = 'ecs-service';\n  public readonly resourceNames: string[] = [];\n\n  constructor(\n    private readonly taskDefinitionResource: any,\n    private readonly servicesReferencingTaskDef: EcsService[],\n  ) {\n    this.resourceNames = servicesReferencingTaskDef.map(ecsService =>\n      `ECS Service '${ecsService.serviceArn.split('/')[2]}'`);\n  }\n\n  public async apply(sdk: ISDK): Promise<any> {\n    // Step 1 - update the changed TaskDefinition, creating a new TaskDefinition Revision\n    // we need to lowercase the evaluated TaskDef from CloudFormation,\n    // as the AWS SDK uses lowercase property names for these\n    const lowercasedTaskDef = transformObjectKeys(this.taskDefinitionResource, lowerCaseFirstCharacter, {\n      // All the properties that take arbitrary string as keys i.e. { \"string\" : \"string\" }\n      // https://docs.aws.amazon.com/AmazonECS/latest/APIReference/API_RegisterTaskDefinition.html#API_RegisterTaskDefinition_RequestSyntax\n      ContainerDefinitions: {\n        DockerLabels: true,\n        FirelensConfiguration: {\n          Options: true,\n        },\n        LogConfiguration: {\n          Options: true,\n        },\n      },\n      Volumes: {\n        DockerVolumeConfiguration: {\n          DriverOpts: true,\n          Labels: true,\n        },\n      },\n    });\n    const registerTaskDefResponse = await sdk.ecs().registerTaskDefinition(lowercasedTaskDef).promise();\n    const taskDefRevArn = registerTaskDefResponse.taskDefinition?.taskDefinitionArn;\n\n    // Step 2 - update the services using that TaskDefinition to point to the new TaskDefinition Revision\n    const servicePerClusterUpdates: { [cluster: string]: Array<{ promise: Promise<any>, ecsService: EcsService }> } = {};\n    for (const ecsService of this.servicesReferencingTaskDef) {\n      const clusterName = ecsService.serviceArn.split('/')[1];\n\n      const existingClusterPromises = servicePerClusterUpdates[clusterName];\n      let clusterPromises: Array<{ promise: Promise<any>, ecsService: EcsService }>;\n      if (existingClusterPromises) {\n        clusterPromises = existingClusterPromises;\n      } else {\n        clusterPromises = [];\n        servicePerClusterUpdates[clusterName] = clusterPromises;\n      }\n      // Forcing New Deployment and setting Minimum Healthy Percent to 0.\n      // As CDK HotSwap is development only, this seems the most efficient way to ensure all tasks are replaced immediately, regardless of original amount.\n      clusterPromises.push({\n        promise: sdk.ecs().updateService({\n          service: ecsService.serviceArn,\n          taskDefinition: taskDefRevArn,\n          cluster: clusterName,\n          forceNewDeployment: true,\n          deploymentConfiguration: {\n            minimumHealthyPercent: 0,\n          },\n        }).promise(),\n        ecsService: ecsService,\n      });\n    }\n    await Promise.all(Object.values(servicePerClusterUpdates)\n      .map(clusterUpdates => {\n        return Promise.all(clusterUpdates.map(serviceUpdate => serviceUpdate.promise));\n      }),\n    );\n\n    // Step 3 - wait for the service deployments triggered in Step 2 to finish\n    // configure a custom Waiter\n    (sdk.ecs() as any).api.waiters.deploymentToFinish = {\n      name: 'DeploymentToFinish',\n      operation: 'describeServices',\n      delay: 10,\n      maxAttempts: 60,\n      acceptors: [\n        {\n          matcher: 'pathAny',\n          argument: 'failures[].reason',\n          expected: 'MISSING',\n          state: 'failure',\n        },\n        {\n          matcher: 'pathAny',\n          argument: 'services[].status',\n          expected: 'DRAINING',\n          state: 'failure',\n        },\n        {\n          matcher: 'pathAny',\n          argument: 'services[].status',\n          expected: 'INACTIVE',\n          state: 'failure',\n        },\n        {\n          matcher: 'path',\n          argument: \"length(services[].deployments[? status == 'PRIMARY' && runningCount < desiredCount][]) == `0`\",\n          expected: true,\n          state: 'success',\n        },\n      ],\n    };\n    // create a custom Waiter that uses the deploymentToFinish configuration added above\n    const deploymentWaiter = new (AWS as any).ResourceWaiter(sdk.ecs(), 'deploymentToFinish');\n    // wait for all of the waiters to finish\n    return Promise.all(Object.entries(servicePerClusterUpdates).map(([clusterName, serviceUpdates]) => {\n      return deploymentWaiter.wait({\n        cluster: clusterName,\n        services: serviceUpdates.map(serviceUpdate => serviceUpdate.ecsService.serviceArn),\n      }).promise();\n    }));\n  }\n}\n"]}
|
|
@@ -29,6 +29,9 @@ async function isStateMachineDefinitionOnlyChange(change, evaluateCfnTemplate) {
|
|
|
29
29
|
return common_1.ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;
|
|
30
30
|
}
|
|
31
31
|
const propertyUpdates = change.propertyUpdates;
|
|
32
|
+
if (Object.keys(propertyUpdates).length === 0) {
|
|
33
|
+
return common_1.ChangeHotswapImpact.IRRELEVANT;
|
|
34
|
+
}
|
|
32
35
|
for (const updatedPropName in propertyUpdates) {
|
|
33
36
|
// ensure that only changes to the definition string result in a hotswap
|
|
34
37
|
if (updatedPropName !== 'DefinitionString') {
|
|
@@ -51,4 +54,4 @@ class StateMachineHotswapOperation {
|
|
|
51
54
|
}).promise();
|
|
52
55
|
}
|
|
53
56
|
}
|
|
54
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
57
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RlcGZ1bmN0aW9ucy1zdGF0ZS1tYWNoaW5lcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInN0ZXBmdW5jdGlvbnMtc3RhdGUtbWFjaGluZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBRUEscUNBQW1IO0FBRTVHLEtBQUssVUFBVSxnQ0FBZ0MsQ0FDcEQsU0FBaUIsRUFBRSxNQUFtQyxFQUFFLG1CQUFtRDtJQUUzRyxNQUFNLDRCQUE0QixHQUFHLE1BQU0sa0NBQWtDLENBQUMsTUFBTSxFQUFFLG1CQUFtQixDQUFDLENBQUM7SUFDM0csSUFBSSw0QkFBNEIsS0FBSyw0QkFBbUIsQ0FBQyx3QkFBd0I7UUFDN0UsNEJBQTRCLEtBQUssNEJBQW1CLENBQUMsVUFBVSxFQUFFO1FBQ25FLE9BQU8sNEJBQTRCLENBQUM7S0FDckM7SUFFRCxNQUFNLDZCQUE2QixHQUFHLE1BQU0sQ0FBQyxRQUFRLEVBQUUsVUFBVSxFQUFFLGdCQUFnQixDQUFDO0lBQ3BGLE1BQU0sZUFBZSxHQUFHLDZCQUE2QjtRQUNuRCxDQUFDLENBQUMsTUFBTSxtQkFBbUIsQ0FBQyxxQkFBcUIsQ0FBQztZQUNoRCxTQUFTLEVBQUUsNkVBQTZFLEdBQUcsNkJBQTZCO1NBQ3pILENBQUM7UUFDRixDQUFDLENBQUMsTUFBTSxtQkFBbUIsQ0FBQyxtQkFBbUIsQ0FBQyxTQUFTLENBQUMsQ0FBQztJQUU3RCxJQUFJLENBQUMsZUFBZSxFQUFFO1FBQ3BCLE9BQU8sNEJBQW1CLENBQUMsd0JBQXdCLENBQUM7S0FDckQ7SUFFRCxPQUFPLElBQUksNEJBQTRCLENBQUM7UUFDdEMsVUFBVSxFQUFFLDRCQUE0QjtRQUN4QyxlQUFlLEVBQUUsZUFBZTtLQUNqQyxDQUFDLENBQUM7QUFDTCxDQUFDO0FBeEJELDRFQXdCQztBQUVELEtBQUssVUFBVSxrQ0FBa0MsQ0FDL0MsTUFBbUMsRUFBRSxtQkFBbUQ7SUFFeEYsTUFBTSxlQUFlLEdBQUcsTUFBTSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUM7SUFDN0MsSUFBSSxlQUFlLEtBQUssa0NBQWtDLEVBQUU7UUFDMUQsT0FBTyw0QkFBbUIsQ0FBQyx3QkFBd0IsQ0FBQztLQUNyRDtJQUVELE1BQU0sZUFBZSxHQUFHLE1BQU0sQ0FBQyxlQUFlLENBQUM7SUFDL0MsSUFBSSxNQUFNLENBQUMsSUFBSSxDQUFDLGVBQWUsQ0FBQyxDQUFDLE1BQU0sS0FBSyxDQUFDLEVBQUU7UUFDN0MsT0FBTyw0QkFBbUIsQ0FBQyxVQUFVLENBQUM7S0FDdkM7SUFFRCxLQUFLLE1BQU0sZUFBZSxJQUFJLGVBQWUsRUFBRTtRQUM3Qyx3RUFBd0U7UUFDeEUsSUFBSSxlQUFlLEtBQUssa0JBQWtCLEVBQUU7WUFDMUMsT0FBTyw0QkFBbUIsQ0FBQyx3QkFBd0IsQ0FBQztTQUNyRDtLQUNGO0lBRUQsT0FBTyxtQkFBbUIsQ0FBQyxxQkFBcUIsQ0FBQyxlQUFlLENBQUMsZ0JBQWdCLENBQUMsUUFBUSxDQUFDLENBQUM7QUFDOUYsQ0FBQztBQU9ELE1BQU0sNEJBQTRCO0lBSWhDLFlBQTZCLG9CQUEwQztRQUExQyx5QkFBb0IsR0FBcEIsb0JBQW9CLENBQXNCO1FBSHZELFlBQU8sR0FBRyw2QkFBNkIsQ0FBQztRQUl0RCxJQUFJLENBQUMsYUFBYSxHQUFHLENBQUMsaUJBQWlCLElBQUksQ0FBQyxvQkFBb0IsQ0FBQyxlQUFlLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUNyRyxDQUFDO0lBRU0sS0FBSyxDQUFDLEtBQUssQ0FBQyxHQUFTO1FBQzFCLDREQUE0RDtRQUM1RCxPQUFPLEdBQUcsQ0FBQyxhQUFhLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQztZQUM1QyxlQUFlLEVBQUUsSUFBSSxDQUFDLG9CQUFvQixDQUFDLGVBQWU7WUFDMUQsVUFBVSxFQUFFLElBQUksQ0FBQyxvQkFBb0IsQ0FBQyxVQUFVO1NBQ2pELENBQUMsQ0FBQyxPQUFPLEVBQUUsQ0FBQztJQUNmLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IElTREsgfSBmcm9tICcuLi9hd3MtYXV0aCc7XG5pbXBvcnQgeyBFdmFsdWF0ZUNsb3VkRm9ybWF0aW9uVGVtcGxhdGUgfSBmcm9tICcuLi9ldmFsdWF0ZS1jbG91ZGZvcm1hdGlvbi10ZW1wbGF0ZSc7XG5pbXBvcnQgeyBDaGFuZ2VIb3Rzd2FwSW1wYWN0LCBDaGFuZ2VIb3Rzd2FwUmVzdWx0LCBIb3Rzd2FwT3BlcmF0aW9uLCBIb3Rzd2FwcGFibGVDaGFuZ2VDYW5kaWRhdGUgfSBmcm9tICcuL2NvbW1vbic7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBpc0hvdHN3YXBwYWJsZVN0YXRlTWFjaGluZUNoYW5nZShcbiAgbG9naWNhbElkOiBzdHJpbmcsIGNoYW5nZTogSG90c3dhcHBhYmxlQ2hhbmdlQ2FuZGlkYXRlLCBldmFsdWF0ZUNmblRlbXBsYXRlOiBFdmFsdWF0ZUNsb3VkRm9ybWF0aW9uVGVtcGxhdGUsXG4pOiBQcm9taXNlPENoYW5nZUhvdHN3YXBSZXN1bHQ+IHtcbiAgY29uc3Qgc3RhdGVNYWNoaW5lRGVmaW5pdGlvbkNoYW5nZSA9IGF3YWl0IGlzU3RhdGVNYWNoaW5lRGVmaW5pdGlvbk9ubHlDaGFuZ2UoY2hhbmdlLCBldmFsdWF0ZUNmblRlbXBsYXRlKTtcbiAgaWYgKHN0YXRlTWFjaGluZURlZmluaXRpb25DaGFuZ2UgPT09IENoYW5nZUhvdHN3YXBJbXBhY3QuUkVRVUlSRVNfRlVMTF9ERVBMT1lNRU5UIHx8XG4gICAgICBzdGF0ZU1hY2hpbmVEZWZpbml0aW9uQ2hhbmdlID09PSBDaGFuZ2VIb3Rzd2FwSW1wYWN0LklSUkVMRVZBTlQpIHtcbiAgICByZXR1cm4gc3RhdGVNYWNoaW5lRGVmaW5pdGlvbkNoYW5nZTtcbiAgfVxuXG4gIGNvbnN0IHN0YXRlTWFjaGluZU5hbWVJbkNmblRlbXBsYXRlID0gY2hhbmdlLm5ld1ZhbHVlPy5Qcm9wZXJ0aWVzPy5TdGF0ZU1hY2hpbmVOYW1lO1xuICBjb25zdCBzdGF0ZU1hY2hpbmVBcm4gPSBzdGF0ZU1hY2hpbmVOYW1lSW5DZm5UZW1wbGF0ZVxuICAgID8gYXdhaXQgZXZhbHVhdGVDZm5UZW1wbGF0ZS5ldmFsdWF0ZUNmbkV4cHJlc3Npb24oe1xuICAgICAgJ0ZuOjpTdWInOiAnYXJuOiR7QVdTOjpQYXJ0aXRpb259OnN0YXRlczoke0FXUzo6UmVnaW9ufToke0FXUzo6QWNjb3VudElkfTpzdGF0ZU1hY2hpbmU6JyArIHN0YXRlTWFjaGluZU5hbWVJbkNmblRlbXBsYXRlLFxuICAgIH0pXG4gICAgOiBhd2FpdCBldmFsdWF0ZUNmblRlbXBsYXRlLmZpbmRQaHlzaWNhbE5hbWVGb3IobG9naWNhbElkKTtcblxuICBpZiAoIXN0YXRlTWFjaGluZUFybikge1xuICAgIHJldHVybiBDaGFuZ2VIb3Rzd2FwSW1wYWN0LlJFUVVJUkVTX0ZVTExfREVQTE9ZTUVOVDtcbiAgfVxuXG4gIHJldHVybiBuZXcgU3RhdGVNYWNoaW5lSG90c3dhcE9wZXJhdGlvbih7XG4gICAgZGVmaW5pdGlvbjogc3RhdGVNYWNoaW5lRGVmaW5pdGlvbkNoYW5nZSxcbiAgICBzdGF0ZU1hY2hpbmVBcm46IHN0YXRlTWFjaGluZUFybixcbiAgfSk7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIGlzU3RhdGVNYWNoaW5lRGVmaW5pdGlvbk9ubHlDaGFuZ2UoXG4gIGNoYW5nZTogSG90c3dhcHBhYmxlQ2hhbmdlQ2FuZGlkYXRlLCBldmFsdWF0ZUNmblRlbXBsYXRlOiBFdmFsdWF0ZUNsb3VkRm9ybWF0aW9uVGVtcGxhdGUsXG4pOiBQcm9taXNlPHN0cmluZyB8IENoYW5nZUhvdHN3YXBJbXBhY3Q+IHtcbiAgY29uc3QgbmV3UmVzb3VyY2VUeXBlID0gY2hhbmdlLm5ld1ZhbHVlLlR5cGU7XG4gIGlmIChuZXdSZXNvdXJjZVR5cGUgIT09ICdBV1M6OlN0ZXBGdW5jdGlvbnM6OlN0YXRlTWFjaGluZScpIHtcbiAgICByZXR1cm4gQ2hhbmdlSG90c3dhcEltcGFjdC5SRVFVSVJFU19GVUxMX0RFUExPWU1FTlQ7XG4gIH1cblxuICBjb25zdCBwcm9wZXJ0eVVwZGF0ZXMgPSBjaGFuZ2UucHJvcGVydHlVcGRhdGVzO1xuICBpZiAoT2JqZWN0LmtleXMocHJvcGVydHlVcGRhdGVzKS5sZW5ndGggPT09IDApIHtcbiAgICByZXR1cm4gQ2hhbmdlSG90c3dhcEltcGFjdC5JUlJFTEVWQU5UO1xuICB9XG5cbiAgZm9yIChjb25zdCB1cGRhdGVkUHJvcE5hbWUgaW4gcHJvcGVydHlVcGRhdGVzKSB7XG4gICAgLy8gZW5zdXJlIHRoYXQgb25seSBjaGFuZ2VzIHRvIHRoZSBkZWZpbml0aW9uIHN0cmluZyByZXN1bHQgaW4gYSBob3Rzd2FwXG4gICAgaWYgKHVwZGF0ZWRQcm9wTmFtZSAhPT0gJ0RlZmluaXRpb25TdHJpbmcnKSB7XG4gICAgICByZXR1cm4gQ2hhbmdlSG90c3dhcEltcGFjdC5SRVFVSVJFU19GVUxMX0RFUExPWU1FTlQ7XG4gICAgfVxuICB9XG5cbiAgcmV0dXJuIGV2YWx1YXRlQ2ZuVGVtcGxhdGUuZXZhbHVhdGVDZm5FeHByZXNzaW9uKHByb3BlcnR5VXBkYXRlcy5EZWZpbml0aW9uU3RyaW5nLm5ld1ZhbHVlKTtcbn1cblxuaW50ZXJmYWNlIFN0YXRlTWFjaGluZVJlc291cmNlIHtcbiAgcmVhZG9ubHkgc3RhdGVNYWNoaW5lQXJuOiBzdHJpbmc7XG4gIHJlYWRvbmx5IGRlZmluaXRpb246IHN0cmluZztcbn1cblxuY2xhc3MgU3RhdGVNYWNoaW5lSG90c3dhcE9wZXJhdGlvbiBpbXBsZW1lbnRzIEhvdHN3YXBPcGVyYXRpb24ge1xuICBwdWJsaWMgcmVhZG9ubHkgc2VydmljZSA9ICdzdGVwZnVuY3Rpb25zLXN0YXRlLW1hY2hpbmUnO1xuICBwdWJsaWMgcmVhZG9ubHkgcmVzb3VyY2VOYW1lczogc3RyaW5nW107XG5cbiAgY29uc3RydWN0b3IocHJpdmF0ZSByZWFkb25seSBzdGVwRnVuY3Rpb25SZXNvdXJjZTogU3RhdGVNYWNoaW5lUmVzb3VyY2UpIHtcbiAgICB0aGlzLnJlc291cmNlTmFtZXMgPSBbYFN0YXRlTWFjaGluZSAnJHt0aGlzLnN0ZXBGdW5jdGlvblJlc291cmNlLnN0YXRlTWFjaGluZUFybi5zcGxpdCgnOicpWzZdfSdgXTtcbiAgfVxuXG4gIHB1YmxpYyBhc3luYyBhcHBseShzZGs6IElTREspOiBQcm9taXNlPGFueT4ge1xuICAgIC8vIG5vdCBwYXNzaW5nIHRoZSBvcHRpb25hbCBwcm9wZXJ0aWVzIGxlYXZlcyB0aGVtIHVuY2hhbmdlZFxuICAgIHJldHVybiBzZGsuc3RlcEZ1bmN0aW9ucygpLnVwZGF0ZVN0YXRlTWFjaGluZSh7XG4gICAgICBzdGF0ZU1hY2hpbmVBcm46IHRoaXMuc3RlcEZ1bmN0aW9uUmVzb3VyY2Uuc3RhdGVNYWNoaW5lQXJuLFxuICAgICAgZGVmaW5pdGlvbjogdGhpcy5zdGVwRnVuY3Rpb25SZXNvdXJjZS5kZWZpbml0aW9uLFxuICAgIH0pLnByb21pc2UoKTtcbiAgfVxufVxuIl19
|
package/lib/commands/doctor.js
CHANGED
|
@@ -48,7 +48,7 @@ function displayCdkEnvironmentVariables() {
|
|
|
48
48
|
logging_1.print('ℹ️ CDK environment variables:');
|
|
49
49
|
let healthy = true;
|
|
50
50
|
for (const key of keys.sort()) {
|
|
51
|
-
if (key === cxapi.CONTEXT_ENV || key === cxapi.OUTDIR_ENV) {
|
|
51
|
+
if (key === cxapi.CONTEXT_ENV || key === cxapi.CONTEXT_OVERFLOW_LOCATION_ENV || key === cxapi.OUTDIR_ENV) {
|
|
52
52
|
logging_1.print(` - ${chalk.red(key)} = ${chalk.green(process.env[key])} (⚠️ reserved for use by the CDK toolkit)`);
|
|
53
53
|
healthy = false;
|
|
54
54
|
}
|
|
@@ -67,4 +67,4 @@ function anonymizeAwsVariable(name, value) {
|
|
|
67
67
|
}
|
|
68
68
|
return value;
|
|
69
69
|
}
|
|
70
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
70
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG9jdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiZG9jdG9yLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLG1DQUFtQztBQUNuQyx5Q0FBeUM7QUFDekMsK0JBQStCO0FBQy9CLCtDQUEwQztBQUMxQyw2Q0FBNkM7QUFHdEMsS0FBSyxVQUFVLFdBQVcsQ0FBQyxRQUF3QjtJQUN4RCxJQUFJLFVBQVUsR0FBVyxDQUFDLENBQUM7SUFDM0IsS0FBSyxNQUFNLFlBQVksSUFBSSxhQUFhLEVBQUU7UUFDeEMsSUFBSSxDQUFDLE1BQU0sWUFBWSxFQUFFLEVBQUU7WUFDekIsVUFBVSxHQUFHLENBQUMsQ0FBQyxDQUFDO1NBQ2pCO0tBQ0Y7SUFDRCxNQUFNLE9BQU8sQ0FBQyxxQkFBcUIsRUFBRSxDQUFDO0lBQ3RDLE9BQU8sVUFBVSxDQUFDO0FBQ3BCLENBQUM7QUFURCxrQ0FTQztBQUVELE1BQU0sYUFBYSxHQUE0QztJQUM3RCx5QkFBeUI7SUFDekIsOEJBQThCO0lBQzlCLDhCQUE4QjtDQUMvQixDQUFDO0FBRUYsd0JBQXdCO0FBRXhCLFNBQVMseUJBQXlCO0lBQ2hDLGVBQUssQ0FBQyxtQkFBbUIsS0FBSyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsZUFBZSxDQUFDLEVBQUUsQ0FBQyxDQUFDO0lBQ2pFLE9BQU8sSUFBSSxDQUFDO0FBQ2QsQ0FBQztBQUVELFNBQVMsOEJBQThCO0lBQ3JDLE1BQU0sSUFBSSxHQUFHLE1BQU0sQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxVQUFVLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQztJQUN4RSxJQUFJLElBQUksQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFO1FBQ3JCLGVBQUssQ0FBQyxpQ0FBaUMsQ0FBQyxDQUFDO1FBQ3pDLE9BQU8sSUFBSSxDQUFDO0tBQ2I7SUFDRCxlQUFLLENBQUMsK0JBQStCLENBQUMsQ0FBQztJQUN2QyxLQUFLLE1BQU0sR0FBRyxJQUFJLElBQUksRUFBRTtRQUN0QixlQUFLLENBQUMsT0FBTyxLQUFLLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEtBQUssQ0FBQyxLQUFLLENBQUMsb0JBQW9CLENBQUMsR0FBRyxFQUFFLE9BQU8sQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFFLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQztLQUNoRztJQUNELE9BQU8sSUFBSSxDQUFDO0FBQ2QsQ0FBQztBQUVELFNBQVMsOEJBQThCO0lBQ3JDLE1BQU0sSUFBSSxHQUFHLE1BQU0sQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxVQUFVLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQztJQUN4RSxJQUFJLElBQUksQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFO1FBQ3JCLGVBQUssQ0FBQyxpQ0FBaUMsQ0FBQyxDQUFDO1FBQ3pDLE9BQU8sSUFBSSxDQUFDO0tBQ2I7SUFDRCxlQUFLLENBQUMsK0JBQStCLENBQUMsQ0FBQztJQUN2QyxJQUFJLE9BQU8sR0FBRyxJQUFJLENBQUM7SUFDbkIsS0FBSyxNQUFNLEdBQUcsSUFBSSxJQUFJLENBQUMsSUFBSSxFQUFFLEVBQUU7UUFDN0IsSUFBSSxHQUFHLEtBQUssS0FBSyxDQUFDLFdBQVcsSUFBSSxHQUFHLEtBQUssS0FBSyxDQUFDLDZCQUE2QixJQUFJLEdBQUcsS0FBSyxLQUFLLENBQUMsVUFBVSxFQUFFO1lBQ3hHLGVBQUssQ0FBQyxPQUFPLEtBQUssQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLE1BQU0sS0FBSyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBRSxDQUFDLDJDQUEyQyxDQUFDLENBQUM7WUFDNUcsT0FBTyxHQUFHLEtBQUssQ0FBQztTQUNqQjthQUFNO1lBQ0wsZUFBSyxDQUFDLE9BQU8sS0FBSyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsTUFBTSxLQUFLLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFFLENBQUMsRUFBRSxDQUFDLENBQUM7U0FDckU7S0FDRjtJQUNELE9BQU8sT0FBTyxDQUFDO0FBQ2pCLENBQUM7QUFFRCxTQUFTLG9CQUFvQixDQUFDLElBQVksRUFBRSxLQUFhO0lBQ3ZELElBQUksSUFBSSxLQUFLLG1CQUFtQixFQUFFO1FBQUUsT0FBTyxLQUFLLENBQUMsS0FBSyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsR0FBRyxZQUFZLENBQUM7S0FBRSxDQUFDLCtDQUErQztJQUM5SCxJQUFJLElBQUksS0FBSyx1QkFBdUIsSUFBSSxJQUFJLEtBQUssbUJBQW1CLElBQUksSUFBSSxLQUFLLG9CQUFvQixFQUFFO1FBQUUsT0FBTyxZQUFZLENBQUM7S0FBRTtJQUMvSCxPQUFPLEtBQUssQ0FBQztBQUNmLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBwcm9jZXNzIGZyb20gJ3Byb2Nlc3MnO1xuaW1wb3J0ICogYXMgY3hhcGkgZnJvbSAnQGF3cy1jZGsvY3gtYXBpJztcbmltcG9ydCAqIGFzIGNoYWxrIGZyb20gJ2NoYWxrJztcbmltcG9ydCB7IHByaW50IH0gZnJvbSAnLi4vLi4vbGliL2xvZ2dpbmcnO1xuaW1wb3J0ICogYXMgdmVyc2lvbiBmcm9tICcuLi8uLi9saWIvdmVyc2lvbic7XG5pbXBvcnQgeyBDb21tYW5kT3B0aW9ucyB9IGZyb20gJy4uL2NvbW1hbmQtYXBpJztcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHJlYWxIYW5kbGVyKF9vcHRpb25zOiBDb21tYW5kT3B0aW9ucyk6IFByb21pc2U8bnVtYmVyPiB7XG4gIGxldCBleGl0U3RhdHVzOiBudW1iZXIgPSAwO1xuICBmb3IgKGNvbnN0IHZlcmlmaWNhdGlvbiBvZiB2ZXJpZmljYXRpb25zKSB7XG4gICAgaWYgKCFhd2FpdCB2ZXJpZmljYXRpb24oKSkge1xuICAgICAgZXhpdFN0YXR1cyA9IC0xO1xuICAgIH1cbiAgfVxuICBhd2FpdCB2ZXJzaW9uLmRpc3BsYXlWZXJzaW9uTWVzc2FnZSgpO1xuICByZXR1cm4gZXhpdFN0YXR1cztcbn1cblxuY29uc3QgdmVyaWZpY2F0aW9uczogQXJyYXk8KCkgPT4gYm9vbGVhbiB8IFByb21pc2U8Ym9vbGVhbj4+ID0gW1xuICBkaXNwbGF5VmVyc2lvbkluZm9ybWF0aW9uLFxuICBkaXNwbGF5QXdzRW52aXJvbm1lbnRWYXJpYWJsZXMsXG4gIGRpc3BsYXlDZGtFbnZpcm9ubWVudFZhcmlhYmxlcyxcbl07XG5cbi8vICMjIyBWZXJpZmljYXRpb25zICMjI1xuXG5mdW5jdGlvbiBkaXNwbGF5VmVyc2lvbkluZm9ybWF0aW9uKCkge1xuICBwcmludChg4oS577iPIENESyBWZXJzaW9uOiAke2NoYWxrLmdyZWVuKHZlcnNpb24uRElTUExBWV9WRVJTSU9OKX1gKTtcbiAgcmV0dXJuIHRydWU7XG59XG5cbmZ1bmN0aW9uIGRpc3BsYXlBd3NFbnZpcm9ubWVudFZhcmlhYmxlcygpIHtcbiAgY29uc3Qga2V5cyA9IE9iamVjdC5rZXlzKHByb2Nlc3MuZW52KS5maWx0ZXIocyA9PiBzLnN0YXJ0c1dpdGgoJ0FXU18nKSk7XG4gIGlmIChrZXlzLmxlbmd0aCA9PT0gMCkge1xuICAgIHByaW50KCfihLnvuI8gTm8gQVdTIGVudmlyb25tZW50IHZhcmlhYmxlcycpO1xuICAgIHJldHVybiB0cnVlO1xuICB9XG4gIHByaW50KCfihLnvuI8gQVdTIGVudmlyb25tZW50IHZhcmlhYmxlczonKTtcbiAgZm9yIChjb25zdCBrZXkgb2Yga2V5cykge1xuICAgIHByaW50KGAgIC0gJHtjaGFsay5ibHVlKGtleSl9ID0gJHtjaGFsay5ncmVlbihhbm9ueW1pemVBd3NWYXJpYWJsZShrZXksIHByb2Nlc3MuZW52W2tleV0hKSl9YCk7XG4gIH1cbiAgcmV0dXJuIHRydWU7XG59XG5cbmZ1bmN0aW9uIGRpc3BsYXlDZGtFbnZpcm9ubWVudFZhcmlhYmxlcygpIHtcbiAgY29uc3Qga2V5cyA9IE9iamVjdC5rZXlzKHByb2Nlc3MuZW52KS5maWx0ZXIocyA9PiBzLnN0YXJ0c1dpdGgoJ0NES18nKSk7XG4gIGlmIChrZXlzLmxlbmd0aCA9PT0gMCkge1xuICAgIHByaW50KCfihLnvuI8gTm8gQ0RLIGVudmlyb25tZW50IHZhcmlhYmxlcycpO1xuICAgIHJldHVybiB0cnVlO1xuICB9XG4gIHByaW50KCfihLnvuI8gQ0RLIGVudmlyb25tZW50IHZhcmlhYmxlczonKTtcbiAgbGV0IGhlYWx0aHkgPSB0cnVlO1xuICBmb3IgKGNvbnN0IGtleSBvZiBrZXlzLnNvcnQoKSkge1xuICAgIGlmIChrZXkgPT09IGN4YXBpLkNPTlRFWFRfRU5WIHx8IGtleSA9PT0gY3hhcGkuQ09OVEVYVF9PVkVSRkxPV19MT0NBVElPTl9FTlYgfHwga2V5ID09PSBjeGFwaS5PVVRESVJfRU5WKSB7XG4gICAgICBwcmludChgICAtICR7Y2hhbGsucmVkKGtleSl9ID0gJHtjaGFsay5ncmVlbihwcm9jZXNzLmVudltrZXldISl9ICjimqDvuI8gcmVzZXJ2ZWQgZm9yIHVzZSBieSB0aGUgQ0RLIHRvb2xraXQpYCk7XG4gICAgICBoZWFsdGh5ID0gZmFsc2U7XG4gICAgfSBlbHNlIHtcbiAgICAgIHByaW50KGAgIC0gJHtjaGFsay5ibHVlKGtleSl9ID0gJHtjaGFsay5ncmVlbihwcm9jZXNzLmVudltrZXldISl9YCk7XG4gICAgfVxuICB9XG4gIHJldHVybiBoZWFsdGh5O1xufVxuXG5mdW5jdGlvbiBhbm9ueW1pemVBd3NWYXJpYWJsZShuYW1lOiBzdHJpbmcsIHZhbHVlOiBzdHJpbmcpIHtcbiAgaWYgKG5hbWUgPT09ICdBV1NfQUNDRVNTX0tFWV9JRCcpIHsgcmV0dXJuIHZhbHVlLnNsaWNlKDAsIDQpICsgJzxyZWRhY3RlZD4nOyB9IC8vIFNob3cgQVNJQS9BS0lBIGtleSB0eXBlLCBidXQgaGlkZSBpZGVudGlmaWVyXG4gIGlmIChuYW1lID09PSAnQVdTX1NFQ1JFVF9BQ0NFU1NfS0VZJyB8fCBuYW1lID09PSAnQVdTX1NFU1NJT05fVE9LRU4nIHx8IG5hbWUgPT09ICdBV1NfU0VDVVJJVFlfVE9LRU4nKSB7IHJldHVybiAnPHJlZGFjdGVkPic7IH1cbiAgcmV0dXJuIHZhbHVlO1xufVxuIl19
|