aws-cdk 2.1004.0 → 2.1006.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/THIRD_PARTY_LICENSES +31 -31
- package/build-info.json +2 -2
- package/db.json.gz +0 -0
- package/lib/api/aws-auth/account-cache.d.ts +1 -1
- package/lib/api/aws-auth/account-cache.js +1 -1
- package/lib/api/aws-auth/awscli-compatible.d.ts +1 -1
- package/lib/api/aws-auth/awscli-compatible.js +9 -10
- package/lib/api/aws-auth/credential-plugins.d.ts +1 -1
- package/lib/api/aws-auth/credential-plugins.js +6 -7
- package/lib/api/aws-auth/provider-caching.d.ts +1 -1
- package/lib/api/aws-auth/provider-caching.js +1 -1
- package/lib/api/aws-auth/sdk-logger.d.ts +2 -2
- package/lib/api/aws-auth/sdk-logger.js +3 -4
- package/lib/api/aws-auth/sdk-provider.d.ts +4 -4
- package/lib/api/aws-auth/sdk-provider.js +11 -13
- package/lib/api/aws-auth/sdk.d.ts +12 -12
- package/lib/api/aws-auth/sdk.js +8 -9
- package/lib/api/aws-auth/tracing.js +3 -4
- package/lib/api/aws-auth/user-agent.js +4 -5
- package/lib/api/bootstrap/bootstrap-environment.d.ts +2 -2
- package/lib/api/bootstrap/bootstrap-environment.js +42 -46
- package/lib/api/bootstrap/bootstrap-props.d.ts +3 -3
- package/lib/api/bootstrap/bootstrap-props.js +1 -1
- package/lib/api/bootstrap/deploy-bootstrap.d.ts +4 -4
- package/lib/api/bootstrap/deploy-bootstrap.js +11 -14
- package/lib/api/bootstrap/legacy-template.d.ts +1 -1
- package/lib/api/bootstrap/legacy-template.js +1 -1
- package/lib/api/{evaluate-cloudformation-template.d.ts → cloudformation/evaluate-cloudformation-template.d.ts} +10 -9
- package/lib/api/cloudformation/evaluate-cloudformation-template.js +440 -0
- package/lib/api/cloudformation/index.d.ts +4 -0
- package/lib/api/cloudformation/index.js +21 -0
- package/lib/api/{deployments → cloudformation}/nested-stack-helpers.d.ts +1 -1
- package/lib/api/cloudformation/nested-stack-helpers.js +86 -0
- package/lib/api/cloudformation/stack-helpers.d.ts +96 -0
- package/lib/api/cloudformation/stack-helpers.js +158 -0
- package/lib/api/{util → cloudformation}/template-body-parameter.d.ts +4 -3
- package/lib/api/cloudformation/template-body-parameter.js +104 -0
- package/lib/api/context.js +3 -3
- package/lib/api/cxapp/cloud-assembly.js +13 -15
- package/lib/api/cxapp/cloud-executable.d.ts +3 -3
- package/lib/api/cxapp/cloud-executable.js +4 -5
- package/lib/api/cxapp/environments.d.ts +3 -3
- package/lib/api/cxapp/environments.js +4 -4
- package/lib/api/cxapp/exec.d.ts +4 -4
- package/lib/api/cxapp/exec.js +20 -23
- package/lib/api/deployments/asset-publishing.d.ts +2 -4
- package/lib/api/deployments/asset-publishing.js +24 -31
- package/lib/api/deployments/assets.d.ts +3 -3
- package/lib/api/deployments/assets.js +12 -13
- package/lib/api/deployments/{cloudformation.d.ts → cfn-api.d.ts} +5 -102
- package/lib/api/deployments/cfn-api.js +438 -0
- package/lib/api/deployments/checks.d.ts +2 -2
- package/lib/api/deployments/checks.js +12 -13
- package/lib/api/deployments/deploy-stack.d.ts +6 -7
- package/lib/api/deployments/deploy-stack.js +69 -80
- package/lib/api/deployments/deployment-result.js +3 -3
- package/lib/api/deployments/deployments.d.ts +7 -7
- package/lib/api/deployments/deployments.js +35 -42
- package/lib/api/deployments/hotswap-deployments.d.ts +7 -5
- package/lib/api/deployments/hotswap-deployments.js +160 -82
- package/lib/api/deployments/index.d.ts +0 -2
- package/lib/api/deployments/index.js +1 -3
- package/lib/api/environment/environment-access.d.ts +6 -6
- package/lib/api/environment/environment-access.js +18 -20
- package/lib/api/environment/environment-resources.d.ts +1 -1
- package/lib/api/environment/environment-resources.js +17 -19
- package/lib/api/environment/index.d.ts +1 -0
- package/lib/api/environment/index.js +2 -1
- package/lib/api/environment/placeholders.js +23 -0
- package/lib/api/garbage-collection/garbage-collector.d.ts +4 -4
- package/lib/api/garbage-collection/garbage-collector.js +57 -67
- package/lib/api/garbage-collection/progress-printer.d.ts +2 -2
- package/lib/api/garbage-collection/progress-printer.js +7 -7
- package/lib/api/garbage-collection/stack-refresh.d.ts +2 -2
- package/lib/api/garbage-collection/stack-refresh.js +12 -15
- package/lib/api/hotswap/appsync-mapping-templates.d.ts +4 -3
- package/lib/api/hotswap/appsync-mapping-templates.js +32 -27
- package/lib/api/hotswap/code-build-projects.d.ts +4 -3
- package/lib/api/hotswap/code-build-projects.js +19 -12
- package/lib/api/hotswap/common.d.ts +27 -64
- package/lib/api/hotswap/common.js +40 -82
- package/lib/api/hotswap/ecs-services.d.ts +4 -3
- package/lib/api/hotswap/ecs-services.js +44 -25
- package/lib/api/hotswap/lambda-functions.d.ts +4 -3
- package/lib/api/hotswap/lambda-functions.js +61 -75
- package/lib/api/hotswap/s3-bucket-deployments.d.ts +5 -9
- package/lib/api/hotswap/s3-bucket-deployments.js +26 -21
- package/lib/api/hotswap/stepfunctions-state-machines.d.ts +4 -3
- package/lib/api/hotswap/stepfunctions-state-machines.js +15 -9
- package/lib/api/logs/find-cloudwatch-logs.d.ts +1 -1
- package/lib/api/logs/find-cloudwatch-logs.js +7 -9
- package/lib/api/logs/logs-monitor.d.ts +2 -2
- package/lib/api/logs/logs-monitor.js +5 -8
- package/lib/api/plugin/plugin.js +6 -10
- package/lib/api/resource-import/importer.d.ts +10 -5
- package/lib/api/resource-import/importer.js +23 -29
- package/lib/api/resource-import/migrator.d.ts +4 -4
- package/lib/api/resource-import/migrator.js +6 -6
- package/lib/api/settings.d.ts +0 -3
- package/lib/api/settings.js +4 -40
- package/lib/api/stack-events/stack-activity-monitor.d.ts +1 -2
- package/lib/api/stack-events/stack-activity-monitor.js +14 -37
- package/lib/api/stack-events/stack-event-poller.js +9 -10
- package/lib/api/stack-events/stack-progress-monitor.d.ts +1 -1
- package/lib/api/stack-events/stack-progress-monitor.js +1 -1
- package/lib/api/toolkit-info.d.ts +3 -3
- package/lib/api/toolkit-info.js +20 -24
- package/lib/{tree.d.ts → api/tree.d.ts} +3 -3
- package/lib/api/tree.js +37 -0
- package/lib/api/util/rwlock.js +4 -4
- package/lib/api/work-graph/work-graph-builder.d.ts +2 -2
- package/lib/api/work-graph/work-graph-builder.js +4 -4
- package/lib/api/work-graph/work-graph-types.d.ts +2 -2
- package/lib/api/work-graph/work-graph-types.js +1 -1
- package/lib/api/work-graph/work-graph.d.ts +2 -2
- package/lib/api/work-graph/work-graph.js +13 -15
- package/lib/cli/activity-printer/base.d.ts +3 -3
- package/lib/cli/activity-printer/base.js +6 -8
- package/lib/cli/activity-printer/current.d.ts +2 -1
- package/lib/cli/activity-printer/current.js +7 -11
- package/lib/cli/activity-printer/history.d.ts +2 -1
- package/lib/cli/activity-printer/history.js +2 -3
- package/lib/cli/cdk-toolkit.d.ts +9 -15
- package/lib/cli/cdk-toolkit.js +90 -73
- package/lib/cli/ci-systems.d.ts +29 -0
- package/lib/cli/ci-systems.js +61 -0
- package/lib/cli/cli-config.js +3 -3
- package/lib/cli/cli.d.ts +1 -1
- package/lib/cli/cli.js +69 -52
- package/lib/cli/convert-to-user-input.js +110 -111
- package/lib/{toolkit → cli/io-host}/cli-io-host.d.ts +22 -3
- package/lib/cli/io-host/cli-io-host.js +356 -0
- package/lib/cli/io-host/index.d.ts +1 -0
- package/lib/{toolkit/error.js → cli/io-host/index.js} +2 -2
- package/lib/cli/messages.d.ts +2 -2
- package/lib/cli/messages.js +2 -3
- package/lib/cli/parse-command-line-arguments.js +1 -1
- package/lib/cli/pretty-print-error.d.ts +1 -0
- package/lib/cli/pretty-print-error.js +35 -0
- package/lib/cli/root-dir.js +4 -4
- package/lib/cli/user-configuration.js +57 -14
- package/lib/cli/user-input.js +1 -1
- package/lib/cli/util/npm.js +3 -3
- package/lib/cli/util/yargs-helpers.d.ts +1 -1
- package/lib/cli/util/yargs-helpers.js +3 -3
- package/lib/cli/version.js +4 -4
- package/lib/commands/context.d.ts +1 -1
- package/lib/commands/context.js +7 -8
- package/lib/commands/diff.d.ts +50 -0
- package/lib/commands/diff.js +215 -0
- package/lib/commands/init/index.d.ts +1 -0
- package/lib/commands/init/index.js +18 -0
- package/lib/commands/init/init-hooks.js +63 -0
- package/lib/commands/init/init.js +435 -0
- package/lib/{os.js → commands/init/os.js} +4 -4
- package/lib/{list-stacks.d.ts → commands/list-stacks.d.ts} +1 -1
- package/lib/{list-stacks.js → commands/list-stacks.js} +2 -2
- package/lib/commands/migrate.d.ts +1 -1
- package/lib/commands/migrate.js +29 -32
- package/lib/context-providers/ami.d.ts +4 -2
- package/lib/context-providers/ami.js +8 -8
- package/lib/context-providers/availability-zones.d.ts +4 -2
- package/lib/context-providers/availability-zones.js +4 -4
- package/lib/context-providers/cc-api-provider.d.ts +9 -13
- package/lib/context-providers/cc-api-provider.js +88 -66
- package/lib/context-providers/endpoint-service-availability-zones.d.ts +4 -2
- package/lib/context-providers/endpoint-service-availability-zones.js +6 -6
- package/lib/context-providers/hosted-zones.d.ts +5 -3
- package/lib/context-providers/hosted-zones.js +11 -11
- package/lib/context-providers/index.d.ts +22 -8
- package/lib/context-providers/index.js +35 -17
- package/lib/context-providers/keys.d.ts +4 -2
- package/lib/context-providers/keys.js +8 -8
- package/lib/context-providers/load-balancers.d.ts +3 -3
- package/lib/context-providers/load-balancers.js +15 -18
- package/lib/context-providers/security-groups.js +10 -12
- package/lib/context-providers/ssm-parameters.d.ts +4 -2
- package/lib/context-providers/ssm-parameters.js +7 -7
- package/lib/context-providers/vpcs.d.ts +4 -2
- package/lib/context-providers/vpcs.js +14 -15
- package/lib/index.js +115212 -114237
- package/lib/init-templates/.init-version.json +1 -1
- package/lib/init-templates/.recommended-feature-flags.json +2 -1
- package/lib/legacy-exports-source.d.ts +4 -5
- package/lib/legacy-exports-source.js +6 -7
- package/lib/legacy-logging-source.js +1 -1
- package/lib/logging.d.ts +3 -3
- package/lib/logging.js +38 -36
- package/lib/notices.d.ts +22 -18
- package/lib/notices.js +67 -73
- package/package.json +18 -20
- package/lib/api/deployments/cloudformation.js +0 -597
- package/lib/api/deployments/nested-stack-helpers.js +0 -88
- package/lib/api/evaluate-cloudformation-template.js +0 -443
- package/lib/api/util/placeholders.js +0 -24
- package/lib/api/util/template-body-parameter.js +0 -103
- package/lib/diff.d.ts +0 -28
- package/lib/diff.js +0 -165
- package/lib/init-hooks.js +0 -63
- package/lib/init.js +0 -437
- package/lib/toolkit/cli-io-host.js +0 -324
- package/lib/toolkit/error.d.ts +0 -1
- package/lib/tree.js +0 -40
- /package/lib/api/{util → environment}/placeholders.d.ts +0 -0
- /package/lib/{init-hooks.d.ts → commands/init/init-hooks.d.ts} +0 -0
- /package/lib/{init.d.ts → commands/init/init.d.ts} +0 -0
- /package/lib/{os.d.ts → commands/init/os.d.ts} +0 -0
package/lib/commands/context.js
CHANGED
|
@@ -3,20 +3,19 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
3
3
|
exports.contextHandler = contextHandler;
|
|
4
4
|
const chalk = require("chalk");
|
|
5
5
|
const minimatch_1 = require("minimatch");
|
|
6
|
+
const api_1 = require("../../../@aws-cdk/tmp-toolkit-helpers/src/api");
|
|
6
7
|
const tables_1 = require("../cli/tables");
|
|
7
8
|
const user_configuration_1 = require("../cli/user-configuration");
|
|
8
9
|
const version = require("../cli/version");
|
|
9
10
|
const logging_1 = require("../logging");
|
|
10
|
-
const error_1 = require("../toolkit/error");
|
|
11
11
|
async function contextHandler(options) {
|
|
12
|
-
var _a;
|
|
13
12
|
if (options.clear) {
|
|
14
13
|
options.context.clear();
|
|
15
14
|
await options.context.save(user_configuration_1.PROJECT_CONTEXT);
|
|
16
15
|
(0, logging_1.info)('All context values cleared.');
|
|
17
16
|
}
|
|
18
17
|
else if (options.reset) {
|
|
19
|
-
invalidateContext(options.context, options.reset,
|
|
18
|
+
invalidateContext(options.context, options.reset, options.force ?? false);
|
|
20
19
|
await options.context.save(user_configuration_1.PROJECT_CONTEXT);
|
|
21
20
|
}
|
|
22
21
|
else {
|
|
@@ -72,7 +71,7 @@ function invalidateContext(context, key, force) {
|
|
|
72
71
|
// Value must be in readonly bag
|
|
73
72
|
(0, logging_1.error)('Only context values specified in %s can be reset through the CLI', chalk.blue(user_configuration_1.PROJECT_CONTEXT));
|
|
74
73
|
if (!force) {
|
|
75
|
-
throw new
|
|
74
|
+
throw new api_1.ToolkitError(`Cannot reset readonly context value with key: ${key}`);
|
|
76
75
|
}
|
|
77
76
|
}
|
|
78
77
|
// check if value is expression matching keys
|
|
@@ -88,12 +87,12 @@ function invalidateContext(context, key, force) {
|
|
|
88
87
|
printReadonly(readonly);
|
|
89
88
|
// throw when none of the matches were reset
|
|
90
89
|
if (!force && unset.length === 0) {
|
|
91
|
-
throw new
|
|
90
|
+
throw new api_1.ToolkitError('None of the matched context values could be reset');
|
|
92
91
|
}
|
|
93
92
|
return;
|
|
94
93
|
}
|
|
95
94
|
if (!force) {
|
|
96
|
-
throw new
|
|
95
|
+
throw new api_1.ToolkitError(`No context value matching key: ${key}`);
|
|
97
96
|
}
|
|
98
97
|
}
|
|
99
98
|
function printUnset(unset) {
|
|
@@ -134,7 +133,7 @@ function keyByNumber(context, n) {
|
|
|
134
133
|
return key;
|
|
135
134
|
}
|
|
136
135
|
}
|
|
137
|
-
throw new
|
|
136
|
+
throw new api_1.ToolkitError(`No context key with number: ${n}`);
|
|
138
137
|
}
|
|
139
138
|
/**
|
|
140
139
|
* Return enumerated keys in a definitive order
|
|
@@ -153,4 +152,4 @@ function enumerate1(xs) {
|
|
|
153
152
|
}
|
|
154
153
|
return ret;
|
|
155
154
|
}
|
|
156
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"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","sourcesContent":["import * as chalk from 'chalk';\nimport { minimatch } from 'minimatch';\nimport { Context } from '../api/context';\nimport { renderTable } from '../cli/tables';\nimport { PROJECT_CONFIG, PROJECT_CONTEXT, USER_DEFAULTS } from '../cli/user-configuration';\nimport * as version from '../cli/version';\nimport { error, warning, info, result } from '../logging';\nimport { ToolkitError } from '../toolkit/error';\n\n/**\n * Options for the context command\n */\nexport interface ContextOptions {\n  /**\n   * The context object sourced from all context locations\n   */\n  context: Context;\n\n  /**\n   * The context key (or its index) to reset\n   *\n   * @default undefined\n   */\n  reset?: string;\n\n  /**\n   * Ignore missing key error\n   *\n   * @default false\n   */\n  force?: boolean;\n\n  /**\n   * Clear all context\n   *\n   * @default false\n   */\n  clear?: boolean;\n\n  /**\n   * Use JSON output instead of YAML when templates are printed to STDOUT\n   *\n   * @default false\n   */\n  json?: boolean;\n}\n\nexport async function contextHandler(options: ContextOptions): Promise<number> {\n  if (options.clear) {\n    options.context.clear();\n    await options.context.save(PROJECT_CONTEXT);\n    info('All context values cleared.');\n  } else if (options.reset) {\n    invalidateContext(options.context, options.reset, options.force ?? false);\n    await options.context.save(PROJECT_CONTEXT);\n  } else {\n    // List -- support '--json' flag\n    if (options.json) {\n      /* istanbul ignore next */\n      const contextValues = options.context.all;\n      result(JSON.stringify(contextValues, undefined, 2));\n    } else {\n      listContext(options.context);\n    }\n  }\n  await version.displayVersionMessage();\n\n  return 0;\n}\n\nfunction listContext(context: Context) {\n  const keys = contextKeys(context);\n\n  if (keys.length === 0) {\n    info('This CDK application does not have any saved context values yet.');\n    info('');\n    info('Context will automatically be saved when you synthesize CDK apps');\n    info('that use environment context information like AZ information, VPCs,');\n    info('SSM parameters, and so on.');\n\n    return;\n  }\n\n  // Print config by default\n  const data_out: any[] = [[chalk.green('#'), chalk.green('Key'), chalk.green('Value')]];\n  for (const [i, key] of keys) {\n    const jsonWithoutNewlines = JSON.stringify(context.all[key], undefined, 2).replace(/\\s+/g, ' ');\n    data_out.push([i, key, jsonWithoutNewlines]);\n  }\n  info('Context found in %s:', chalk.blue(PROJECT_CONFIG));\n  info('');\n  info(renderTable(data_out, process.stdout.columns));\n\n  // eslint-disable-next-line max-len\n  info(`Run ${chalk.blue('cdk context --reset KEY_OR_NUMBER')} to remove a context key. It will be refreshed on the next CDK synthesis run.`);\n}\n\nfunction invalidateContext(context: Context, key: string, force: boolean) {\n  const i = parseInt(key, 10);\n  if (`${i}` === key) {\n    // was a number and we fully parsed it.\n    key = keyByNumber(context, i);\n  }\n  // Unset!\n  if (context.has(key)) {\n    context.unset(key);\n    // check if the value was actually unset.\n    if (!context.has(key)) {\n      info('Context value %s reset. It will be refreshed on next synthesis', chalk.blue(key));\n      return;\n    }\n\n    // Value must be in readonly bag\n    error('Only context values specified in %s can be reset through the CLI', chalk.blue(PROJECT_CONTEXT));\n    if (!force) {\n      throw new ToolkitError(`Cannot reset readonly context value with key: ${key}`);\n    }\n  }\n\n  // check if value is expression matching keys\n  const matches = keysByExpression(context, key);\n\n  if (matches.length > 0) {\n    matches.forEach((match) => {\n      context.unset(match);\n    });\n\n    const { unset, readonly } = getUnsetAndReadonly(context, matches);\n\n    // output the reset values\n    printUnset(unset);\n\n    // warn about values not reset\n    printReadonly(readonly);\n\n    // throw when none of the matches were reset\n    if (!force && unset.length === 0) {\n      throw new ToolkitError('None of the matched context values could be reset');\n    }\n    return;\n  }\n  if (!force) {\n    throw new ToolkitError(`No context value matching key: ${key}`);\n  }\n}\n\nfunction printUnset(unset: string[]) {\n  if (unset.length === 0) return;\n  info('The following matched context values reset. They will be refreshed on next synthesis');\n  unset.forEach((match) => {\n    info('  %s', match);\n  });\n}\n\nfunction printReadonly(readonly: string[]) {\n  if (readonly.length === 0) return;\n  warning('The following matched context values could not be reset through the CLI');\n  readonly.forEach((match) => {\n    info('  %s', match);\n  });\n  info('');\n  info('This usually means they are configured in %s or %s', chalk.blue(PROJECT_CONFIG), chalk.blue(USER_DEFAULTS));\n}\n\nfunction keysByExpression(context: Context, expression: string) {\n  return context.keys.filter(minimatch.filter(expression));\n}\n\nfunction getUnsetAndReadonly(context: Context, matches: string[]) {\n  return matches.reduce<{ unset: string[]; readonly: string[] }>((acc, match) => {\n    if (context.has(match)) {\n      acc.readonly.push(match);\n    } else {\n      acc.unset.push(match);\n    }\n    return acc;\n  }, { unset: [], readonly: [] });\n}\n\nfunction keyByNumber(context: Context, n: number) {\n  for (const [i, key] of contextKeys(context)) {\n    if (n === i) {\n      return key;\n    }\n  }\n  throw new ToolkitError(`No context key with number: ${n}`);\n}\n\n/**\n * Return enumerated keys in a definitive order\n */\nfunction contextKeys(context: Context): [number, string][] {\n  const keys = context.keys;\n  keys.sort();\n  return enumerate1(keys);\n}\n\nfunction enumerate1<T>(xs: T[]): Array<[number, T]> {\n  const ret = new Array<[number, T]>();\n  let i = 1;\n  for (const x of xs) {\n    ret.push([i, x]);\n    i += 1;\n  }\n  return ret;\n}\n"]}
|
|
155
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"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","sourcesContent":["import * as chalk from 'chalk';\nimport { minimatch } from 'minimatch';\nimport { ToolkitError } from '../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport type { Context } from '../api/context';\nimport { renderTable } from '../cli/tables';\nimport { PROJECT_CONFIG, PROJECT_CONTEXT, USER_DEFAULTS } from '../cli/user-configuration';\nimport * as version from '../cli/version';\nimport { error, warning, info, result } from '../logging';\n\n/**\n * Options for the context command\n */\nexport interface ContextOptions {\n  /**\n   * The context object sourced from all context locations\n   */\n  context: Context;\n\n  /**\n   * The context key (or its index) to reset\n   *\n   * @default undefined\n   */\n  reset?: string;\n\n  /**\n   * Ignore missing key error\n   *\n   * @default false\n   */\n  force?: boolean;\n\n  /**\n   * Clear all context\n   *\n   * @default false\n   */\n  clear?: boolean;\n\n  /**\n   * Use JSON output instead of YAML when templates are printed to STDOUT\n   *\n   * @default false\n   */\n  json?: boolean;\n}\n\nexport async function contextHandler(options: ContextOptions): Promise<number> {\n  if (options.clear) {\n    options.context.clear();\n    await options.context.save(PROJECT_CONTEXT);\n    info('All context values cleared.');\n  } else if (options.reset) {\n    invalidateContext(options.context, options.reset, options.force ?? false);\n    await options.context.save(PROJECT_CONTEXT);\n  } else {\n    // List -- support '--json' flag\n    if (options.json) {\n      /* istanbul ignore next */\n      const contextValues = options.context.all;\n      result(JSON.stringify(contextValues, undefined, 2));\n    } else {\n      listContext(options.context);\n    }\n  }\n  await version.displayVersionMessage();\n\n  return 0;\n}\n\nfunction listContext(context: Context) {\n  const keys = contextKeys(context);\n\n  if (keys.length === 0) {\n    info('This CDK application does not have any saved context values yet.');\n    info('');\n    info('Context will automatically be saved when you synthesize CDK apps');\n    info('that use environment context information like AZ information, VPCs,');\n    info('SSM parameters, and so on.');\n\n    return;\n  }\n\n  // Print config by default\n  const data_out: any[] = [[chalk.green('#'), chalk.green('Key'), chalk.green('Value')]];\n  for (const [i, key] of keys) {\n    const jsonWithoutNewlines = JSON.stringify(context.all[key], undefined, 2).replace(/\\s+/g, ' ');\n    data_out.push([i, key, jsonWithoutNewlines]);\n  }\n  info('Context found in %s:', chalk.blue(PROJECT_CONFIG));\n  info('');\n  info(renderTable(data_out, process.stdout.columns));\n\n  // eslint-disable-next-line max-len\n  info(`Run ${chalk.blue('cdk context --reset KEY_OR_NUMBER')} to remove a context key. It will be refreshed on the next CDK synthesis run.`);\n}\n\nfunction invalidateContext(context: Context, key: string, force: boolean) {\n  const i = parseInt(key, 10);\n  if (`${i}` === key) {\n    // was a number and we fully parsed it.\n    key = keyByNumber(context, i);\n  }\n  // Unset!\n  if (context.has(key)) {\n    context.unset(key);\n    // check if the value was actually unset.\n    if (!context.has(key)) {\n      info('Context value %s reset. It will be refreshed on next synthesis', chalk.blue(key));\n      return;\n    }\n\n    // Value must be in readonly bag\n    error('Only context values specified in %s can be reset through the CLI', chalk.blue(PROJECT_CONTEXT));\n    if (!force) {\n      throw new ToolkitError(`Cannot reset readonly context value with key: ${key}`);\n    }\n  }\n\n  // check if value is expression matching keys\n  const matches = keysByExpression(context, key);\n\n  if (matches.length > 0) {\n    matches.forEach((match) => {\n      context.unset(match);\n    });\n\n    const { unset, readonly } = getUnsetAndReadonly(context, matches);\n\n    // output the reset values\n    printUnset(unset);\n\n    // warn about values not reset\n    printReadonly(readonly);\n\n    // throw when none of the matches were reset\n    if (!force && unset.length === 0) {\n      throw new ToolkitError('None of the matched context values could be reset');\n    }\n    return;\n  }\n  if (!force) {\n    throw new ToolkitError(`No context value matching key: ${key}`);\n  }\n}\n\nfunction printUnset(unset: string[]) {\n  if (unset.length === 0) return;\n  info('The following matched context values reset. They will be refreshed on next synthesis');\n  unset.forEach((match) => {\n    info('  %s', match);\n  });\n}\n\nfunction printReadonly(readonly: string[]) {\n  if (readonly.length === 0) return;\n  warning('The following matched context values could not be reset through the CLI');\n  readonly.forEach((match) => {\n    info('  %s', match);\n  });\n  info('');\n  info('This usually means they are configured in %s or %s', chalk.blue(PROJECT_CONFIG), chalk.blue(USER_DEFAULTS));\n}\n\nfunction keysByExpression(context: Context, expression: string) {\n  return context.keys.filter(minimatch.filter(expression));\n}\n\nfunction getUnsetAndReadonly(context: Context, matches: string[]) {\n  return matches.reduce<{ unset: string[]; readonly: string[] }>((acc, match) => {\n    if (context.has(match)) {\n      acc.readonly.push(match);\n    } else {\n      acc.unset.push(match);\n    }\n    return acc;\n  }, { unset: [], readonly: [] });\n}\n\nfunction keyByNumber(context: Context, n: number) {\n  for (const [i, key] of contextKeys(context)) {\n    if (n === i) {\n      return key;\n    }\n  }\n  throw new ToolkitError(`No context key with number: ${n}`);\n}\n\n/**\n * Return enumerated keys in a definitive order\n */\nfunction contextKeys(context: Context): [number, string][] {\n  const keys = context.keys;\n  keys.sort();\n  return enumerate1(keys);\n}\n\nfunction enumerate1<T>(xs: T[]): Array<[number, T]> {\n  const ret = new Array<[number, T]>();\n  let i = 1;\n  for (const x of xs) {\n    ret.push([i, x]);\n    i += 1;\n  }\n  return ret;\n}\n"]}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { type DescribeChangeSetOutput } from '@aws-cdk/cloudformation-diff';
|
|
2
|
+
import type * as cxapi from '@aws-cdk/cx-api';
|
|
3
|
+
import type { NestedStackTemplates } from '../api/cloudformation';
|
|
4
|
+
/**
|
|
5
|
+
* Output of formatStackDiff
|
|
6
|
+
*/
|
|
7
|
+
export interface FormatStackDiffOutput {
|
|
8
|
+
/**
|
|
9
|
+
* Number of stacks with diff changes
|
|
10
|
+
*/
|
|
11
|
+
readonly numStacksWithChanges: number;
|
|
12
|
+
/**
|
|
13
|
+
* Complete formatted diff
|
|
14
|
+
*/
|
|
15
|
+
readonly formattedDiff: string;
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Formats the differences between two template states and returns it as a string.
|
|
19
|
+
*
|
|
20
|
+
* @param oldTemplate the old/current state of the stack.
|
|
21
|
+
* @param newTemplate the new/target state of the stack.
|
|
22
|
+
* @param strict do not filter out AWS::CDK::Metadata or Rules
|
|
23
|
+
* @param context lines of context to use in arbitrary JSON diff
|
|
24
|
+
* @param quiet silences \'There were no differences\' messages
|
|
25
|
+
*
|
|
26
|
+
* @returns the formatted diff, and the number of stacks in this stack tree that have differences, including the top-level root stack
|
|
27
|
+
*/
|
|
28
|
+
export declare function formatStackDiff(oldTemplate: any, newTemplate: cxapi.CloudFormationStackArtifact, strict: boolean, context: number, quiet: boolean, stackName?: string, changeSet?: DescribeChangeSetOutput, isImport?: boolean, nestedStackTemplates?: {
|
|
29
|
+
[nestedStackLogicalId: string]: NestedStackTemplates;
|
|
30
|
+
}): FormatStackDiffOutput;
|
|
31
|
+
export declare enum RequireApproval {
|
|
32
|
+
Never = "never",
|
|
33
|
+
AnyChange = "any-change",
|
|
34
|
+
Broadening = "broadening"
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Output of formatSecurityDiff
|
|
38
|
+
*/
|
|
39
|
+
export interface FormatSecurityDiffOutput {
|
|
40
|
+
/**
|
|
41
|
+
* Complete formatted security diff, if it is prompt-worthy
|
|
42
|
+
*/
|
|
43
|
+
readonly formattedDiff?: string;
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Formats the security changes of this diff, if the change is impactful enough according to the approval level
|
|
47
|
+
*
|
|
48
|
+
* Returns the diff if the changes are prompt-worthy, an empty object otherwise.
|
|
49
|
+
*/
|
|
50
|
+
export declare function formatSecurityDiff(oldTemplate: any, newTemplate: cxapi.CloudFormationStackArtifact, requireApproval: RequireApproval, stackName?: string, changeSet?: DescribeChangeSetOutput): FormatSecurityDiffOutput;
|
|
@@ -0,0 +1,215 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.RequireApproval = void 0;
|
|
4
|
+
exports.formatStackDiff = formatStackDiff;
|
|
5
|
+
exports.formatSecurityDiff = formatSecurityDiff;
|
|
6
|
+
const stream_1 = require("stream");
|
|
7
|
+
const util_1 = require("util");
|
|
8
|
+
const cxschema = require("@aws-cdk/cloud-assembly-schema");
|
|
9
|
+
const cloudformation_diff_1 = require("@aws-cdk/cloudformation-diff");
|
|
10
|
+
const chalk = require("chalk");
|
|
11
|
+
const api_1 = require("../../../@aws-cdk/tmp-toolkit-helpers/src/api");
|
|
12
|
+
const logging_1 = require("../logging");
|
|
13
|
+
/*
|
|
14
|
+
* Custom writable stream that collects text into a string buffer.
|
|
15
|
+
* Used on classes that take in and directly write to a stream, but
|
|
16
|
+
* we intend to capture the output rather than print.
|
|
17
|
+
*/
|
|
18
|
+
class StringWriteStream extends stream_1.Writable {
|
|
19
|
+
constructor() {
|
|
20
|
+
super();
|
|
21
|
+
this.buffer = [];
|
|
22
|
+
}
|
|
23
|
+
_write(chunk, _encoding, callback) {
|
|
24
|
+
this.buffer.push(chunk.toString());
|
|
25
|
+
callback();
|
|
26
|
+
}
|
|
27
|
+
toString() {
|
|
28
|
+
return this.buffer.join('');
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Formats the differences between two template states and returns it as a string.
|
|
33
|
+
*
|
|
34
|
+
* @param oldTemplate the old/current state of the stack.
|
|
35
|
+
* @param newTemplate the new/target state of the stack.
|
|
36
|
+
* @param strict do not filter out AWS::CDK::Metadata or Rules
|
|
37
|
+
* @param context lines of context to use in arbitrary JSON diff
|
|
38
|
+
* @param quiet silences \'There were no differences\' messages
|
|
39
|
+
*
|
|
40
|
+
* @returns the formatted diff, and the number of stacks in this stack tree that have differences, including the top-level root stack
|
|
41
|
+
*/
|
|
42
|
+
function formatStackDiff(oldTemplate, newTemplate, strict, context, quiet, stackName, changeSet, isImport, nestedStackTemplates) {
|
|
43
|
+
let diff = (0, cloudformation_diff_1.fullDiff)(oldTemplate, newTemplate.template, changeSet, isImport);
|
|
44
|
+
// The stack diff is formatted via `Formatter`, which takes in a stream
|
|
45
|
+
// and sends its output directly to that stream. To faciliate use of the
|
|
46
|
+
// global CliIoHost, we create our own stream to capture the output of
|
|
47
|
+
// `Formatter` and return the output as a string for the consumer of
|
|
48
|
+
// `formatStackDiff` to decide what to do with it.
|
|
49
|
+
const stream = new StringWriteStream();
|
|
50
|
+
let numStacksWithChanges = 0;
|
|
51
|
+
let formattedDiff = '';
|
|
52
|
+
let filteredChangesCount = 0;
|
|
53
|
+
try {
|
|
54
|
+
// must output the stack name if there are differences, even if quiet
|
|
55
|
+
if (stackName && (!quiet || !diff.isEmpty)) {
|
|
56
|
+
stream.write((0, util_1.format)('Stack %s\n', chalk.bold(stackName)));
|
|
57
|
+
}
|
|
58
|
+
if (!quiet && isImport) {
|
|
59
|
+
stream.write('Parameters and rules created during migration do not affect resource configuration.\n');
|
|
60
|
+
}
|
|
61
|
+
// detect and filter out mangled characters from the diff
|
|
62
|
+
if (diff.differenceCount && !strict) {
|
|
63
|
+
const mangledNewTemplate = JSON.parse((0, cloudformation_diff_1.mangleLikeCloudFormation)(JSON.stringify(newTemplate.template)));
|
|
64
|
+
const mangledDiff = (0, cloudformation_diff_1.fullDiff)(oldTemplate, mangledNewTemplate, changeSet);
|
|
65
|
+
filteredChangesCount = Math.max(0, diff.differenceCount - mangledDiff.differenceCount);
|
|
66
|
+
if (filteredChangesCount > 0) {
|
|
67
|
+
diff = mangledDiff;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
// filter out 'AWS::CDK::Metadata' resources from the template
|
|
71
|
+
// filter out 'CheckBootstrapVersion' rules from the template
|
|
72
|
+
if (!strict) {
|
|
73
|
+
obscureDiff(diff);
|
|
74
|
+
}
|
|
75
|
+
if (!diff.isEmpty) {
|
|
76
|
+
numStacksWithChanges++;
|
|
77
|
+
// formatDifferences updates the stream with the formatted stack diff
|
|
78
|
+
(0, cloudformation_diff_1.formatDifferences)(stream, diff, {
|
|
79
|
+
...logicalIdMapFromTemplate(oldTemplate),
|
|
80
|
+
...buildLogicalToPathMap(newTemplate),
|
|
81
|
+
}, context);
|
|
82
|
+
// store the stream containing a formatted stack diff
|
|
83
|
+
formattedDiff = stream.toString();
|
|
84
|
+
}
|
|
85
|
+
else if (!quiet) {
|
|
86
|
+
(0, logging_1.info)(chalk.green('There were no differences'));
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
finally {
|
|
90
|
+
stream.end();
|
|
91
|
+
}
|
|
92
|
+
if (filteredChangesCount > 0) {
|
|
93
|
+
(0, logging_1.info)(chalk.yellow(`Omitted ${filteredChangesCount} changes because they are likely mangled non-ASCII characters. Use --strict to print them.`));
|
|
94
|
+
}
|
|
95
|
+
for (const nestedStackLogicalId of Object.keys(nestedStackTemplates ?? {})) {
|
|
96
|
+
if (!nestedStackTemplates) {
|
|
97
|
+
break;
|
|
98
|
+
}
|
|
99
|
+
const nestedStack = nestedStackTemplates[nestedStackLogicalId];
|
|
100
|
+
newTemplate._template = nestedStack.generatedTemplate;
|
|
101
|
+
const nextDiff = formatStackDiff(nestedStack.deployedTemplate, newTemplate, strict, context, quiet, nestedStack.physicalName ?? nestedStackLogicalId, undefined, isImport, nestedStack.nestedStackTemplates);
|
|
102
|
+
numStacksWithChanges += nextDiff.numStacksWithChanges;
|
|
103
|
+
formattedDiff += nextDiff.formattedDiff;
|
|
104
|
+
}
|
|
105
|
+
return {
|
|
106
|
+
numStacksWithChanges,
|
|
107
|
+
formattedDiff,
|
|
108
|
+
};
|
|
109
|
+
}
|
|
110
|
+
var RequireApproval;
|
|
111
|
+
(function (RequireApproval) {
|
|
112
|
+
RequireApproval["Never"] = "never";
|
|
113
|
+
RequireApproval["AnyChange"] = "any-change";
|
|
114
|
+
RequireApproval["Broadening"] = "broadening";
|
|
115
|
+
})(RequireApproval || (exports.RequireApproval = RequireApproval = {}));
|
|
116
|
+
/**
|
|
117
|
+
* Formats the security changes of this diff, if the change is impactful enough according to the approval level
|
|
118
|
+
*
|
|
119
|
+
* Returns the diff if the changes are prompt-worthy, an empty object otherwise.
|
|
120
|
+
*/
|
|
121
|
+
function formatSecurityDiff(oldTemplate, newTemplate, requireApproval, stackName, changeSet) {
|
|
122
|
+
const diff = (0, cloudformation_diff_1.fullDiff)(oldTemplate, newTemplate.template, changeSet);
|
|
123
|
+
if (diffRequiresApproval(diff, requireApproval)) {
|
|
124
|
+
(0, logging_1.info)((0, util_1.format)('Stack %s\n', chalk.bold(stackName)));
|
|
125
|
+
// eslint-disable-next-line max-len
|
|
126
|
+
(0, logging_1.warning)(`This deployment will make potentially sensitive changes according to your current security approval level (--require-approval ${requireApproval}).`);
|
|
127
|
+
(0, logging_1.warning)('Please confirm you intend to make the following modifications:\n');
|
|
128
|
+
// The security diff is formatted via `Formatter`, which takes in a stream
|
|
129
|
+
// and sends its output directly to that stream. To faciliate use of the
|
|
130
|
+
// global CliIoHost, we create our own stream to capture the output of
|
|
131
|
+
// `Formatter` and return the output as a string for the consumer of
|
|
132
|
+
// `formatSecurityDiff` to decide what to do with it.
|
|
133
|
+
const stream = new StringWriteStream();
|
|
134
|
+
try {
|
|
135
|
+
// formatSecurityChanges updates the stream with the formatted security diff
|
|
136
|
+
(0, cloudformation_diff_1.formatSecurityChanges)(stream, diff, buildLogicalToPathMap(newTemplate));
|
|
137
|
+
}
|
|
138
|
+
finally {
|
|
139
|
+
stream.end();
|
|
140
|
+
}
|
|
141
|
+
// store the stream containing a formatted stack diff
|
|
142
|
+
const formattedDiff = stream.toString();
|
|
143
|
+
return { formattedDiff };
|
|
144
|
+
}
|
|
145
|
+
return {};
|
|
146
|
+
}
|
|
147
|
+
/**
|
|
148
|
+
* Return whether the diff has security-impacting changes that need confirmation
|
|
149
|
+
*
|
|
150
|
+
* TODO: Filter the security impact determination based off of an enum that allows
|
|
151
|
+
* us to pick minimum "severities" to alert on.
|
|
152
|
+
*/
|
|
153
|
+
function diffRequiresApproval(diff, requireApproval) {
|
|
154
|
+
switch (requireApproval) {
|
|
155
|
+
case RequireApproval.Never: return false;
|
|
156
|
+
case RequireApproval.AnyChange: return diff.permissionsAnyChanges;
|
|
157
|
+
case RequireApproval.Broadening: return diff.permissionsBroadened;
|
|
158
|
+
default: throw new api_1.ToolkitError(`Unrecognized approval level: ${requireApproval}`);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
function buildLogicalToPathMap(stack) {
|
|
162
|
+
const map = {};
|
|
163
|
+
for (const md of stack.findMetadataByType(cxschema.ArtifactMetadataEntryType.LOGICAL_ID)) {
|
|
164
|
+
map[md.data] = md.path;
|
|
165
|
+
}
|
|
166
|
+
return map;
|
|
167
|
+
}
|
|
168
|
+
function logicalIdMapFromTemplate(template) {
|
|
169
|
+
const ret = {};
|
|
170
|
+
for (const [logicalId, resource] of Object.entries(template.Resources ?? {})) {
|
|
171
|
+
const path = resource?.Metadata?.['aws:cdk:path'];
|
|
172
|
+
if (path) {
|
|
173
|
+
ret[logicalId] = path;
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
return ret;
|
|
177
|
+
}
|
|
178
|
+
/**
|
|
179
|
+
* Remove any template elements that we don't want to show users.
|
|
180
|
+
* This is currently:
|
|
181
|
+
* - AWS::CDK::Metadata resource
|
|
182
|
+
* - CheckBootstrapVersion Rule
|
|
183
|
+
*/
|
|
184
|
+
function obscureDiff(diff) {
|
|
185
|
+
if (diff.unknown) {
|
|
186
|
+
// see https://github.com/aws/aws-cdk/issues/17942
|
|
187
|
+
diff.unknown = diff.unknown.filter(change => {
|
|
188
|
+
if (!change) {
|
|
189
|
+
return true;
|
|
190
|
+
}
|
|
191
|
+
if (change.newValue?.CheckBootstrapVersion) {
|
|
192
|
+
return false;
|
|
193
|
+
}
|
|
194
|
+
if (change.oldValue?.CheckBootstrapVersion) {
|
|
195
|
+
return false;
|
|
196
|
+
}
|
|
197
|
+
return true;
|
|
198
|
+
});
|
|
199
|
+
}
|
|
200
|
+
if (diff.resources) {
|
|
201
|
+
diff.resources = diff.resources.filter(change => {
|
|
202
|
+
if (!change) {
|
|
203
|
+
return true;
|
|
204
|
+
}
|
|
205
|
+
if (change.newResourceType === 'AWS::CDK::Metadata') {
|
|
206
|
+
return false;
|
|
207
|
+
}
|
|
208
|
+
if (change.oldResourceType === 'AWS::CDK::Metadata') {
|
|
209
|
+
return false;
|
|
210
|
+
}
|
|
211
|
+
return true;
|
|
212
|
+
});
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diff.js","sourceRoot":"","sources":["diff.ts"],"names":[],"mappings":"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","sourcesContent":["import { Writable } from 'stream';\nimport { format } from 'util';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport {\n  type DescribeChangeSetOutput,\n  type TemplateDiff,\n  formatDifferences,\n  formatSecurityChanges,\n  fullDiff,\n  mangleLikeCloudFormation,\n} from '@aws-cdk/cloudformation-diff';\nimport type * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport { ToolkitError } from '../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport type { NestedStackTemplates } from '../api/cloudformation';\nimport { info, warning } from '../logging';\n\n/*\n * Custom writable stream that collects text into a string buffer.\n * Used on classes that take in and directly write to a stream, but\n * we intend to capture the output rather than print.\n */\nclass StringWriteStream extends Writable {\n  private buffer: string[] = [];\n\n  constructor() {\n    super();\n  }\n\n  _write(chunk: any, _encoding: string, callback: (error?: Error | null) => void): void {\n    this.buffer.push(chunk.toString());\n    callback();\n  }\n\n  toString(): string {\n    return this.buffer.join('');\n  }\n}\n\n/**\n * Output of formatStackDiff\n */\nexport interface FormatStackDiffOutput {\n  /**\n   * Number of stacks with diff changes\n   */\n  readonly numStacksWithChanges: number;\n\n  /**\n   * Complete formatted diff\n   */\n  readonly formattedDiff: string;\n}\n\n/**\n * Formats the differences between two template states and returns it as a string.\n *\n * @param oldTemplate the old/current state of the stack.\n * @param newTemplate the new/target state of the stack.\n * @param strict      do not filter out AWS::CDK::Metadata or Rules\n * @param context     lines of context to use in arbitrary JSON diff\n * @param quiet       silences \\'There were no differences\\' messages\n *\n * @returns the formatted diff, and the number of stacks in this stack tree that have differences, including the top-level root stack\n */\nexport function formatStackDiff(\n  oldTemplate: any,\n  newTemplate: cxapi.CloudFormationStackArtifact,\n  strict: boolean,\n  context: number,\n  quiet: boolean,\n  stackName?: string,\n  changeSet?: DescribeChangeSetOutput,\n  isImport?: boolean,\n  nestedStackTemplates?: { [nestedStackLogicalId: string]: NestedStackTemplates }): FormatStackDiffOutput {\n  let diff = fullDiff(oldTemplate, newTemplate.template, changeSet, isImport);\n\n  // The stack diff is formatted via `Formatter`, which takes in a stream\n  // and sends its output directly to that stream. To faciliate use of the\n  // global CliIoHost, we create our own stream to capture the output of\n  // `Formatter` and return the output as a string for the consumer of\n  // `formatStackDiff` to decide what to do with it.\n  const stream = new StringWriteStream();\n\n  let numStacksWithChanges = 0;\n  let formattedDiff = '';\n  let filteredChangesCount = 0;\n  try {\n    // must output the stack name if there are differences, even if quiet\n    if (stackName && (!quiet || !diff.isEmpty)) {\n      stream.write(format('Stack %s\\n', chalk.bold(stackName)));\n    }\n\n    if (!quiet && isImport) {\n      stream.write('Parameters and rules created during migration do not affect resource configuration.\\n');\n    }\n\n    // detect and filter out mangled characters from the diff\n    if (diff.differenceCount && !strict) {\n      const mangledNewTemplate = JSON.parse(mangleLikeCloudFormation(JSON.stringify(newTemplate.template)));\n      const mangledDiff = fullDiff(oldTemplate, mangledNewTemplate, changeSet);\n      filteredChangesCount = Math.max(0, diff.differenceCount - mangledDiff.differenceCount);\n      if (filteredChangesCount > 0) {\n        diff = mangledDiff;\n      }\n    }\n\n    // filter out 'AWS::CDK::Metadata' resources from the template\n    // filter out 'CheckBootstrapVersion' rules from the template\n    if (!strict) {\n      obscureDiff(diff);\n    }\n\n    if (!diff.isEmpty) {\n      numStacksWithChanges++;\n\n      // formatDifferences updates the stream with the formatted stack diff\n      formatDifferences(stream, diff, {\n        ...logicalIdMapFromTemplate(oldTemplate),\n        ...buildLogicalToPathMap(newTemplate),\n      }, context);\n\n      // store the stream containing a formatted stack diff\n      formattedDiff = stream.toString();\n    } else if (!quiet) {\n      info(chalk.green('There were no differences'));\n    }\n  } finally {\n    stream.end();\n  }\n\n  if (filteredChangesCount > 0) {\n    info(chalk.yellow(`Omitted ${filteredChangesCount} changes because they are likely mangled non-ASCII characters. Use --strict to print them.`));\n  }\n\n  for (const nestedStackLogicalId of Object.keys(nestedStackTemplates ?? {})) {\n    if (!nestedStackTemplates) {\n      break;\n    }\n    const nestedStack = nestedStackTemplates[nestedStackLogicalId];\n\n    (newTemplate as any)._template = nestedStack.generatedTemplate;\n    const nextDiff = formatStackDiff(\n      nestedStack.deployedTemplate,\n      newTemplate,\n      strict,\n      context,\n      quiet,\n      nestedStack.physicalName ?? nestedStackLogicalId,\n      undefined,\n      isImport,\n      nestedStack.nestedStackTemplates,\n    );\n    numStacksWithChanges += nextDiff.numStacksWithChanges;\n    formattedDiff += nextDiff.formattedDiff;\n  }\n\n  return {\n    numStacksWithChanges,\n    formattedDiff,\n  };\n}\n\nexport enum RequireApproval {\n  Never = 'never',\n\n  AnyChange = 'any-change',\n\n  Broadening = 'broadening',\n}\n\n/**\n * Output of formatSecurityDiff\n */\nexport interface FormatSecurityDiffOutput {\n  /**\n   * Complete formatted security diff, if it is prompt-worthy\n   */\n  readonly formattedDiff?: string;\n}\n\n/**\n * Formats the security changes of this diff, if the change is impactful enough according to the approval level\n *\n * Returns the diff if the changes are prompt-worthy, an empty object otherwise.\n */\nexport function formatSecurityDiff(\n  oldTemplate: any,\n  newTemplate: cxapi.CloudFormationStackArtifact,\n  requireApproval: RequireApproval,\n  stackName?: string,\n  changeSet?: DescribeChangeSetOutput,\n): FormatSecurityDiffOutput {\n  const diff = fullDiff(oldTemplate, newTemplate.template, changeSet);\n\n  if (diffRequiresApproval(diff, requireApproval)) {\n    info(format('Stack %s\\n', chalk.bold(stackName)));\n\n    // eslint-disable-next-line max-len\n    warning(`This deployment will make potentially sensitive changes according to your current security approval level (--require-approval ${requireApproval}).`);\n    warning('Please confirm you intend to make the following modifications:\\n');\n\n    // The security diff is formatted via `Formatter`, which takes in a stream\n    // and sends its output directly to that stream. To faciliate use of the\n    // global CliIoHost, we create our own stream to capture the output of\n    // `Formatter` and return the output as a string for the consumer of\n    // `formatSecurityDiff` to decide what to do with it.\n    const stream = new StringWriteStream();\n    try {\n      // formatSecurityChanges updates the stream with the formatted security diff\n      formatSecurityChanges(stream, diff, buildLogicalToPathMap(newTemplate));\n    } finally {\n      stream.end();\n    }\n    // store the stream containing a formatted stack diff\n    const formattedDiff = stream.toString();\n    return { formattedDiff };\n  }\n  return {};\n}\n\n/**\n * Return whether the diff has security-impacting changes that need confirmation\n *\n * TODO: Filter the security impact determination based off of an enum that allows\n * us to pick minimum \"severities\" to alert on.\n */\nfunction diffRequiresApproval(diff: TemplateDiff, requireApproval: RequireApproval) {\n  switch (requireApproval) {\n    case RequireApproval.Never: return false;\n    case RequireApproval.AnyChange: return diff.permissionsAnyChanges;\n    case RequireApproval.Broadening: return diff.permissionsBroadened;\n    default: throw new ToolkitError(`Unrecognized approval level: ${requireApproval}`);\n  }\n}\n\nfunction buildLogicalToPathMap(stack: cxapi.CloudFormationStackArtifact) {\n  const map: { [id: string]: string } = {};\n  for (const md of stack.findMetadataByType(cxschema.ArtifactMetadataEntryType.LOGICAL_ID)) {\n    map[md.data as string] = md.path;\n  }\n  return map;\n}\n\nfunction logicalIdMapFromTemplate(template: any) {\n  const ret: Record<string, string> = {};\n\n  for (const [logicalId, resource] of Object.entries(template.Resources ?? {})) {\n    const path = (resource as any)?.Metadata?.['aws:cdk:path'];\n    if (path) {\n      ret[logicalId] = path;\n    }\n  }\n  return ret;\n}\n\n/**\n * Remove any template elements that we don't want to show users.\n * This is currently:\n * - AWS::CDK::Metadata resource\n * - CheckBootstrapVersion Rule\n */\nfunction obscureDiff(diff: TemplateDiff) {\n  if (diff.unknown) {\n    // see https://github.com/aws/aws-cdk/issues/17942\n    diff.unknown = diff.unknown.filter(change => {\n      if (!change) {\n        return true;\n      }\n      if (change.newValue?.CheckBootstrapVersion) {\n        return false;\n      }\n      if (change.oldValue?.CheckBootstrapVersion) {\n        return false;\n      }\n      return true;\n    });\n  }\n\n  if (diff.resources) {\n    diff.resources = diff.resources.filter(change => {\n      if (!change) {\n        return true;\n      }\n      if (change.newResourceType === 'AWS::CDK::Metadata') {\n        return false;\n      }\n      if (change.oldResourceType === 'AWS::CDK::Metadata') {\n        return false;\n      }\n      return true;\n    });\n  }\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from './init';
|
|
@@ -0,0 +1,18 @@
|
|
|
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("./init"), exports);
|
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEseUNBQXVCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9pbml0JztcbiJdfQ==
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.invokeBuiltinHooks = invokeBuiltinHooks;
|
|
4
|
+
const path = require("path");
|
|
5
|
+
const os_1 = require("./os");
|
|
6
|
+
const api_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api");
|
|
7
|
+
const util_1 = require("../../util");
|
|
8
|
+
/**
|
|
9
|
+
* Invoke hooks for the given init template
|
|
10
|
+
*
|
|
11
|
+
* Sometimes templates need more complex logic than just replacing tokens. A 'hook' can be
|
|
12
|
+
* used to do additional processing other than copying files.
|
|
13
|
+
*
|
|
14
|
+
* Hooks used to be defined externally to the CLI, by running arbitrarily
|
|
15
|
+
* substituted shell scripts in the target directory.
|
|
16
|
+
*
|
|
17
|
+
* In practice, they're all TypeScript files and all the same, and the dynamism
|
|
18
|
+
* that the original solution allowed wasn't used at all. Worse, since the CLI
|
|
19
|
+
* is now bundled the hooks can't even reuse code from the CLI libraries at all
|
|
20
|
+
* anymore, so all shared code would have to be copy/pasted.
|
|
21
|
+
*
|
|
22
|
+
* Bundle hooks as built-ins into the CLI, so they get bundled and can take advantage
|
|
23
|
+
* of all shared code.
|
|
24
|
+
*/
|
|
25
|
+
async function invokeBuiltinHooks(target, context) {
|
|
26
|
+
switch (target.language) {
|
|
27
|
+
case 'csharp':
|
|
28
|
+
if (['app', 'sample-app'].includes(target.templateName)) {
|
|
29
|
+
return dotnetAddProject(target.targetDirectory, context);
|
|
30
|
+
}
|
|
31
|
+
break;
|
|
32
|
+
case 'fsharp':
|
|
33
|
+
if (['app', 'sample-app'].includes(target.templateName)) {
|
|
34
|
+
return dotnetAddProject(target.targetDirectory, context, 'fsproj');
|
|
35
|
+
}
|
|
36
|
+
break;
|
|
37
|
+
case 'python':
|
|
38
|
+
// We can't call this file 'requirements.template.txt' because Dependabot needs to be able to find it.
|
|
39
|
+
// Therefore, keep the in-repo name but still substitute placeholders.
|
|
40
|
+
await context.substitutePlaceholdersIn('requirements.txt');
|
|
41
|
+
break;
|
|
42
|
+
case 'java':
|
|
43
|
+
// We can't call this file 'pom.template.xml'... for the same reason as Python above.
|
|
44
|
+
await context.substitutePlaceholdersIn('pom.xml');
|
|
45
|
+
break;
|
|
46
|
+
case 'javascript':
|
|
47
|
+
case 'typescript':
|
|
48
|
+
// See above, but for 'package.json'.
|
|
49
|
+
await context.substitutePlaceholdersIn('package.json');
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
async function dotnetAddProject(targetDirectory, context, ext = 'csproj') {
|
|
53
|
+
const pname = context.placeholder('name.PascalCased');
|
|
54
|
+
const slnPath = path.join(targetDirectory, 'src', `${pname}.sln`);
|
|
55
|
+
const csprojPath = path.join(targetDirectory, 'src', pname, `${pname}.${ext}`);
|
|
56
|
+
try {
|
|
57
|
+
await (0, os_1.shell)(['dotnet', 'sln', slnPath, 'add', csprojPath]);
|
|
58
|
+
}
|
|
59
|
+
catch (e) {
|
|
60
|
+
throw new api_1.ToolkitError(`Could not add project ${pname}.${ext} to solution ${pname}.sln. ${(0, util_1.formatErrorMessage)(e)}`);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5pdC1ob29rcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImluaXQtaG9va3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFpREEsZ0RBOEJDO0FBL0VELDZCQUE2QjtBQUM3Qiw2QkFBNkI7QUFDN0IsMEVBQWdGO0FBQ2hGLHFDQUFnRDtBQTZCaEQ7Ozs7Ozs7Ozs7Ozs7Ozs7R0FnQkc7QUFDSSxLQUFLLFVBQVUsa0JBQWtCLENBQUMsTUFBa0IsRUFBRSxPQUFvQjtJQUMvRSxRQUFRLE1BQU0sQ0FBQyxRQUFRLEVBQUUsQ0FBQztRQUN4QixLQUFLLFFBQVE7WUFDWCxJQUFJLENBQUMsS0FBSyxFQUFFLFlBQVksQ0FBQyxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsWUFBWSxDQUFDLEVBQUUsQ0FBQztnQkFDeEQsT0FBTyxnQkFBZ0IsQ0FBQyxNQUFNLENBQUMsZUFBZSxFQUFFLE9BQU8sQ0FBQyxDQUFDO1lBQzNELENBQUM7WUFDRCxNQUFNO1FBRVIsS0FBSyxRQUFRO1lBQ1gsSUFBSSxDQUFDLEtBQUssRUFBRSxZQUFZLENBQUMsQ0FBQyxRQUFRLENBQUMsTUFBTSxDQUFDLFlBQVksQ0FBQyxFQUFFLENBQUM7Z0JBQ3hELE9BQU8sZ0JBQWdCLENBQUMsTUFBTSxDQUFDLGVBQWUsRUFBRSxPQUFPLEVBQUUsUUFBUSxDQUFDLENBQUM7WUFDckUsQ0FBQztZQUNELE1BQU07UUFFUixLQUFLLFFBQVE7WUFDWCxzR0FBc0c7WUFDdEcsc0VBQXNFO1lBQ3RFLE1BQU0sT0FBTyxDQUFDLHdCQUF3QixDQUFDLGtCQUFrQixDQUFDLENBQUM7WUFDM0QsTUFBTTtRQUVSLEtBQUssTUFBTTtZQUNULHFGQUFxRjtZQUNyRixNQUFNLE9BQU8sQ0FBQyx3QkFBd0IsQ0FBQyxTQUFTLENBQUMsQ0FBQztZQUNsRCxNQUFNO1FBRVIsS0FBSyxZQUFZLENBQUM7UUFDbEIsS0FBSyxZQUFZO1lBQ2YscUNBQXFDO1lBQ3JDLE1BQU0sT0FBTyxDQUFDLHdCQUF3QixDQUFDLGNBQWMsQ0FBQyxDQUFDO0lBQzNELENBQUM7QUFDSCxDQUFDO0FBRUQsS0FBSyxVQUFVLGdCQUFnQixDQUFDLGVBQXVCLEVBQUUsT0FBb0IsRUFBRSxHQUFHLEdBQUcsUUFBUTtJQUMzRixNQUFNLEtBQUssR0FBRyxPQUFPLENBQUMsV0FBVyxDQUFDLGtCQUFrQixDQUFDLENBQUM7SUFDdEQsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxlQUFlLEVBQUUsS0FBSyxFQUFFLEdBQUcsS0FBSyxNQUFNLENBQUMsQ0FBQztJQUNsRSxNQUFNLFVBQVUsR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDLGVBQWUsRUFBRSxLQUFLLEVBQUUsS0FBSyxFQUFFLEdBQUcsS0FBSyxJQUFJLEdBQUcsRUFBRSxDQUFDLENBQUM7SUFDL0UsSUFBSSxDQUFDO1FBQ0gsTUFBTSxJQUFBLFVBQUssRUFBQyxDQUFDLFFBQVEsRUFBRSxLQUFLLEVBQUUsT0FBTyxFQUFFLEtBQUssRUFBRSxVQUFVLENBQUMsQ0FBQyxDQUFDO0lBQzdELENBQUM7SUFBQyxPQUFPLENBQU0sRUFBRSxDQUFDO1FBQ2hCLE1BQU0sSUFBSSxrQkFBWSxDQUFDLHlCQUF5QixLQUFLLElBQUksR0FBRyxnQkFBZ0IsS0FBSyxTQUFTLElBQUEseUJBQWtCLEVBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDO0lBQ3JILENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgcGF0aCBmcm9tICdwYXRoJztcbmltcG9ydCB7IHNoZWxsIH0gZnJvbSAnLi9vcyc7XG5pbXBvcnQgeyBUb29sa2l0RXJyb3IgfSBmcm9tICcuLi8uLi8uLi8uLi9AYXdzLWNkay90bXAtdG9vbGtpdC1oZWxwZXJzL3NyYy9hcGknO1xuaW1wb3J0IHsgZm9ybWF0RXJyb3JNZXNzYWdlIH0gZnJvbSAnLi4vLi4vdXRpbCc7XG5cbmV4cG9ydCB0eXBlIFN1YnN0aXR1dGVQbGFjZWhvbGRlcnMgPSAoLi4uZmlsZU5hbWVzOiBzdHJpbmdbXSkgPT4gUHJvbWlzZTx2b2lkPjtcblxuLyoqXG4gKiBIZWxwZXJzIHBhc3NlZCB0byBob29rIGZ1bmN0aW9uc1xuICovXG5leHBvcnQgaW50ZXJmYWNlIEhvb2tDb250ZXh0IHtcbiAgLyoqXG4gICAqIENhbGxiYWNrIGZ1bmN0aW9uIHRvIHJlcGxhY2UgcGxhY2Vob2xkZXJzIG9uIGFyYml0cmFyeSBmaWxlc1xuICAgKlxuICAgKiBUaGlzIG1ha2VzIHRva2VuIHN1YnN0aXR1dGlvbiBhdmFpbGFibGUgdG8gbm9uLWAudGVtcGxhdGVgIGZpbGVzLlxuICAgKi9cbiAgcmVhZG9ubHkgc3Vic3RpdHV0ZVBsYWNlaG9sZGVyc0luOiBTdWJzdGl0dXRlUGxhY2Vob2xkZXJzO1xuXG4gIC8qKlxuICAgKiBSZXR1cm4gYSBzaW5nbGUgcGxhY2Vob2xkZXJcbiAgICovXG4gIHBsYWNlaG9sZGVyKG5hbWU6IHN0cmluZyk6IHN0cmluZztcbn1cblxuZXhwb3J0IHR5cGUgSW52b2tlSG9vayA9ICh0YXJnZXREaXJlY3Rvcnk6IHN0cmluZywgY29udGV4dDogSG9va0NvbnRleHQpID0+IFByb21pc2U8dm9pZD47XG5cbmV4cG9ydCBpbnRlcmZhY2UgSG9va1RhcmdldCB7XG4gIHJlYWRvbmx5IHRhcmdldERpcmVjdG9yeTogc3RyaW5nO1xuICByZWFkb25seSB0ZW1wbGF0ZU5hbWU6IHN0cmluZztcbiAgcmVhZG9ubHkgbGFuZ3VhZ2U6IHN0cmluZztcbn1cblxuLyoqXG4gKiBJbnZva2UgaG9va3MgZm9yIHRoZSBnaXZlbiBpbml0IHRlbXBsYXRlXG4gKlxuICogU29tZXRpbWVzIHRlbXBsYXRlcyBuZWVkIG1vcmUgY29tcGxleCBsb2dpYyB0aGFuIGp1c3QgcmVwbGFjaW5nIHRva2Vucy4gQSAnaG9vaycgY2FuIGJlXG4gKiB1c2VkIHRvIGRvIGFkZGl0aW9uYWwgcHJvY2Vzc2luZyBvdGhlciB0aGFuIGNvcHlpbmcgZmlsZXMuXG4gKlxuICogSG9va3MgdXNlZCB0byBiZSBkZWZpbmVkIGV4dGVybmFsbHkgdG8gdGhlIENMSSwgYnkgcnVubmluZyBhcmJpdHJhcmlseVxuICogc3Vic3RpdHV0ZWQgc2hlbGwgc2NyaXB0cyBpbiB0aGUgdGFyZ2V0IGRpcmVjdG9yeS5cbiAqXG4gKiBJbiBwcmFjdGljZSwgdGhleSdyZSBhbGwgVHlwZVNjcmlwdCBmaWxlcyBhbmQgYWxsIHRoZSBzYW1lLCBhbmQgdGhlIGR5bmFtaXNtXG4gKiB0aGF0IHRoZSBvcmlnaW5hbCBzb2x1dGlvbiBhbGxvd2VkIHdhc24ndCB1c2VkIGF0IGFsbC4gV29yc2UsIHNpbmNlIHRoZSBDTElcbiAqIGlzIG5vdyBidW5kbGVkIHRoZSBob29rcyBjYW4ndCBldmVuIHJldXNlIGNvZGUgZnJvbSB0aGUgQ0xJIGxpYnJhcmllcyBhdCBhbGxcbiAqIGFueW1vcmUsIHNvIGFsbCBzaGFyZWQgY29kZSB3b3VsZCBoYXZlIHRvIGJlIGNvcHkvcGFzdGVkLlxuICpcbiAqIEJ1bmRsZSBob29rcyBhcyBidWlsdC1pbnMgaW50byB0aGUgQ0xJLCBzbyB0aGV5IGdldCBidW5kbGVkIGFuZCBjYW4gdGFrZSBhZHZhbnRhZ2VcbiAqIG9mIGFsbCBzaGFyZWQgY29kZS5cbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGludm9rZUJ1aWx0aW5Ib29rcyh0YXJnZXQ6IEhvb2tUYXJnZXQsIGNvbnRleHQ6IEhvb2tDb250ZXh0KSB7XG4gIHN3aXRjaCAodGFyZ2V0Lmxhbmd1YWdlKSB7XG4gICAgY2FzZSAnY3NoYXJwJzpcbiAgICAgIGlmIChbJ2FwcCcsICdzYW1wbGUtYXBwJ10uaW5jbHVkZXModGFyZ2V0LnRlbXBsYXRlTmFtZSkpIHtcbiAgICAgICAgcmV0dXJuIGRvdG5ldEFkZFByb2plY3QodGFyZ2V0LnRhcmdldERpcmVjdG9yeSwgY29udGV4dCk7XG4gICAgICB9XG4gICAgICBicmVhaztcblxuICAgIGNhc2UgJ2ZzaGFycCc6XG4gICAgICBpZiAoWydhcHAnLCAnc2FtcGxlLWFwcCddLmluY2x1ZGVzKHRhcmdldC50ZW1wbGF0ZU5hbWUpKSB7XG4gICAgICAgIHJldHVybiBkb3RuZXRBZGRQcm9qZWN0KHRhcmdldC50YXJnZXREaXJlY3RvcnksIGNvbnRleHQsICdmc3Byb2onKTtcbiAgICAgIH1cbiAgICAgIGJyZWFrO1xuXG4gICAgY2FzZSAncHl0aG9uJzpcbiAgICAgIC8vIFdlIGNhbid0IGNhbGwgdGhpcyBmaWxlICdyZXF1aXJlbWVudHMudGVtcGxhdGUudHh0JyBiZWNhdXNlIERlcGVuZGFib3QgbmVlZHMgdG8gYmUgYWJsZSB0byBmaW5kIGl0LlxuICAgICAgLy8gVGhlcmVmb3JlLCBrZWVwIHRoZSBpbi1yZXBvIG5hbWUgYnV0IHN0aWxsIHN1YnN0aXR1dGUgcGxhY2Vob2xkZXJzLlxuICAgICAgYXdhaXQgY29udGV4dC5zdWJzdGl0dXRlUGxhY2Vob2xkZXJzSW4oJ3JlcXVpcmVtZW50cy50eHQnKTtcbiAgICAgIGJyZWFrO1xuXG4gICAgY2FzZSAnamF2YSc6XG4gICAgICAvLyBXZSBjYW4ndCBjYWxsIHRoaXMgZmlsZSAncG9tLnRlbXBsYXRlLnhtbCcuLi4gZm9yIHRoZSBzYW1lIHJlYXNvbiBhcyBQeXRob24gYWJvdmUuXG4gICAgICBhd2FpdCBjb250ZXh0LnN1YnN0aXR1dGVQbGFjZWhvbGRlcnNJbigncG9tLnhtbCcpO1xuICAgICAgYnJlYWs7XG5cbiAgICBjYXNlICdqYXZhc2NyaXB0JzpcbiAgICBjYXNlICd0eXBlc2NyaXB0JzpcbiAgICAgIC8vIFNlZSBhYm92ZSwgYnV0IGZvciAncGFja2FnZS5qc29uJy5cbiAgICAgIGF3YWl0IGNvbnRleHQuc3Vic3RpdHV0ZVBsYWNlaG9sZGVyc0luKCdwYWNrYWdlLmpzb24nKTtcbiAgfVxufVxuXG5hc3luYyBmdW5jdGlvbiBkb3RuZXRBZGRQcm9qZWN0KHRhcmdldERpcmVjdG9yeTogc3RyaW5nLCBjb250ZXh0OiBIb29rQ29udGV4dCwgZXh0ID0gJ2NzcHJvaicpIHtcbiAgY29uc3QgcG5hbWUgPSBjb250ZXh0LnBsYWNlaG9sZGVyKCduYW1lLlBhc2NhbENhc2VkJyk7XG4gIGNvbnN0IHNsblBhdGggPSBwYXRoLmpvaW4odGFyZ2V0RGlyZWN0b3J5LCAnc3JjJywgYCR7cG5hbWV9LnNsbmApO1xuICBjb25zdCBjc3Byb2pQYXRoID0gcGF0aC5qb2luKHRhcmdldERpcmVjdG9yeSwgJ3NyYycsIHBuYW1lLCBgJHtwbmFtZX0uJHtleHR9YCk7XG4gIHRyeSB7XG4gICAgYXdhaXQgc2hlbGwoWydkb3RuZXQnLCAnc2xuJywgc2xuUGF0aCwgJ2FkZCcsIGNzcHJvalBhdGhdKTtcbiAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgdGhyb3cgbmV3IFRvb2xraXRFcnJvcihgQ291bGQgbm90IGFkZCBwcm9qZWN0ICR7cG5hbWV9LiR7ZXh0fSB0byBzb2x1dGlvbiAke3BuYW1lfS5zbG4uICR7Zm9ybWF0RXJyb3JNZXNzYWdlKGUpfWApO1xuICB9XG59XG4iXX0=
|