@aws-cdk/toolkit-lib 0.3.3 → 0.3.4

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.
Files changed (126) hide show
  1. package/api-extractor.json +3 -0
  2. package/build-info.json +2 -2
  3. package/db.json.gz +0 -0
  4. package/lib/actions/bootstrap/private/helpers.js +3 -3
  5. package/lib/actions/diff/index.d.ts +1 -1
  6. package/lib/actions/diff/index.js +1 -1
  7. package/lib/actions/diff/private/helpers.js +8 -8
  8. package/lib/actions/refactor/index.d.ts +12 -0
  9. package/lib/actions/refactor/index.js +1 -1
  10. package/lib/api/aws-auth/awscli-compatible.js +2 -2
  11. package/lib/api/aws-auth/credential-plugins.js +2 -2
  12. package/lib/api/aws-auth/sdk-provider.js +2 -2
  13. package/lib/api/aws-auth/sdk.js +2 -2
  14. package/lib/api/bootstrap/bootstrap-environment.js +2 -2
  15. package/lib/api/cloud-assembly/environment.js +1 -1
  16. package/lib/api/cloud-assembly/index.d.ts +1 -1
  17. package/lib/api/cloud-assembly/index.js +3 -2
  18. package/lib/api/cloud-assembly/private/context-aware-source.js +3 -3
  19. package/lib/api/cloud-assembly/private/exec.js +3 -3
  20. package/lib/api/cloud-assembly/private/prepare-source.js +4 -4
  21. package/lib/api/cloud-assembly/private/source-builder.d.ts +10 -8
  22. package/lib/api/cloud-assembly/private/source-builder.js +36 -19
  23. package/lib/api/cloud-assembly/private/stack-assembly.js +7 -7
  24. package/lib/api/cloud-assembly/source-builder.d.ts +19 -0
  25. package/lib/api/cloud-assembly/source-builder.js +1 -1
  26. package/lib/api/cloud-assembly/stack-collection.js +2 -2
  27. package/lib/api/cloudformation/evaluate-cloudformation-template.d.ts +1 -1
  28. package/lib/api/cloudformation/evaluate-cloudformation-template.js +2 -2
  29. package/lib/api/cloudformation/stack-helpers.js +2 -2
  30. package/lib/api/cloudformation/template-body-parameter.js +2 -2
  31. package/lib/api/context.js +2 -2
  32. package/lib/api/deployments/asset-publishing.js +2 -2
  33. package/lib/api/deployments/assets.js +5 -6
  34. package/lib/api/deployments/cfn-api.js +3 -3
  35. package/lib/api/deployments/checks.js +2 -2
  36. package/lib/api/deployments/deploy-stack.js +2 -2
  37. package/lib/api/deployments/deployment-result.js +2 -2
  38. package/lib/api/deployments/deployments.js +3 -3
  39. package/lib/api/diff/diff-formatter.js +2 -2
  40. package/lib/api/environment/environment-access.js +2 -2
  41. package/lib/api/environment/environment-resources.js +2 -2
  42. package/lib/api/environment/placeholders.js +1 -1
  43. package/lib/api/garbage-collection/garbage-collector.js +2 -2
  44. package/lib/api/garbage-collection/progress-printer.js +2 -2
  45. package/lib/api/garbage-collection/stack-refresh.js +2 -2
  46. package/lib/api/hotswap/appsync-mapping-templates.js +2 -2
  47. package/lib/api/hotswap/common.js +2 -2
  48. package/lib/api/hotswap/hotswap-deployments.js +2 -2
  49. package/lib/api/hotswap/lambda-functions.js +2 -2
  50. package/lib/api/index.d.ts +0 -1
  51. package/lib/api/index.js +1 -2
  52. package/lib/api/io/private/level-priority.js +1 -1
  53. package/lib/api/io/private/sdk-logger.d.ts +1 -1
  54. package/lib/api/io/private/sdk-logger.js +6 -6
  55. package/lib/api/io/private/span.js +1 -1
  56. package/lib/api/logs-monitor/logs-monitor.js +2 -2
  57. package/lib/api/notices/cached-data-source.d.ts +13 -0
  58. package/lib/api/notices/cached-data-source.js +77 -0
  59. package/lib/api/notices/filter.d.ts +59 -0
  60. package/lib/api/notices/filter.js +189 -0
  61. package/lib/api/notices/index.d.ts +1 -0
  62. package/lib/api/notices/index.js +18 -0
  63. package/lib/api/notices/notices.d.ts +111 -0
  64. package/lib/api/notices/notices.js +131 -0
  65. package/lib/api/notices/types.d.ts +37 -0
  66. package/lib/api/notices/types.js +3 -0
  67. package/lib/api/notices/web-data-source.d.ts +9 -0
  68. package/lib/api/notices/web-data-source.js +70 -0
  69. package/lib/api/plugin/context-provider-plugin.js +1 -1
  70. package/lib/api/plugin/plugin.d.ts +0 -2
  71. package/lib/api/plugin/plugin.js +4 -9
  72. package/lib/api/refactoring/cloudformation.d.ts +23 -0
  73. package/lib/api/refactoring/cloudformation.js +54 -1
  74. package/lib/api/refactoring/exclude.d.ts +29 -0
  75. package/lib/api/refactoring/exclude.js +94 -0
  76. package/lib/api/refactoring/index.d.ts +4 -23
  77. package/lib/api/refactoring/index.js +28 -60
  78. package/lib/api/resource-import/importer.js +2 -2
  79. package/lib/api/rwlock.js +2 -2
  80. package/lib/api/settings.js +2 -2
  81. package/lib/api/shared-private.js +31 -6186
  82. package/lib/api/toolkit-info.js +2 -2
  83. package/lib/api/work-graph/work-graph-builder.js +3 -3
  84. package/lib/api/work-graph/work-graph.js +2 -2
  85. package/lib/context-providers/ami.js +2 -2
  86. package/lib/context-providers/cc-api-provider.js +2 -2
  87. package/lib/context-providers/hosted-zones.js +2 -2
  88. package/lib/context-providers/index.js +4 -4
  89. package/lib/context-providers/keys.js +2 -2
  90. package/lib/context-providers/load-balancers.js +2 -2
  91. package/lib/context-providers/security-groups.js +2 -2
  92. package/lib/context-providers/ssm-parameters.js +2 -2
  93. package/lib/context-providers/vpcs.js +3 -3
  94. package/lib/index.d.ts +3 -1
  95. package/lib/index.js +4 -2
  96. package/lib/index_bg.wasm +0 -0
  97. package/lib/payloads/deploy.d.ts +7 -0
  98. package/lib/payloads/deploy.js +1 -1
  99. package/lib/payloads/hotswap.d.ts +1 -0
  100. package/lib/payloads/hotswap.js +1 -1
  101. package/lib/private/activity-printer/base.js +1 -1
  102. package/lib/toolkit/non-interactive-io-host.js +5 -4
  103. package/lib/toolkit/toolkit.d.ts +2 -2
  104. package/lib/toolkit/toolkit.js +40 -27
  105. package/lib/util/directories.js +2 -2
  106. package/lib/util/index.d.ts +1 -0
  107. package/lib/util/index.js +2 -1
  108. package/lib/util/json.js +1 -1
  109. package/lib/util/net.d.ts +9 -0
  110. package/lib/util/net.js +60 -0
  111. package/lib/util/objects.js +2 -2
  112. package/lib/util/types.js +1 -1
  113. package/lib/util/version-range.js +2 -2
  114. package/package.json +25 -26
  115. package/tsdoc.json +15 -0
  116. package/lib/api/notices.d.ts +0 -210
  117. package/lib/api/notices.js +0 -430
  118. package/lib/api/shared-private.js.map +0 -7
  119. package/lib/api/shared-public.d.ts +0 -2139
  120. package/lib/api/shared-public.js +0 -3272
  121. package/lib/api/shared-public.js.map +0 -7
  122. package/lib/private/util.d.ts +0 -1
  123. package/lib/private/util.js +0 -787
  124. package/lib/private/util.js.map +0 -7
  125. /package/lib/{api → toolkit}/toolkit-error.d.ts +0 -0
  126. /package/lib/{api → toolkit}/toolkit-error.js +0 -0
@@ -4,4 +4,4 @@ exports.isContextProviderPlugin = isContextProviderPlugin;
4
4
  function isContextProviderPlugin(x) {
5
5
  return typeof x === 'object' && !!x && !!x.getValue;
6
6
  }
7
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udGV4dC1wcm92aWRlci1wbHVnaW4uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJjb250ZXh0LXByb3ZpZGVyLXBsdWdpbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUlBLDBEQUVDO0FBRkQsU0FBZ0IsdUJBQXVCLENBQUMsQ0FBVTtJQUNoRCxPQUFPLE9BQU8sQ0FBQyxLQUFLLFFBQVEsSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBRSxDQUFTLENBQUMsUUFBUSxDQUFDO0FBQy9ELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIENvbnRleHRQcm92aWRlclBsdWdpbiB7XG4gIGdldFZhbHVlKGFyZ3M6IHtba2V5OiBzdHJpbmddOiBhbnl9KTogUHJvbWlzZTxhbnk+O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gaXNDb250ZXh0UHJvdmlkZXJQbHVnaW4oeDogdW5rbm93bik6IHggaXMgQ29udGV4dFByb3ZpZGVyUGx1Z2luIHtcbiAgcmV0dXJuIHR5cGVvZiB4ID09PSAnb2JqZWN0JyAmJiAhIXggJiYgISEoeCBhcyBhbnkpLmdldFZhbHVlO1xufVxuIl19
7
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udGV4dC1wcm92aWRlci1wbHVnaW4uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJjb250ZXh0LXByb3ZpZGVyLXBsdWdpbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUlBLDBEQUVDO0FBRkQsU0FBZ0IsdUJBQXVCLENBQUMsQ0FBVTtJQUNoRCxPQUFPLE9BQU8sQ0FBQyxLQUFLLFFBQVEsSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBRSxDQUFTLENBQUMsUUFBUSxDQUFDO0FBQy9ELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIENvbnRleHRQcm92aWRlclBsdWdpbiB7XG4gIGdldFZhbHVlKGFyZ3M6IHsgW2tleTogc3RyaW5nXTogYW55IH0pOiBQcm9taXNlPGFueT47XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBpc0NvbnRleHRQcm92aWRlclBsdWdpbih4OiB1bmtub3duKTogeCBpcyBDb250ZXh0UHJvdmlkZXJQbHVnaW4ge1xuICByZXR1cm4gdHlwZW9mIHggPT09ICdvYmplY3QnICYmICEheCAmJiAhISh4IGFzIGFueSkuZ2V0VmFsdWU7XG59XG4iXX0=
@@ -1,8 +1,6 @@
1
1
  import type { CredentialProviderSource, IPluginHost } from '@aws-cdk/cli-plugin-contract';
2
2
  import { type ContextProviderPlugin } from './context-provider-plugin';
3
3
  import type { IIoHost } from '../io';
4
- export declare let TESTING: boolean;
5
- export declare function markTesting(): void;
6
4
  /**
7
5
  * Class to manage a plugin collection
8
6
  *
@@ -1,15 +1,10 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.PluginHost = exports.TESTING = void 0;
4
- exports.markTesting = markTesting;
3
+ exports.PluginHost = void 0;
5
4
  const util_1 = require("util");
6
5
  const context_provider_plugin_1 = require("./context-provider-plugin");
7
- const private_1 = require("../private");
8
- const toolkit_error_1 = require("../toolkit-error");
9
- exports.TESTING = false;
10
- function markTesting() {
11
- exports.TESTING = true;
12
- }
6
+ const toolkit_error_1 = require("../../toolkit/toolkit-error");
7
+ const private_1 = require("../io/private");
13
8
  /**
14
9
  * Class to manage a plugin collection
15
10
  *
@@ -129,4 +124,4 @@ class PluginHost {
129
124
  }
130
125
  }
131
126
  exports.PluginHost = PluginHost;
132
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin.js","sourceRoot":"","sources":["plugin.ts"],"names":[],"mappings":";;;AASA,kCAEC;AAXD,+BAA+B;AAE/B,uEAAgG;AAEhG,wCAAyD;AACzD,oDAAgD;AAErC,QAAA,OAAO,GAAG,KAAK,CAAC;AAE3B,SAAgB,WAAW;IACzB,eAAO,GAAG,IAAI,CAAC;AACjB,CAAC;AAED;;;;;;GAMG;AACH,MAAa,UAAU;IACrB;;;OAGG;IACa,yBAAyB,GAAG,IAAI,KAAK,EAA4B,CAAC;IAElE,sBAAsB,GAA0C,EAAE,CAAC;IAE5E,MAAM,CAAW;IAEP,aAAa,GAAG,IAAI,GAAG,EAAU,CAAC;IAEnD;;;;;;;;;OASG;IACI,IAAI,CAAC,UAAkB,EAAE,MAAgB;QAC9C,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,OAAO,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;YAC7C,IAAI,MAAM,EAAE,CAAC;gBACX,IAAI,2BAAiB,CAAC,kBAAQ,CAAC,UAAU,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,oBAAoB,QAAQ,SAAS,UAAU,EAAE,CAAC,CAAC;YACtH,CAAC;YACD,OAAO,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QAChC,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,+EAA+E;YAC/E,yEAAyE;YACzE,gFAAgF;YAChF,wEAAwE;YACxE,MAAM,IAAI,4BAAY,CAAC,kDAAkD,UAAU,MAAM,CAAC,EAAE,CAAC,CAAC;QAChG,CAAC;IACH,CAAC;IAED;;;;OAIG;IACI,OAAO,CAAC,QAAgB;QAC7B,IAAI,CAAC;YACH,IAAI,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC;gBACrC,OAAO;YACT,CAAC;YAED,0DAA0D;YAC1D,MAAM,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC,CAAC;YACjC,mBAAmB;YACnB,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;gBACtB,MAAM,IAAI,4BAAY,CAAC,UAAU,QAAQ,yDAAyD,CAAC,CAAC;YACtG,CAAC;YACD,IAAI,MAAM,CAAC,IAAI,EAAE,CAAC;gBAChB,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACpB,CAAC;YAED,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QACnC,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,MAAM,4BAAY,CAAC,SAAS,CAAC,2BAA2B,QAAQ,GAAG,EAAE,CAAC,CAAC,CAAC;QAC1E,CAAC;QAED,SAAS,QAAQ,CAAC,CAAM;YACtB,OAAO,CAAC,IAAI,IAAI,IAAI,CAAC,CAAC,OAAO,KAAK,GAAG,CAAC;QACxC,CAAC;IACH,CAAC;IAED;;;;OAIG;IACI,gCAAgC,CAAC,MAAgC;QACtE,uDAAuD;QACvD,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAC9C,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA+BG;IACI,4BAA4B,CAAC,kBAA0B,EAAE,QAA+B;QAC7F,IAAI,CAAC,IAAA,iDAAuB,EAAC,QAAQ,CAAC,EAAE,CAAC;YACvC,MAAM,IAAI,4BAAY,CAAC,kEAAkE,IAAA,cAAO,EAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;QAChH,CAAC;QACD,IAAI,CAAC,sBAAsB,CAAC,kBAAkB,CAAC,GAAG,QAAQ,CAAC;IAC7D,CAAC;CACF;AAtHD,gCAsHC","sourcesContent":["import { inspect } from 'util';\nimport type { CredentialProviderSource, IPluginHost, Plugin } from '@aws-cdk/cli-plugin-contract';\nimport { type ContextProviderPlugin, isContextProviderPlugin } from './context-provider-plugin';\nimport type { IIoHost } from '../io';\nimport { IoDefaultMessages, IoHelper } from '../private';\nimport { ToolkitError } from '../toolkit-error';\n\nexport let TESTING = false;\n\nexport function markTesting() {\n  TESTING = true;\n}\n\n/**\n * Class to manage a plugin collection\n *\n * It provides a `load()` function that loads a JavaScript\n * module from disk, and gives it access to the `IPluginHost` interface\n * to register itself.\n */\nexport class PluginHost implements IPluginHost {\n  /**\n   * Access the currently registered CredentialProviderSources. New sources can\n   * be registered using the +registerCredentialProviderSource+ method.\n   */\n  public readonly credentialProviderSources = new Array<CredentialProviderSource>();\n\n  public readonly contextProviderPlugins: Record<string, ContextProviderPlugin> = {};\n\n  public ioHost?: IIoHost;\n\n  private readonly alreadyLoaded = new Set<string>();\n\n  /**\n   * Loads a plug-in into this PluginHost.\n   *\n   * Will use `require.resolve()` to get the most accurate representation of what\n   * code will get loaded in error messages. As such, it will not work in\n   * unit tests with Jest virtual modules becauase of <https://github.com/jestjs/jest/issues/9543>.\n   *\n   * @param moduleSpec the specification (path or name) of the plug-in module to be loaded.\n   * @param ioHost the I/O host to use for printing progress information\n   */\n  public load(moduleSpec: string, ioHost?: IIoHost) {\n    try {\n      const resolved = require.resolve(moduleSpec);\n      if (ioHost) {\n        new IoDefaultMessages(IoHelper.fromIoHost(ioHost, 'init')).debug(`Loading plug-in: ${resolved} from ${moduleSpec}`);\n      }\n      return this._doLoad(resolved);\n    } catch (e: any) {\n      // according to Node.js docs `MODULE_NOT_FOUND` is the only possible error here\n      // @see https://nodejs.org/api/modules.html#requireresolverequest-options\n      // Not using `withCause()` here, since the node error contains a \"Require Stack\"\n      // as part of the error message that is inherently useless to our users.\n      throw new ToolkitError(`Unable to resolve plug-in: Cannot find module '${moduleSpec}': ${e}`);\n    }\n  }\n\n  /**\n   * Do the loading given an already-resolved module name\n   *\n   * @internal\n   */\n  public _doLoad(resolved: string) {\n    try {\n      if (this.alreadyLoaded.has(resolved)) {\n        return;\n      }\n\n      /* eslint-disable @typescript-eslint/no-require-imports */\n      const plugin = require(resolved);\n      /* eslint-enable */\n      if (!isPlugin(plugin)) {\n        throw new ToolkitError(`Module ${resolved} is not a valid plug-in, or has an unsupported version.`);\n      }\n      if (plugin.init) {\n        plugin.init(this);\n      }\n\n      this.alreadyLoaded.add(resolved);\n    } catch (e: any) {\n      throw ToolkitError.withCause(`Unable to load plug-in '${resolved}'`, e);\n    }\n\n    function isPlugin(x: any): x is Plugin {\n      return x != null && x.version === '1';\n    }\n  }\n\n  /**\n   * Allows plug-ins to register new CredentialProviderSources.\n   *\n   * @param source a new CredentialProviderSource to register.\n   */\n  public registerCredentialProviderSource(source: CredentialProviderSource) {\n    // Forward to the right credentials-related plugin host\n    this.credentialProviderSources.push(source);\n  }\n\n  /**\n   * (EXPERIMENTAL) Allow plugins to register context providers\n   *\n   * Context providers are objects with the following method:\n   *\n   * ```ts\n   *   getValue(args: {[key: string]: any}): Promise<any>;\n   * ```\n   *\n   * Currently, they cannot reuse the CDK's authentication mechanisms, so they\n   * must be prepared to either not make AWS calls or use their own source of\n   * AWS credentials.\n   *\n   * This feature is experimental, and only intended to be used internally at Amazon\n   * as a trial.\n   *\n   * After registering with 'my-plugin-name', the provider must be addressed as follows:\n   *\n   * ```ts\n   * const value = ContextProvider.getValue(this, {\n   *   providerName: 'plugin',\n   *   props: {\n   *     pluginName: 'my-plugin-name',\n   *     myParameter1: 'xyz',\n   *   },\n   *   includeEnvironment: true | false,\n   *   dummyValue: 'what-to-return-on-the-first-pass',\n   * })\n   * ```\n   *\n   * @experimental\n   */\n  public registerContextProviderAlpha(pluginProviderName: string, provider: ContextProviderPlugin) {\n    if (!isContextProviderPlugin(provider)) {\n      throw new ToolkitError(`Object you gave me does not look like a ContextProviderPlugin: ${inspect(provider)}`);\n    }\n    this.contextProviderPlugins[pluginProviderName] = provider;\n  }\n}\n"]}
127
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin.js","sourceRoot":"","sources":["plugin.ts"],"names":[],"mappings":";;;AAAA,+BAA+B;AAE/B,uEAAgG;AAChG,+DAA2D;AAE3D,2CAA4D;AAE5D;;;;;;GAMG;AACH,MAAa,UAAU;IACrB;;;OAGG;IACa,yBAAyB,GAAG,IAAI,KAAK,EAA4B,CAAC;IAElE,sBAAsB,GAA0C,EAAE,CAAC;IAE5E,MAAM,CAAW;IAEP,aAAa,GAAG,IAAI,GAAG,EAAU,CAAC;IAEnD;;;;;;;;;OASG;IACI,IAAI,CAAC,UAAkB,EAAE,MAAgB;QAC9C,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,OAAO,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;YAC7C,IAAI,MAAM,EAAE,CAAC;gBACX,IAAI,2BAAiB,CAAC,kBAAQ,CAAC,UAAU,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,oBAAoB,QAAQ,SAAS,UAAU,EAAE,CAAC,CAAC;YACtH,CAAC;YACD,OAAO,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QAChC,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,+EAA+E;YAC/E,yEAAyE;YACzE,gFAAgF;YAChF,wEAAwE;YACxE,MAAM,IAAI,4BAAY,CAAC,kDAAkD,UAAU,MAAM,CAAC,EAAE,CAAC,CAAC;QAChG,CAAC;IACH,CAAC;IAED;;;;OAIG;IACI,OAAO,CAAC,QAAgB;QAC7B,IAAI,CAAC;YACH,IAAI,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC;gBACrC,OAAO;YACT,CAAC;YAED,0DAA0D;YAC1D,MAAM,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC,CAAC;YACjC,mBAAmB;YACnB,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;gBACtB,MAAM,IAAI,4BAAY,CAAC,UAAU,QAAQ,yDAAyD,CAAC,CAAC;YACtG,CAAC;YACD,IAAI,MAAM,CAAC,IAAI,EAAE,CAAC;gBAChB,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACpB,CAAC;YAED,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QACnC,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,MAAM,4BAAY,CAAC,SAAS,CAAC,2BAA2B,QAAQ,GAAG,EAAE,CAAC,CAAC,CAAC;QAC1E,CAAC;QAED,SAAS,QAAQ,CAAC,CAAM;YACtB,OAAO,CAAC,IAAI,IAAI,IAAI,CAAC,CAAC,OAAO,KAAK,GAAG,CAAC;QACxC,CAAC;IACH,CAAC;IAED;;;;OAIG;IACI,gCAAgC,CAAC,MAAgC;QACtE,uDAAuD;QACvD,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAC9C,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA+BG;IACI,4BAA4B,CAAC,kBAA0B,EAAE,QAA+B;QAC7F,IAAI,CAAC,IAAA,iDAAuB,EAAC,QAAQ,CAAC,EAAE,CAAC;YACvC,MAAM,IAAI,4BAAY,CAAC,kEAAkE,IAAA,cAAO,EAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;QAChH,CAAC;QACD,IAAI,CAAC,sBAAsB,CAAC,kBAAkB,CAAC,GAAG,QAAQ,CAAC;IAC7D,CAAC;CACF;AAtHD,gCAsHC","sourcesContent":["import { inspect } from 'util';\nimport type { CredentialProviderSource, IPluginHost, Plugin } from '@aws-cdk/cli-plugin-contract';\nimport { type ContextProviderPlugin, isContextProviderPlugin } from './context-provider-plugin';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\nimport type { IIoHost } from '../io';\nimport { IoDefaultMessages, IoHelper } from '../io/private';\n\n/**\n * Class to manage a plugin collection\n *\n * It provides a `load()` function that loads a JavaScript\n * module from disk, and gives it access to the `IPluginHost` interface\n * to register itself.\n */\nexport class PluginHost implements IPluginHost {\n  /**\n   * Access the currently registered CredentialProviderSources. New sources can\n   * be registered using the +registerCredentialProviderSource+ method.\n   */\n  public readonly credentialProviderSources = new Array<CredentialProviderSource>();\n\n  public readonly contextProviderPlugins: Record<string, ContextProviderPlugin> = {};\n\n  public ioHost?: IIoHost;\n\n  private readonly alreadyLoaded = new Set<string>();\n\n  /**\n   * Loads a plug-in into this PluginHost.\n   *\n   * Will use `require.resolve()` to get the most accurate representation of what\n   * code will get loaded in error messages. As such, it will not work in\n   * unit tests with Jest virtual modules becauase of <https://github.com/jestjs/jest/issues/9543>.\n   *\n   * @param moduleSpec the specification (path or name) of the plug-in module to be loaded.\n   * @param ioHost the I/O host to use for printing progress information\n   */\n  public load(moduleSpec: string, ioHost?: IIoHost) {\n    try {\n      const resolved = require.resolve(moduleSpec);\n      if (ioHost) {\n        new IoDefaultMessages(IoHelper.fromIoHost(ioHost, 'init')).debug(`Loading plug-in: ${resolved} from ${moduleSpec}`);\n      }\n      return this._doLoad(resolved);\n    } catch (e: any) {\n      // according to Node.js docs `MODULE_NOT_FOUND` is the only possible error here\n      // @see https://nodejs.org/api/modules.html#requireresolverequest-options\n      // Not using `withCause()` here, since the node error contains a \"Require Stack\"\n      // as part of the error message that is inherently useless to our users.\n      throw new ToolkitError(`Unable to resolve plug-in: Cannot find module '${moduleSpec}': ${e}`);\n    }\n  }\n\n  /**\n   * Do the loading given an already-resolved module name\n   *\n   * @internal\n   */\n  public _doLoad(resolved: string) {\n    try {\n      if (this.alreadyLoaded.has(resolved)) {\n        return;\n      }\n\n      /* eslint-disable @typescript-eslint/no-require-imports */\n      const plugin = require(resolved);\n      /* eslint-enable */\n      if (!isPlugin(plugin)) {\n        throw new ToolkitError(`Module ${resolved} is not a valid plug-in, or has an unsupported version.`);\n      }\n      if (plugin.init) {\n        plugin.init(this);\n      }\n\n      this.alreadyLoaded.add(resolved);\n    } catch (e: any) {\n      throw ToolkitError.withCause(`Unable to load plug-in '${resolved}'`, e);\n    }\n\n    function isPlugin(x: any): x is Plugin {\n      return x != null && x.version === '1';\n    }\n  }\n\n  /**\n   * Allows plug-ins to register new CredentialProviderSources.\n   *\n   * @param source a new CredentialProviderSource to register.\n   */\n  public registerCredentialProviderSource(source: CredentialProviderSource) {\n    // Forward to the right credentials-related plugin host\n    this.credentialProviderSources.push(source);\n  }\n\n  /**\n   * (EXPERIMENTAL) Allow plugins to register context providers\n   *\n   * Context providers are objects with the following method:\n   *\n   * ```ts\n   *   getValue(args: {[key: string]: any}): Promise<any>;\n   * ```\n   *\n   * Currently, they cannot reuse the CDK's authentication mechanisms, so they\n   * must be prepared to either not make AWS calls or use their own source of\n   * AWS credentials.\n   *\n   * This feature is experimental, and only intended to be used internally at Amazon\n   * as a trial.\n   *\n   * After registering with 'my-plugin-name', the provider must be addressed as follows:\n   *\n   * ```ts\n   * const value = ContextProvider.getValue(this, {\n   *   providerName: 'plugin',\n   *   props: {\n   *     pluginName: 'my-plugin-name',\n   *     myParameter1: 'xyz',\n   *   },\n   *   includeEnvironment: true | false,\n   *   dummyValue: 'what-to-return-on-the-first-pass',\n   * })\n   * ```\n   *\n   * @experimental\n   */\n  public registerContextProviderAlpha(pluginProviderName: string, provider: ContextProviderPlugin) {\n    if (!isContextProviderPlugin(provider)) {\n      throw new ToolkitError(`Object you gave me does not look like a ContextProviderPlugin: ${inspect(provider)}`);\n    }\n    this.contextProviderPlugins[pluginProviderName] = provider;\n  }\n}\n"]}
@@ -1,3 +1,4 @@
1
+ import type { TypedMapping } from '@aws-cdk/cloudformation-diff';
1
2
  import type * as cxapi from '@aws-cdk/cx-api';
2
3
  export interface CloudFormationTemplate {
3
4
  Resources?: {
@@ -13,3 +14,25 @@ export interface CloudFormationStack {
13
14
  readonly stackName: string;
14
15
  readonly template: CloudFormationTemplate;
15
16
  }
17
+ /**
18
+ * This class mirrors the `ResourceLocation` interface from CloudFormation,
19
+ * but is richer, since it has a reference to the stack object, rather than
20
+ * merely the stack name.
21
+ */
22
+ export declare class ResourceLocation {
23
+ readonly stack: CloudFormationStack;
24
+ readonly logicalResourceId: string;
25
+ constructor(stack: CloudFormationStack, logicalResourceId: string);
26
+ toPath(): string;
27
+ getType(): string;
28
+ equalTo(other: ResourceLocation): boolean;
29
+ }
30
+ /**
31
+ * A mapping between a source and a destination location.
32
+ */
33
+ export declare class ResourceMapping {
34
+ readonly source: ResourceLocation;
35
+ readonly destination: ResourceLocation;
36
+ constructor(source: ResourceLocation, destination: ResourceLocation);
37
+ toTypedMapping(): TypedMapping;
38
+ }
@@ -1,3 +1,56 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xvdWRmb3JtYXRpb24uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJjbG91ZGZvcm1hdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgKiBhcyBjeGFwaSBmcm9tICdAYXdzLWNkay9jeC1hcGknO1xuXG5leHBvcnQgaW50ZXJmYWNlIENsb3VkRm9ybWF0aW9uVGVtcGxhdGUge1xuICBSZXNvdXJjZXM/OiB7XG4gICAgW2xvZ2ljYWxJZDogc3RyaW5nXToge1xuICAgICAgVHlwZTogc3RyaW5nO1xuICAgICAgUHJvcGVydGllcz86IGFueTtcbiAgICAgIE1ldGFkYXRhPzogUmVjb3JkPHN0cmluZywgYW55PjtcbiAgICB9O1xuICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENsb3VkRm9ybWF0aW9uU3RhY2sge1xuICByZWFkb25seSBlbnZpcm9ubWVudDogY3hhcGkuRW52aXJvbm1lbnQ7XG4gIHJlYWRvbmx5IHN0YWNrTmFtZTogc3RyaW5nO1xuICByZWFkb25seSB0ZW1wbGF0ZTogQ2xvdWRGb3JtYXRpb25UZW1wbGF0ZTtcbn1cbiJdfQ==
3
+ exports.ResourceMapping = exports.ResourceLocation = void 0;
4
+ /**
5
+ * This class mirrors the `ResourceLocation` interface from CloudFormation,
6
+ * but is richer, since it has a reference to the stack object, rather than
7
+ * merely the stack name.
8
+ */
9
+ class ResourceLocation {
10
+ stack;
11
+ logicalResourceId;
12
+ constructor(stack, logicalResourceId) {
13
+ this.stack = stack;
14
+ this.logicalResourceId = logicalResourceId;
15
+ }
16
+ toPath() {
17
+ const stack = this.stack;
18
+ const resource = stack.template.Resources?.[this.logicalResourceId];
19
+ const result = resource?.Metadata?.['aws:cdk:path'];
20
+ if (result != null) {
21
+ return result;
22
+ }
23
+ // If the path is not available, we can use stack name and logical ID
24
+ return `${stack.stackName}.${this.logicalResourceId}`;
25
+ }
26
+ getType() {
27
+ const resource = this.stack.template.Resources?.[this.logicalResourceId ?? ''];
28
+ return resource?.Type ?? 'Unknown';
29
+ }
30
+ equalTo(other) {
31
+ return this.logicalResourceId === other.logicalResourceId && this.stack.stackName === other.stack.stackName;
32
+ }
33
+ }
34
+ exports.ResourceLocation = ResourceLocation;
35
+ /**
36
+ * A mapping between a source and a destination location.
37
+ */
38
+ class ResourceMapping {
39
+ source;
40
+ destination;
41
+ constructor(source, destination) {
42
+ this.source = source;
43
+ this.destination = destination;
44
+ }
45
+ toTypedMapping() {
46
+ return {
47
+ // the type is the same in both source and destination,
48
+ // so we can use either one
49
+ type: this.source.getType(),
50
+ sourcePath: this.source.toPath(),
51
+ destinationPath: this.destination.toPath(),
52
+ };
53
+ }
54
+ }
55
+ exports.ResourceMapping = ResourceMapping;
56
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xvdWRmb3JtYXRpb24uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJjbG91ZGZvcm1hdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFtQkE7Ozs7R0FJRztBQUNILE1BQWEsZ0JBQWdCO0lBQ0M7SUFBNEM7SUFBeEUsWUFBNEIsS0FBMEIsRUFBa0IsaUJBQXlCO1FBQXJFLFVBQUssR0FBTCxLQUFLLENBQXFCO1FBQWtCLHNCQUFpQixHQUFqQixpQkFBaUIsQ0FBUTtJQUNqRyxDQUFDO0lBRU0sTUFBTTtRQUNYLE1BQU0sS0FBSyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUM7UUFDekIsTUFBTSxRQUFRLEdBQUcsS0FBSyxDQUFDLFFBQVEsQ0FBQyxTQUFTLEVBQUUsQ0FBQyxJQUFJLENBQUMsaUJBQWlCLENBQUMsQ0FBQztRQUNwRSxNQUFNLE1BQU0sR0FBRyxRQUFRLEVBQUUsUUFBUSxFQUFFLENBQUMsY0FBYyxDQUFDLENBQUM7UUFFcEQsSUFBSSxNQUFNLElBQUksSUFBSSxFQUFFLENBQUM7WUFDbkIsT0FBTyxNQUFNLENBQUM7UUFDaEIsQ0FBQztRQUVELHFFQUFxRTtRQUNyRSxPQUFPLEdBQUcsS0FBSyxDQUFDLFNBQVMsSUFBSSxJQUFJLENBQUMsaUJBQWlCLEVBQUUsQ0FBQztJQUN4RCxDQUFDO0lBRU0sT0FBTztRQUNaLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsUUFBUSxDQUFDLFNBQVMsRUFBRSxDQUFDLElBQUksQ0FBQyxpQkFBaUIsSUFBSSxFQUFFLENBQUMsQ0FBQztRQUMvRSxPQUFPLFFBQVEsRUFBRSxJQUFJLElBQUksU0FBUyxDQUFDO0lBQ3JDLENBQUM7SUFFTSxPQUFPLENBQUMsS0FBdUI7UUFDcEMsT0FBTyxJQUFJLENBQUMsaUJBQWlCLEtBQUssS0FBSyxDQUFDLGlCQUFpQixJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsU0FBUyxLQUFLLEtBQUssQ0FBQyxLQUFLLENBQUMsU0FBUyxDQUFDO0lBQzlHLENBQUM7Q0FDRjtBQXpCRCw0Q0F5QkM7QUFFRDs7R0FFRztBQUNILE1BQWEsZUFBZTtJQUNFO0lBQTBDO0lBQXRFLFlBQTRCLE1BQXdCLEVBQWtCLFdBQTZCO1FBQXZFLFdBQU0sR0FBTixNQUFNLENBQWtCO1FBQWtCLGdCQUFXLEdBQVgsV0FBVyxDQUFrQjtJQUNuRyxDQUFDO0lBRU0sY0FBYztRQUNuQixPQUFPO1lBQ0wsdURBQXVEO1lBQ3ZELDJCQUEyQjtZQUMzQixJQUFJLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLEVBQUU7WUFDM0IsVUFBVSxFQUFFLElBQUksQ0FBQyxNQUFNLENBQUMsTUFBTSxFQUFFO1lBQ2hDLGVBQWUsRUFBRSxJQUFJLENBQUMsV0FBVyxDQUFDLE1BQU0sRUFBRTtTQUMzQyxDQUFDO0lBQ0osQ0FBQztDQUNGO0FBYkQsMENBYUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IFR5cGVkTWFwcGluZyB9IGZyb20gJ0Bhd3MtY2RrL2Nsb3VkZm9ybWF0aW9uLWRpZmYnO1xuaW1wb3J0IHR5cGUgKiBhcyBjeGFwaSBmcm9tICdAYXdzLWNkay9jeC1hcGknO1xuXG5leHBvcnQgaW50ZXJmYWNlIENsb3VkRm9ybWF0aW9uVGVtcGxhdGUge1xuICBSZXNvdXJjZXM/OiB7XG4gICAgW2xvZ2ljYWxJZDogc3RyaW5nXToge1xuICAgICAgVHlwZTogc3RyaW5nO1xuICAgICAgUHJvcGVydGllcz86IGFueTtcbiAgICAgIE1ldGFkYXRhPzogUmVjb3JkPHN0cmluZywgYW55PjtcbiAgICB9O1xuICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENsb3VkRm9ybWF0aW9uU3RhY2sge1xuICByZWFkb25seSBlbnZpcm9ubWVudDogY3hhcGkuRW52aXJvbm1lbnQ7XG4gIHJlYWRvbmx5IHN0YWNrTmFtZTogc3RyaW5nO1xuICByZWFkb25seSB0ZW1wbGF0ZTogQ2xvdWRGb3JtYXRpb25UZW1wbGF0ZTtcbn1cblxuLyoqXG4gKiBUaGlzIGNsYXNzIG1pcnJvcnMgdGhlIGBSZXNvdXJjZUxvY2F0aW9uYCBpbnRlcmZhY2UgZnJvbSBDbG91ZEZvcm1hdGlvbixcbiAqIGJ1dCBpcyByaWNoZXIsIHNpbmNlIGl0IGhhcyBhIHJlZmVyZW5jZSB0byB0aGUgc3RhY2sgb2JqZWN0LCByYXRoZXIgdGhhblxuICogbWVyZWx5IHRoZSBzdGFjayBuYW1lLlxuICovXG5leHBvcnQgY2xhc3MgUmVzb3VyY2VMb2NhdGlvbiB7XG4gIGNvbnN0cnVjdG9yKHB1YmxpYyByZWFkb25seSBzdGFjazogQ2xvdWRGb3JtYXRpb25TdGFjaywgcHVibGljIHJlYWRvbmx5IGxvZ2ljYWxSZXNvdXJjZUlkOiBzdHJpbmcpIHtcbiAgfVxuXG4gIHB1YmxpYyB0b1BhdGgoKTogc3RyaW5nIHtcbiAgICBjb25zdCBzdGFjayA9IHRoaXMuc3RhY2s7XG4gICAgY29uc3QgcmVzb3VyY2UgPSBzdGFjay50ZW1wbGF0ZS5SZXNvdXJjZXM/Llt0aGlzLmxvZ2ljYWxSZXNvdXJjZUlkXTtcbiAgICBjb25zdCByZXN1bHQgPSByZXNvdXJjZT8uTWV0YWRhdGE/LlsnYXdzOmNkazpwYXRoJ107XG5cbiAgICBpZiAocmVzdWx0ICE9IG51bGwpIHtcbiAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgfVxuXG4gICAgLy8gSWYgdGhlIHBhdGggaXMgbm90IGF2YWlsYWJsZSwgd2UgY2FuIHVzZSBzdGFjayBuYW1lIGFuZCBsb2dpY2FsIElEXG4gICAgcmV0dXJuIGAke3N0YWNrLnN0YWNrTmFtZX0uJHt0aGlzLmxvZ2ljYWxSZXNvdXJjZUlkfWA7XG4gIH1cblxuICBwdWJsaWMgZ2V0VHlwZSgpOiBzdHJpbmcge1xuICAgIGNvbnN0IHJlc291cmNlID0gdGhpcy5zdGFjay50ZW1wbGF0ZS5SZXNvdXJjZXM/Llt0aGlzLmxvZ2ljYWxSZXNvdXJjZUlkID8/ICcnXTtcbiAgICByZXR1cm4gcmVzb3VyY2U/LlR5cGUgPz8gJ1Vua25vd24nO1xuICB9XG5cbiAgcHVibGljIGVxdWFsVG8ob3RoZXI6IFJlc291cmNlTG9jYXRpb24pOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5sb2dpY2FsUmVzb3VyY2VJZCA9PT0gb3RoZXIubG9naWNhbFJlc291cmNlSWQgJiYgdGhpcy5zdGFjay5zdGFja05hbWUgPT09IG90aGVyLnN0YWNrLnN0YWNrTmFtZTtcbiAgfVxufVxuXG4vKipcbiAqIEEgbWFwcGluZyBiZXR3ZWVuIGEgc291cmNlIGFuZCBhIGRlc3RpbmF0aW9uIGxvY2F0aW9uLlxuICovXG5leHBvcnQgY2xhc3MgUmVzb3VyY2VNYXBwaW5nIHtcbiAgY29uc3RydWN0b3IocHVibGljIHJlYWRvbmx5IHNvdXJjZTogUmVzb3VyY2VMb2NhdGlvbiwgcHVibGljIHJlYWRvbmx5IGRlc3RpbmF0aW9uOiBSZXNvdXJjZUxvY2F0aW9uKSB7XG4gIH1cblxuICBwdWJsaWMgdG9UeXBlZE1hcHBpbmcoKTogVHlwZWRNYXBwaW5nIHtcbiAgICByZXR1cm4ge1xuICAgICAgLy8gdGhlIHR5cGUgaXMgdGhlIHNhbWUgaW4gYm90aCBzb3VyY2UgYW5kIGRlc3RpbmF0aW9uLFxuICAgICAgLy8gc28gd2UgY2FuIHVzZSBlaXRoZXIgb25lXG4gICAgICB0eXBlOiB0aGlzLnNvdXJjZS5nZXRUeXBlKCksXG4gICAgICBzb3VyY2VQYXRoOiB0aGlzLnNvdXJjZS50b1BhdGgoKSxcbiAgICAgIGRlc3RpbmF0aW9uUGF0aDogdGhpcy5kZXN0aW5hdGlvbi50b1BhdGgoKSxcbiAgICB9O1xuICB9XG59XG5cbiJdfQ==
@@ -0,0 +1,29 @@
1
+ import type { AssemblyManifest } from '@aws-cdk/cloud-assembly-schema';
2
+ import type { ResourceLocation } from './cloudformation';
3
+ export interface ExcludeList {
4
+ isExcluded(location: ResourceLocation): boolean;
5
+ }
6
+ export declare class ManifestExcludeList implements ExcludeList {
7
+ private readonly excludedLocations;
8
+ constructor(manifest: AssemblyManifest);
9
+ private getExcludedLocations;
10
+ isExcluded(location: ResourceLocation): boolean;
11
+ }
12
+ export declare class InMemoryExcludeList implements ExcludeList {
13
+ private readonly excludedLocations;
14
+ private readonly excludedPaths;
15
+ constructor(items: string[]);
16
+ isExcluded(location: ResourceLocation): boolean;
17
+ }
18
+ export declare class UnionExcludeList implements ExcludeList {
19
+ private readonly excludeLists;
20
+ constructor(excludeLists: ExcludeList[]);
21
+ isExcluded(location: ResourceLocation): boolean;
22
+ }
23
+ export declare class NeverExclude implements ExcludeList {
24
+ isExcluded(_location: ResourceLocation): boolean;
25
+ }
26
+ export declare class AlwaysExclude implements ExcludeList {
27
+ isExcluded(_location: ResourceLocation): boolean;
28
+ }
29
+ export declare function fromManifestAndExclusionList(manifest: AssemblyManifest, exclude?: string[]): ExcludeList;
@@ -0,0 +1,94 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.AlwaysExclude = exports.NeverExclude = exports.UnionExcludeList = exports.InMemoryExcludeList = exports.ManifestExcludeList = void 0;
4
+ exports.fromManifestAndExclusionList = fromManifestAndExclusionList;
5
+ const cloud_assembly_schema_1 = require("@aws-cdk/cloud-assembly-schema");
6
+ class ManifestExcludeList {
7
+ excludedLocations;
8
+ constructor(manifest) {
9
+ this.excludedLocations = this.getExcludedLocations(manifest);
10
+ }
11
+ getExcludedLocations(asmManifest) {
12
+ // First, we need to filter the artifacts to only include CloudFormation stacks
13
+ const stackManifests = Object.entries(asmManifest.artifacts ?? {}).filter(([_, manifest]) => manifest.type === cloud_assembly_schema_1.ArtifactType.AWS_CLOUDFORMATION_STACK);
14
+ const result = [];
15
+ for (let [stackName, manifest] of stackManifests) {
16
+ const locations = Object.values(manifest.metadata ?? {})
17
+ // Then pick only the resources in each stack marked with DO_NOT_REFACTOR
18
+ .filter((entries) => entries.some((entry) => entry.type === cloud_assembly_schema_1.ArtifactMetadataEntryType.DO_NOT_REFACTOR && entry.data === true))
19
+ // Finally, get the logical ID of each resource
20
+ .map((entries) => {
21
+ const logicalIdEntry = entries.find((entry) => entry.type === cloud_assembly_schema_1.ArtifactMetadataEntryType.LOGICAL_ID);
22
+ const location = {
23
+ StackName: stackName,
24
+ LogicalResourceId: logicalIdEntry.data,
25
+ };
26
+ return location;
27
+ });
28
+ result.push(...locations);
29
+ }
30
+ return result;
31
+ }
32
+ isExcluded(location) {
33
+ return this.excludedLocations.some((loc) => loc.StackName === location.stack.stackName && loc.LogicalResourceId === location.logicalResourceId);
34
+ }
35
+ }
36
+ exports.ManifestExcludeList = ManifestExcludeList;
37
+ class InMemoryExcludeList {
38
+ excludedLocations;
39
+ excludedPaths;
40
+ constructor(items) {
41
+ this.excludedLocations = [];
42
+ this.excludedPaths = [];
43
+ if (items.length === 0) {
44
+ return;
45
+ }
46
+ const locationRegex = /^[A-Za-z0-9]+\.[A-Za-z0-9]+$/;
47
+ items.forEach((item) => {
48
+ if (locationRegex.test(item)) {
49
+ const [stackName, logicalId] = item.split('.');
50
+ this.excludedLocations.push({
51
+ StackName: stackName,
52
+ LogicalResourceId: logicalId,
53
+ });
54
+ }
55
+ else {
56
+ this.excludedPaths.push(item);
57
+ }
58
+ });
59
+ }
60
+ isExcluded(location) {
61
+ const containsLocation = this.excludedLocations.some((loc) => {
62
+ return loc.StackName === location.stack.stackName && loc.LogicalResourceId === location.logicalResourceId;
63
+ });
64
+ const containsPath = this.excludedPaths.some((path) => location.toPath() === path);
65
+ return containsLocation || containsPath;
66
+ }
67
+ }
68
+ exports.InMemoryExcludeList = InMemoryExcludeList;
69
+ class UnionExcludeList {
70
+ excludeLists;
71
+ constructor(excludeLists) {
72
+ this.excludeLists = excludeLists;
73
+ }
74
+ isExcluded(location) {
75
+ return this.excludeLists.some((excludeList) => excludeList.isExcluded(location));
76
+ }
77
+ }
78
+ exports.UnionExcludeList = UnionExcludeList;
79
+ class NeverExclude {
80
+ isExcluded(_location) {
81
+ return false;
82
+ }
83
+ }
84
+ exports.NeverExclude = NeverExclude;
85
+ class AlwaysExclude {
86
+ isExcluded(_location) {
87
+ return true;
88
+ }
89
+ }
90
+ exports.AlwaysExclude = AlwaysExclude;
91
+ function fromManifestAndExclusionList(manifest, exclude) {
92
+ return new UnionExcludeList([new ManifestExcludeList(manifest), new InMemoryExcludeList(exclude ?? [])]);
93
+ }
94
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exclude.js","sourceRoot":"","sources":["exclude.ts"],"names":[],"mappings":"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","sourcesContent":["import type { AssemblyManifest } from '@aws-cdk/cloud-assembly-schema';\nimport { ArtifactMetadataEntryType, ArtifactType } from '@aws-cdk/cloud-assembly-schema';\nimport type { ResourceLocation as CfnResourceLocation } from '@aws-sdk/client-cloudformation';\nimport type { ResourceLocation } from './cloudformation';\n\nexport interface ExcludeList {\n  isExcluded(location: ResourceLocation): boolean;\n}\n\nexport class ManifestExcludeList implements ExcludeList {\n  private readonly excludedLocations: CfnResourceLocation[];\n\n  constructor(manifest: AssemblyManifest) {\n    this.excludedLocations = this.getExcludedLocations(manifest);\n  }\n\n  private getExcludedLocations(asmManifest: AssemblyManifest): CfnResourceLocation[] {\n    // First, we need to filter the artifacts to only include CloudFormation stacks\n    const stackManifests = Object.entries(asmManifest.artifacts ?? {}).filter(\n      ([_, manifest]) => manifest.type === ArtifactType.AWS_CLOUDFORMATION_STACK,\n    );\n\n    const result: CfnResourceLocation[] = [];\n    for (let [stackName, manifest] of stackManifests) {\n      const locations = Object.values(manifest.metadata ?? {})\n        // Then pick only the resources in each stack marked with DO_NOT_REFACTOR\n        .filter((entries) =>\n          entries.some((entry) => entry.type === ArtifactMetadataEntryType.DO_NOT_REFACTOR && entry.data === true),\n        )\n        // Finally, get the logical ID of each resource\n        .map((entries) => {\n          const logicalIdEntry = entries.find((entry) => entry.type === ArtifactMetadataEntryType.LOGICAL_ID);\n          const location: CfnResourceLocation = {\n            StackName: stackName,\n            LogicalResourceId: logicalIdEntry!.data! as string,\n          };\n          return location;\n        });\n      result.push(...locations);\n    }\n    return result;\n  }\n\n  isExcluded(location: ResourceLocation): boolean {\n    return this.excludedLocations.some(\n      (loc) => loc.StackName === location.stack.stackName && loc.LogicalResourceId === location.logicalResourceId,\n    );\n  }\n}\n\nexport class InMemoryExcludeList implements ExcludeList {\n  private readonly excludedLocations: CfnResourceLocation[];\n  private readonly excludedPaths: string[];\n\n  constructor(items: string[]) {\n    this.excludedLocations = [];\n    this.excludedPaths = [];\n\n    if (items.length === 0) {\n      return;\n    }\n\n    const locationRegex = /^[A-Za-z0-9]+\\.[A-Za-z0-9]+$/;\n\n    items.forEach((item: string) => {\n      if (locationRegex.test(item)) {\n        const [stackName, logicalId] = item.split('.');\n        this.excludedLocations.push({\n          StackName: stackName,\n          LogicalResourceId: logicalId,\n        });\n      } else {\n        this.excludedPaths.push(item);\n      }\n    });\n  }\n\n  isExcluded(location: ResourceLocation): boolean {\n    const containsLocation = this.excludedLocations.some((loc) => {\n      return loc.StackName === location.stack.stackName && loc.LogicalResourceId === location.logicalResourceId;\n    });\n\n    const containsPath = this.excludedPaths.some((path) => location.toPath() === path);\n    return containsLocation || containsPath;\n  }\n}\n\nexport class UnionExcludeList implements ExcludeList {\n  constructor(private readonly excludeLists: ExcludeList[]) {\n  }\n\n  isExcluded(location: ResourceLocation): boolean {\n    return this.excludeLists.some((excludeList) => excludeList.isExcluded(location));\n  }\n}\n\nexport class NeverExclude implements ExcludeList {\n  isExcluded(_location: ResourceLocation): boolean {\n    return false;\n  }\n}\n\nexport class AlwaysExclude implements ExcludeList {\n  isExcluded(_location: ResourceLocation): boolean {\n    return true;\n  }\n}\n\nexport function fromManifestAndExclusionList(manifest: AssemblyManifest, exclude?: string[]): ExcludeList {\n  return new UnionExcludeList([new ManifestExcludeList(manifest), new InMemoryExcludeList(exclude ?? [])]);\n}\n\n"]}
@@ -1,6 +1,9 @@
1
1
  import type { TypedMapping } from '@aws-cdk/cloudformation-diff';
2
2
  import type { SdkProvider } from '../aws-auth/private';
3
3
  import type { CloudFormationStack } from './cloudformation';
4
+ import { ResourceMapping, ResourceLocation } from './cloudformation';
5
+ import { type ExcludeList } from './exclude';
6
+ export * from './exclude';
4
7
  /**
5
8
  * Represents a set of possible movements of a resource from one location
6
9
  * to another. In the ideal case, there is only one source and only one
@@ -12,28 +15,6 @@ export declare class AmbiguityError extends Error {
12
15
  constructor(movements: ResourceMovement[]);
13
16
  paths(): [string[], string[]][];
14
17
  }
15
- /**
16
- * This class mirrors the `ResourceLocation` interface from CloudFormation,
17
- * but is richer, since it has a reference to the stack object, rather than
18
- * merely the stack name.
19
- */
20
- export declare class ResourceLocation {
21
- readonly stack: CloudFormationStack;
22
- readonly logicalResourceId: string;
23
- constructor(stack: CloudFormationStack, logicalResourceId: string);
24
- toPath(): string;
25
- getType(): string;
26
- equalTo(other: ResourceLocation): boolean;
27
- }
28
- /**
29
- * A mapping between a source and a destination location.
30
- */
31
- export declare class ResourceMapping {
32
- readonly source: ResourceLocation;
33
- readonly destination: ResourceLocation;
34
- constructor(source: ResourceLocation, destination: ResourceLocation);
35
- toTypedMapping(): TypedMapping;
36
- }
37
18
  export declare function resourceMovements(before: CloudFormationStack[], after: CloudFormationStack[]): ResourceMovement[];
38
19
  export declare function ambiguousMovements(movements: ResourceMovement[]): ResourceMovement[];
39
20
  /**
@@ -46,6 +27,6 @@ export declare function resourceMappings(movements: ResourceMovement[], stacks?:
46
27
  * that were moved from one location (stack + logical ID) to another. The comparison
47
28
  * is done per environment.
48
29
  */
49
- export declare function findResourceMovements(stacks: CloudFormationStack[], sdkProvider: SdkProvider): Promise<ResourceMovement[]>;
30
+ export declare function findResourceMovements(stacks: CloudFormationStack[], sdkProvider: SdkProvider, exclude?: ExcludeList): Promise<ResourceMovement[]>;
50
31
  export declare function formatTypedMappings(mappings: TypedMapping[]): string;
51
32
  export declare function formatAmbiguousMappings(paths: [string[], string[]][]): string;
@@ -1,6 +1,20 @@
1
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
+ };
2
16
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.ResourceMapping = exports.ResourceLocation = exports.AmbiguityError = void 0;
17
+ exports.AmbiguityError = void 0;
4
18
  exports.resourceMovements = resourceMovements;
5
19
  exports.ambiguousMovements = ambiguousMovements;
6
20
  exports.resourceMappings = resourceMappings;
@@ -11,7 +25,10 @@ const cloudformation_diff_1 = require("@aws-cdk/cloudformation-diff");
11
25
  const util_1 = require("../../util");
12
26
  const plugin_1 = require("../plugin");
13
27
  const streams_1 = require("../streams");
28
+ const cloudformation_1 = require("./cloudformation");
14
29
  const digest_1 = require("./digest");
30
+ const exclude_1 = require("./exclude");
31
+ __exportStar(require("./exclude"), exports);
15
32
  class AmbiguityError extends Error {
16
33
  movements;
17
34
  constructor(movements) {
@@ -26,58 +43,6 @@ class AmbiguityError extends Error {
26
43
  }
27
44
  }
28
45
  exports.AmbiguityError = AmbiguityError;
29
- /**
30
- * This class mirrors the `ResourceLocation` interface from CloudFormation,
31
- * but is richer, since it has a reference to the stack object, rather than
32
- * merely the stack name.
33
- */
34
- class ResourceLocation {
35
- stack;
36
- logicalResourceId;
37
- constructor(stack, logicalResourceId) {
38
- this.stack = stack;
39
- this.logicalResourceId = logicalResourceId;
40
- }
41
- toPath() {
42
- const stack = this.stack;
43
- const resource = stack.template.Resources?.[this.logicalResourceId];
44
- const result = resource?.Metadata?.['aws:cdk:path'];
45
- if (result != null) {
46
- return result;
47
- }
48
- // If the path is not available, we can use stack name and logical ID
49
- return `${stack.stackName}.${this.logicalResourceId}`;
50
- }
51
- getType() {
52
- const resource = this.stack.template.Resources?.[this.logicalResourceId ?? ''];
53
- return resource?.Type ?? 'Unknown';
54
- }
55
- equalTo(other) {
56
- return this.logicalResourceId === other.logicalResourceId && this.stack.stackName === other.stack.stackName;
57
- }
58
- }
59
- exports.ResourceLocation = ResourceLocation;
60
- /**
61
- * A mapping between a source and a destination location.
62
- */
63
- class ResourceMapping {
64
- source;
65
- destination;
66
- constructor(source, destination) {
67
- this.source = source;
68
- this.destination = destination;
69
- }
70
- toTypedMapping() {
71
- return {
72
- // the type is the same in both source and destination,
73
- // so we can use either one
74
- type: this.source.getType(),
75
- sourcePath: this.source.toPath(),
76
- destinationPath: this.destination.toPath(),
77
- };
78
- }
79
- }
80
- exports.ResourceMapping = ResourceMapping;
81
46
  function groupByKey(entries) {
82
47
  const result = {};
83
48
  for (const [hash, location] of entries) {
@@ -106,7 +71,7 @@ function ambiguousMovements(movements) {
106
71
  *
107
72
  */
108
73
  function resourceMappings(movements, stacks) {
109
- const predicate = stacks == null
74
+ const stacksPredicate = stacks == null
110
75
  ? () => true
111
76
  : (m) => {
112
77
  // Any movement that involves one of the selected stacks (either moving from or to)
@@ -116,8 +81,8 @@ function resourceMappings(movements, stacks) {
116
81
  };
117
82
  return movements
118
83
  .filter(([pre, post]) => pre.length === 1 && post.length === 1 && !pre[0].equalTo(post[0]))
119
- .map(([pre, post]) => new ResourceMapping(pre[0], post[0]))
120
- .filter(predicate);
84
+ .map(([pre, post]) => new cloudformation_1.ResourceMapping(pre[0], post[0]))
85
+ .filter(stacksPredicate);
121
86
  }
122
87
  function removeUnmovedResources(m) {
123
88
  const result = {};
@@ -157,7 +122,7 @@ function zip(m1, m2) {
157
122
  function resourceDigests(stack) {
158
123
  const digests = (0, digest_1.computeResourceDigests)(stack.template);
159
124
  return Object.entries(digests).map(([logicalId, digest]) => {
160
- const location = new ResourceLocation(stack, logicalId);
125
+ const location = new cloudformation_1.ResourceLocation(stack, logicalId);
161
126
  return [digest, location];
162
127
  });
163
128
  }
@@ -166,7 +131,7 @@ function resourceDigests(stack) {
166
131
  * that were moved from one location (stack + logical ID) to another. The comparison
167
132
  * is done per environment.
168
133
  */
169
- async function findResourceMovements(stacks, sdkProvider) {
134
+ async function findResourceMovements(stacks, sdkProvider, exclude = new exclude_1.NeverExclude()) {
170
135
  const stackGroups = new Map();
171
136
  // Group stacks by environment
172
137
  for (const stack of stacks) {
@@ -185,7 +150,10 @@ async function findResourceMovements(stacks, sdkProvider) {
185
150
  for (const [_, [before, after]] of stackGroups) {
186
151
  result.push(...resourceMovements(before, after));
187
152
  }
188
- return result;
153
+ return result.filter(mov => {
154
+ const after = mov[1];
155
+ return after.every(l => !exclude.isExcluded(l));
156
+ });
189
157
  }
190
158
  async function getDeployedStacks(sdkProvider, environment) {
191
159
  const cfn = (await sdkProvider.forEnvironment(environment, plugin_1.Mode.ForReading)).sdk.cloudFormation();
@@ -220,4 +188,4 @@ function formatAmbiguousMappings(paths) {
220
188
  (0, cloudformation_diff_1.formatAmbiguousMappings)(stream, paths);
221
189
  return stream.toString();
222
190
  }
223
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import type { TypedMapping } from '@aws-cdk/cloudformation-diff';\nimport {\n  formatAmbiguousMappings as fmtAmbiguousMappings,\n  formatTypedMappings as fmtTypedMappings,\n} from '@aws-cdk/cloudformation-diff';\nimport type * as cxapi from '@aws-cdk/cx-api';\nimport type { StackSummary } from '@aws-sdk/client-cloudformation';\nimport { deserializeStructure } from '../../util';\nimport type { SdkProvider } from '../aws-auth/private';\nimport { Mode } from '../plugin';\nimport { StringWriteStream } from '../streams';\nimport type { CloudFormationStack } from './cloudformation';\nimport { computeResourceDigests, hashObject } from './digest';\n\n/**\n * Represents a set of possible movements of a resource from one location\n * to another. In the ideal case, there is only one source and only one\n * destination.\n */\nexport type ResourceMovement = [ResourceLocation[], ResourceLocation[]];\n\nexport class AmbiguityError extends Error {\n  constructor(public readonly movements: ResourceMovement[]) {\n    super('Ambiguous resource mappings');\n  }\n\n  public paths(): [string[], string[]][] {\n    return this.movements.map(([a, b]) => [convert(a), convert(b)]);\n\n    function convert(locations: ResourceLocation[]): string[] {\n      return locations.map((l) => l.toPath());\n    }\n  }\n}\n\n/**\n * This class mirrors the `ResourceLocation` interface from CloudFormation,\n * but is richer, since it has a reference to the stack object, rather than\n * merely the stack name.\n */\nexport class ResourceLocation {\n  constructor(public readonly stack: CloudFormationStack, public readonly logicalResourceId: string) {\n  }\n\n  public toPath(): string {\n    const stack = this.stack;\n    const resource = stack.template.Resources?.[this.logicalResourceId];\n    const result = resource?.Metadata?.['aws:cdk:path'];\n\n    if (result != null) {\n      return result;\n    }\n\n    // If the path is not available, we can use stack name and logical ID\n    return `${stack.stackName}.${this.logicalResourceId}`;\n  }\n\n  public getType(): string {\n    const resource = this.stack.template.Resources?.[this.logicalResourceId ?? ''];\n    return resource?.Type ?? 'Unknown';\n  }\n\n  public equalTo(other: ResourceLocation): boolean {\n    return this.logicalResourceId === other.logicalResourceId && this.stack.stackName === other.stack.stackName;\n  }\n}\n\n/**\n * A mapping between a source and a destination location.\n */\nexport class ResourceMapping {\n  constructor(public readonly source: ResourceLocation, public readonly destination: ResourceLocation) {\n  }\n\n  public toTypedMapping(): TypedMapping {\n    return {\n      // the type is the same in both source and destination,\n      // so we can use either one\n      type: this.source.getType(),\n      sourcePath: this.source.toPath(),\n      destinationPath: this.destination.toPath(),\n    };\n  }\n}\n\nfunction groupByKey<A>(entries: [string, A][]): Record<string, A[]> {\n  const result: Record<string, A[]> = {};\n\n  for (const [hash, location] of entries) {\n    if (hash in result) {\n      result[hash].push(location);\n    } else {\n      result[hash] = [location];\n    }\n  }\n\n  return result;\n}\n\nexport function resourceMovements(before: CloudFormationStack[], after: CloudFormationStack[]): ResourceMovement[] {\n  return Object.values(\n    removeUnmovedResources(\n      zip(groupByKey(before.flatMap(resourceDigests)), groupByKey(after.flatMap(resourceDigests))),\n    ),\n  );\n}\n\nexport function ambiguousMovements(movements: ResourceMovement[]) {\n  // A movement is considered ambiguous if these two conditions are met:\n  //  1. Both sides have at least one element (otherwise, it's just addition or deletion)\n  //  2. At least one side has more than one element\n  return movements\n    .filter(([pre, post]) => pre.length > 0 && post.length > 0)\n    .filter(([pre, post]) => pre.length > 1 || post.length > 1);\n}\n\n/**\n * Converts a list of unambiguous resource movements into a list of resource mappings.\n *\n */\nexport function resourceMappings(movements: ResourceMovement[], stacks?: CloudFormationStack[]): ResourceMapping[] {\n  const predicate = stacks == null\n    ? () => true\n    : (m: ResourceMapping) => {\n      // Any movement that involves one of the selected stacks (either moving from or to)\n      // is considered a candidate for refactoring.\n      const stackNames = [m.source.stack.stackName, m.destination.stack.stackName];\n      return stacks.some((stack) => stackNames.includes(stack.stackName));\n    };\n\n  return movements\n    .filter(([pre, post]) => pre.length === 1 && post.length === 1 && !pre[0].equalTo(post[0]))\n    .map(([pre, post]) => new ResourceMapping(pre[0], post[0]))\n    .filter(predicate);\n}\n\nfunction removeUnmovedResources(\n  m: Record<string, ResourceMovement>,\n): Record<string, ResourceMovement> {\n  const result: Record<string, ResourceMovement> = {};\n  for (const [hash, [before, after]] of Object.entries(m)) {\n    const common = before.filter((b) => after.some((a) => a.equalTo(b)));\n    result[hash] = [\n      before.filter((b) => !common.some((c) => b.equalTo(c))),\n      after.filter((a) => !common.some((c) => a.equalTo(c))),\n    ];\n  }\n\n  return result;\n}\n\n/**\n * For each hash, identifying a single resource, zip the two lists of locations,\n * producing a resource movement\n */\nfunction zip(\n  m1: Record<string, ResourceLocation[]>,\n  m2: Record<string, ResourceLocation[]>,\n): Record<string, ResourceMovement> {\n  const result: Record<string, ResourceMovement> = {};\n\n  for (const [hash, locations] of Object.entries(m1)) {\n    if (hash in m2) {\n      result[hash] = [locations, m2[hash]];\n    } else {\n      result[hash] = [locations, []];\n    }\n  }\n\n  for (const [hash, locations] of Object.entries(m2)) {\n    if (!(hash in m1)) {\n      result[hash] = [[], locations];\n    }\n  }\n\n  return result;\n}\n\n/**\n * Computes a list of pairs [digest, location] for each resource in the stack.\n */\nfunction resourceDigests(stack: CloudFormationStack): [string, ResourceLocation][] {\n  const digests = computeResourceDigests(stack.template);\n\n  return Object.entries(digests).map(([logicalId, digest]) => {\n    const location: ResourceLocation = new ResourceLocation(stack, logicalId);\n    return [digest, location];\n  });\n}\n\n/**\n * Compares the deployed state to the cloud assembly state, and finds all resources\n * that were moved from one location (stack + logical ID) to another. The comparison\n * is done per environment.\n */\nexport async function findResourceMovements(\n  stacks: CloudFormationStack[],\n  sdkProvider: SdkProvider,\n): Promise<ResourceMovement[]> {\n  const stackGroups: Map<string, [CloudFormationStack[], CloudFormationStack[]]> = new Map();\n\n  // Group stacks by environment\n  for (const stack of stacks) {\n    const environment = stack.environment;\n    const key = hashObject(environment);\n    if (stackGroups.has(key)) {\n      stackGroups.get(key)![1].push(stack);\n    } else {\n      // The first time we see an environment, we need to fetch all stacks deployed to it.\n      const before = await getDeployedStacks(sdkProvider, environment);\n      stackGroups.set(key, [before, [stack]]);\n    }\n  }\n\n  const result: ResourceMovement[] = [];\n  for (const [_, [before, after]] of stackGroups) {\n    result.push(...resourceMovements(before, after));\n  }\n  return result;\n}\n\nasync function getDeployedStacks(\n  sdkProvider: SdkProvider,\n  environment: cxapi.Environment,\n): Promise<CloudFormationStack[]> {\n  const cfn = (await sdkProvider.forEnvironment(environment, Mode.ForReading)).sdk.cloudFormation();\n\n  const summaries = await cfn.paginatedListStacks({\n    StackStatusFilter: [\n      'CREATE_COMPLETE',\n      'UPDATE_COMPLETE',\n      'UPDATE_ROLLBACK_COMPLETE',\n      'IMPORT_COMPLETE',\n      'ROLLBACK_COMPLETE',\n    ],\n  });\n\n  const normalize = async (summary: StackSummary) => {\n    const templateCommandOutput = await cfn.getTemplate({ StackName: summary.StackName! });\n    const template = deserializeStructure(templateCommandOutput.TemplateBody ?? '{}');\n    return {\n      environment,\n      stackName: summary.StackName!,\n      template,\n    };\n  };\n\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  return Promise.all(summaries.map(normalize));\n}\n\nexport function formatTypedMappings(mappings: TypedMapping[]): string {\n  const stream = new StringWriteStream();\n  fmtTypedMappings(stream, mappings);\n  return stream.toString();\n}\n\nexport function formatAmbiguousMappings(paths: [string[], string[]][]): string {\n  const stream = new StringWriteStream();\n  fmtAmbiguousMappings(stream, paths);\n  return stream.toString();\n}\n"]}
191
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import type { TypedMapping } from '@aws-cdk/cloudformation-diff';\nimport {\n  formatAmbiguousMappings as fmtAmbiguousMappings,\n  formatTypedMappings as fmtTypedMappings,\n} from '@aws-cdk/cloudformation-diff';\nimport type * as cxapi from '@aws-cdk/cx-api';\nimport type { StackSummary } from '@aws-sdk/client-cloudformation';\nimport { deserializeStructure } from '../../util';\nimport type { SdkProvider } from '../aws-auth/private';\nimport { Mode } from '../plugin';\nimport { StringWriteStream } from '../streams';\nimport type { CloudFormationStack } from './cloudformation';\nimport { ResourceMapping, ResourceLocation } from './cloudformation';\nimport { computeResourceDigests, hashObject } from './digest';\nimport { NeverExclude, type ExcludeList } from './exclude';\n\nexport * from './exclude';\n\n/**\n * Represents a set of possible movements of a resource from one location\n * to another. In the ideal case, there is only one source and only one\n * destination.\n */\nexport type ResourceMovement = [ResourceLocation[], ResourceLocation[]];\n\nexport class AmbiguityError extends Error {\n  constructor(public readonly movements: ResourceMovement[]) {\n    super('Ambiguous resource mappings');\n  }\n\n  public paths(): [string[], string[]][] {\n    return this.movements.map(([a, b]) => [convert(a), convert(b)]);\n\n    function convert(locations: ResourceLocation[]): string[] {\n      return locations.map((l) => l.toPath());\n    }\n  }\n}\n\nfunction groupByKey<A>(entries: [string, A][]): Record<string, A[]> {\n  const result: Record<string, A[]> = {};\n\n  for (const [hash, location] of entries) {\n    if (hash in result) {\n      result[hash].push(location);\n    } else {\n      result[hash] = [location];\n    }\n  }\n\n  return result;\n}\n\nexport function resourceMovements(before: CloudFormationStack[], after: CloudFormationStack[]): ResourceMovement[] {\n  return Object.values(\n    removeUnmovedResources(\n      zip(groupByKey(before.flatMap(resourceDigests)), groupByKey(after.flatMap(resourceDigests))),\n    ),\n  );\n}\n\nexport function ambiguousMovements(movements: ResourceMovement[]) {\n  // A movement is considered ambiguous if these two conditions are met:\n  //  1. Both sides have at least one element (otherwise, it's just addition or deletion)\n  //  2. At least one side has more than one element\n  return movements\n    .filter(([pre, post]) => pre.length > 0 && post.length > 0)\n    .filter(([pre, post]) => pre.length > 1 || post.length > 1);\n}\n\n/**\n * Converts a list of unambiguous resource movements into a list of resource mappings.\n *\n */\nexport function resourceMappings(\n  movements: ResourceMovement[],\n  stacks?: CloudFormationStack[],\n): ResourceMapping[] {\n  const stacksPredicate =\n    stacks == null\n      ? () => true\n      : (m: ResourceMapping) => {\n        // Any movement that involves one of the selected stacks (either moving from or to)\n        // is considered a candidate for refactoring.\n        const stackNames = [m.source.stack.stackName, m.destination.stack.stackName];\n        return stacks.some((stack) => stackNames.includes(stack.stackName));\n      };\n\n  return movements\n    .filter(([pre, post]) => pre.length === 1 && post.length === 1 && !pre[0].equalTo(post[0]))\n    .map(([pre, post]) => new ResourceMapping(pre[0], post[0]))\n    .filter(stacksPredicate);\n}\n\nfunction removeUnmovedResources(m: Record<string, ResourceMovement>): Record<string, ResourceMovement> {\n  const result: Record<string, ResourceMovement> = {};\n  for (const [hash, [before, after]] of Object.entries(m)) {\n    const common = before.filter((b) => after.some((a) => a.equalTo(b)));\n    result[hash] = [\n      before.filter((b) => !common.some((c) => b.equalTo(c))),\n      after.filter((a) => !common.some((c) => a.equalTo(c))),\n    ];\n  }\n\n  return result;\n}\n\n/**\n * For each hash, identifying a single resource, zip the two lists of locations,\n * producing a resource movement\n */\nfunction zip(\n  m1: Record<string, ResourceLocation[]>,\n  m2: Record<string, ResourceLocation[]>,\n): Record<string, ResourceMovement> {\n  const result: Record<string, ResourceMovement> = {};\n\n  for (const [hash, locations] of Object.entries(m1)) {\n    if (hash in m2) {\n      result[hash] = [locations, m2[hash]];\n    } else {\n      result[hash] = [locations, []];\n    }\n  }\n\n  for (const [hash, locations] of Object.entries(m2)) {\n    if (!(hash in m1)) {\n      result[hash] = [[], locations];\n    }\n  }\n\n  return result;\n}\n\n/**\n * Computes a list of pairs [digest, location] for each resource in the stack.\n */\nfunction resourceDigests(stack: CloudFormationStack): [string, ResourceLocation][] {\n  const digests = computeResourceDigests(stack.template);\n\n  return Object.entries(digests).map(([logicalId, digest]) => {\n    const location: ResourceLocation = new ResourceLocation(stack, logicalId);\n    return [digest, location];\n  });\n}\n\n/**\n * Compares the deployed state to the cloud assembly state, and finds all resources\n * that were moved from one location (stack + logical ID) to another. The comparison\n * is done per environment.\n */\nexport async function findResourceMovements(\n  stacks: CloudFormationStack[],\n  sdkProvider: SdkProvider,\n  exclude: ExcludeList = new NeverExclude(),\n): Promise<ResourceMovement[]> {\n  const stackGroups: Map<string, [CloudFormationStack[], CloudFormationStack[]]> = new Map();\n\n  // Group stacks by environment\n  for (const stack of stacks) {\n    const environment = stack.environment;\n    const key = hashObject(environment);\n    if (stackGroups.has(key)) {\n      stackGroups.get(key)![1].push(stack);\n    } else {\n      // The first time we see an environment, we need to fetch all stacks deployed to it.\n      const before = await getDeployedStacks(sdkProvider, environment);\n      stackGroups.set(key, [before, [stack]]);\n    }\n  }\n\n  const result: ResourceMovement[] = [];\n  for (const [_, [before, after]] of stackGroups) {\n    result.push(...resourceMovements(before, after));\n  }\n\n  return result.filter(mov => {\n    const after = mov[1];\n    return after.every(l => !exclude.isExcluded(l));\n  });\n}\n\nasync function getDeployedStacks(\n  sdkProvider: SdkProvider,\n  environment: cxapi.Environment,\n): Promise<CloudFormationStack[]> {\n  const cfn = (await sdkProvider.forEnvironment(environment, Mode.ForReading)).sdk.cloudFormation();\n\n  const summaries = await cfn.paginatedListStacks({\n    StackStatusFilter: [\n      'CREATE_COMPLETE',\n      'UPDATE_COMPLETE',\n      'UPDATE_ROLLBACK_COMPLETE',\n      'IMPORT_COMPLETE',\n      'ROLLBACK_COMPLETE',\n    ],\n  });\n\n  const normalize = async (summary: StackSummary) => {\n    const templateCommandOutput = await cfn.getTemplate({ StackName: summary.StackName! });\n    const template = deserializeStructure(templateCommandOutput.TemplateBody ?? '{}');\n    return {\n      environment,\n      stackName: summary.StackName!,\n      template,\n    };\n  };\n\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  return Promise.all(summaries.map(normalize));\n}\n\nexport function formatTypedMappings(mappings: TypedMapping[]): string {\n  const stream = new StringWriteStream();\n  fmtTypedMappings(stream, mappings);\n  return stream.toString();\n}\n\nexport function formatAmbiguousMappings(paths: [string[], string[]][]): string {\n  const stream = new StringWriteStream();\n  fmtAmbiguousMappings(stream, paths);\n  return stream.toString();\n}\n"]}