aws-cdk 2.14.0 → 2.15.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.
Files changed (35) hide show
  1. package/README.md +2 -1
  2. package/build-info.json +2 -2
  3. package/does-not-exist.json +1 -0
  4. package/lib/api/cloudformation-deployments.d.ts +1 -6
  5. package/lib/api/cloudformation-deployments.js +6 -72
  6. package/lib/api/evaluate-cloudformation-template.d.ts +9 -2
  7. package/lib/api/evaluate-cloudformation-template.js +14 -2
  8. package/lib/api/hotswap-deployments.js +34 -7
  9. package/lib/api/logs/find-cloudwatch-logs.js +2 -2
  10. package/lib/api/nested-stack-helpers.d.ts +28 -0
  11. package/lib/api/nested-stack-helpers.js +97 -0
  12. package/lib/cli.js +2 -2
  13. package/lib/index.js +1139 -389
  14. package/lib/notices.js +16 -7
  15. package/package.json +9 -9
  16. package/test/api/hotswap/hotswap-deployments.test.js +38 -2
  17. package/test/api/hotswap/hotswap-test-setup.d.ts +6 -4
  18. package/test/api/hotswap/hotswap-test-setup.js +46 -6
  19. package/test/api/hotswap/nested-stacks-hotswap.test.d.ts +1 -0
  20. package/test/api/hotswap/nested-stacks-hotswap.test.js +840 -0
  21. package/test/api/hotswap/state-machine-hotswap-deployments.test.js +2 -2
  22. package/test/integ/cli/cli.integtest.js +2 -2
  23. package/test/nested-stack-templates/one-lambda-one-stack-stack-with-asset-parameters.nested.template.json +52 -0
  24. package/test/nested-stack-templates/one-lambda-one-stack-stack.nested.template.json +26 -0
  25. package/test/nested-stack-templates/one-lambda-stack-with-asset-parameters.nested.template.json +29 -0
  26. package/test/nested-stack-templates/one-lambda-stack.nested.template.json +17 -0
  27. package/test/nested-stack-templates/one-lambda-version-stack.nested.template.json +20 -0
  28. package/test/{diff-nested-stacks-templates → nested-stack-templates}/one-output-one-param-stack.nested.template.json +0 -0
  29. package/test/{diff-nested-stacks-templates → nested-stack-templates}/one-resource-one-stack-stack.nested.template.json +0 -0
  30. package/test/{diff-nested-stacks-templates → nested-stack-templates}/one-resource-stack.nested.template.json +0 -0
  31. package/test/{diff-nested-stacks-templates → nested-stack-templates}/one-resource-two-stacks-stack.nested.template.json +0 -0
  32. package/test/nested-stack-templates/one-unnamed-lambda-stack.nested.template.json +16 -0
  33. package/test/nested-stack-templates/one-unnamed-lambda-two-stacks-stack.nested.template.json +34 -0
  34. package/test/notices.test.js +26 -3
  35. package/test/util.js +2 -2
package/lib/notices.js CHANGED
@@ -60,13 +60,18 @@ function formatNotices(data) {
60
60
  exports.formatNotices = formatNotices;
61
61
  class WebsiteNoticeDataSource {
62
62
  fetch() {
63
+ const timeout = 3000;
63
64
  return new Promise((resolve) => {
64
65
  try {
65
- const req = https.get('https://cli.cdk.dev-tools.aws.dev/notices.json', res => {
66
+ const req = https.get('https://cli.cdk.dev-tools.aws.dev/notices.json', { timeout }, res => {
67
+ const startTime = Date.now();
66
68
  if (res.statusCode === 200) {
67
69
  res.setEncoding('utf8');
68
70
  let rawData = '';
69
71
  res.on('data', (chunk) => {
72
+ if (Date.now() - startTime > timeout) {
73
+ resolve([]);
74
+ }
70
75
  rawData += chunk;
71
76
  });
72
77
  res.on('end', () => {
@@ -93,6 +98,7 @@ class WebsiteNoticeDataSource {
93
98
  logging_1.debug(`Error on request: ${e}`);
94
99
  resolve([]);
95
100
  });
101
+ req.on('timeout', _ => resolve([]));
96
102
  }
97
103
  catch (e) {
98
104
  logging_1.debug(`HTTPS 'get' call threw an error: ${e}`);
@@ -127,15 +133,18 @@ class CachedDataSource {
127
133
  }
128
134
  }
129
135
  async load() {
136
+ const defaultValue = {
137
+ expiration: 0,
138
+ notices: [],
139
+ };
130
140
  try {
131
- return await fs.readJSON(this.fileName);
141
+ return fs.existsSync(this.fileName)
142
+ ? await fs.readJSON(this.fileName)
143
+ : defaultValue;
132
144
  }
133
145
  catch (e) {
134
146
  logging_1.debug(`Failed to load notices from cache: ${e}`);
135
- return {
136
- expiration: 0,
137
- notices: [],
138
- };
147
+ return defaultValue;
139
148
  }
140
149
  }
141
150
  async save(cached) {
@@ -271,4 +280,4 @@ function some(node, predicate) {
271
280
  return false;
272
281
  }
273
282
  }
274
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"notices.js","sourceRoot":"","sources":["notices.ts"],"names":[],"mappings":"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","sourcesContent":["import * as https from 'https';\nimport * as path from 'path';\nimport * as fs from 'fs-extra';\nimport * as semver from 'semver';\nimport { debug, print } from './logging';\nimport { flatMap } from './util';\nimport { cdkCacheDir } from './util/directories';\nimport { versionNumber } from './version';\n\nconst CACHE_FILE_PATH = path.join(cdkCacheDir(), 'notices.json');\n\nexport interface DisplayNoticesProps {\n  /**\n   * The cloud assembly directory. Usually 'cdk.out'.\n   */\n  readonly outdir: string;\n\n  /**\n   * Issue numbers of notices that have been acknowledged by a user\n   * of the current CDK repository. These notices will be skipped.\n   */\n  readonly acknowledgedIssueNumbers: number[];\n\n  /**\n   * Whether cached notices should be ignored. Setting this property\n   * to true will force the CLI to download fresh data\n   *\n   * @default false\n   */\n  readonly ignoreCache?: boolean;\n}\n\nexport async function refreshNotices() {\n  const dataSource = dataSourceReference(false);\n  return dataSource.fetch();\n}\n\nexport async function displayNotices(props: DisplayNoticesProps) {\n  const dataSource = dataSourceReference(props.ignoreCache ?? false);\n  print(await generateMessage(dataSource, props));\n  return 0;\n}\n\nexport async function generateMessage(dataSource: NoticeDataSource, props: DisplayNoticesProps) {\n  const data = await dataSource.fetch();\n  const individualMessages = formatNotices(filterNotices(data, {\n    outdir: props.outdir,\n    acknowledgedIssueNumbers: new Set(props.acknowledgedIssueNumbers),\n  }));\n\n  if (individualMessages.length > 0) {\n    return finalMessage(individualMessages, data[0].issueNumber);\n  }\n  return '';\n}\n\nfunction dataSourceReference(ignoreCache: boolean): NoticeDataSource {\n  return new CachedDataSource(CACHE_FILE_PATH, new WebsiteNoticeDataSource(), ignoreCache);\n}\n\nfunction finalMessage(individualMessages: string[], exampleNumber: number): string {\n  return [\n    '\\nNOTICES',\n    ...individualMessages,\n    `If you don’t want to see a notice anymore, use \"cdk acknowledge <id>\". For example, \"cdk acknowledge ${exampleNumber}\".`,\n  ].join('\\n\\n');\n}\n\nexport interface FilterNoticeOptions {\n  outdir?: string,\n  cliVersion?: string,\n  frameworkVersion?: string,\n  acknowledgedIssueNumbers?: Set<number>,\n}\n\nexport function filterNotices(data: Notice[], options: FilterNoticeOptions): Notice[] {\n  const filter = new NoticeFilter({\n    cliVersion: options.cliVersion ?? versionNumber(),\n    acknowledgedIssueNumbers: options.acknowledgedIssueNumbers ?? new Set(),\n    tree: loadTree(options.outdir ?? 'cdk.out').tree,\n  });\n  return data.filter(notice => filter.apply(notice));\n}\n\nexport function formatNotices(data: Notice[]): string[] {\n  return data.map(formatNotice);\n}\n\nexport interface Component {\n  name: string;\n  version: string;\n}\n\nexport interface Notice {\n  title: string;\n  issueNumber: number;\n  overview: string;\n  components: Component[];\n  schemaVersion: string;\n}\n\nexport interface NoticeDataSource {\n  fetch(): Promise<Notice[]>,\n}\n\nexport class WebsiteNoticeDataSource implements NoticeDataSource {\n  fetch(): Promise<Notice[]> {\n    return new Promise((resolve) => {\n      try {\n        const req = https.get('https://cli.cdk.dev-tools.aws.dev/notices.json', res => {\n          if (res.statusCode === 200) {\n            res.setEncoding('utf8');\n            let rawData = '';\n            res.on('data', (chunk) => {\n              rawData += chunk;\n            });\n            res.on('end', () => {\n              try {\n                const data = JSON.parse(rawData).notices as Notice[];\n                resolve(data ?? []);\n              } catch (e) {\n                debug(`Failed to parse notices: ${e}`);\n                resolve([]);\n              }\n            });\n            res.on('error', e => {\n              debug(`Failed to fetch notices: ${e}`);\n              resolve([]);\n            });\n          } else {\n            debug(`Failed to fetch notices. Status code: ${res.statusCode}`);\n            resolve([]);\n          }\n        });\n        req.on('error', e => {\n          debug(`Error on request: ${e}`);\n          resolve([]);\n        });\n      } catch (e) {\n        debug(`HTTPS 'get' call threw an error: ${e}`);\n        resolve([]);\n      }\n    });\n  }\n}\n\ninterface CachedNotices {\n  expiration: number,\n  notices: Notice[],\n}\n\nconst TIME_TO_LIVE = 60 * 60 * 1000; // 1 hour\n\nexport class CachedDataSource implements NoticeDataSource {\n  constructor(\n    private readonly fileName: string,\n    private readonly dataSource: NoticeDataSource,\n    private readonly skipCache?: boolean) {\n  }\n\n  async fetch(): Promise<Notice[]> {\n    const cachedData = await this.load();\n    const data = cachedData.notices;\n    const expiration = cachedData.expiration ?? 0;\n\n    if (Date.now() > expiration || this.skipCache) {\n      const freshData = {\n        expiration: Date.now() + TIME_TO_LIVE,\n        notices: await this.dataSource.fetch(),\n      };\n      await this.save(freshData);\n      return freshData.notices;\n    } else {\n      return data;\n    }\n  }\n\n  private async load(): Promise<CachedNotices> {\n    try {\n      return await fs.readJSON(this.fileName) as CachedNotices;\n    } catch (e) {\n      debug(`Failed to load notices from cache: ${e}`);\n      return {\n        expiration: 0,\n        notices: [],\n      };\n    }\n  }\n\n  private async save(cached: CachedNotices): Promise<void> {\n    try {\n      await fs.writeJSON(this.fileName, cached);\n    } catch (e) {\n      debug(`Failed to store notices in the cache: ${e}`);\n    }\n  }\n}\n\nexport interface NoticeFilterProps {\n  cliVersion: string,\n  acknowledgedIssueNumbers: Set<number>,\n  tree: ConstructTreeNode,\n}\n\nexport class NoticeFilter {\n  private readonly acknowledgedIssueNumbers: Set<number>;\n\n  constructor(private readonly props: NoticeFilterProps) {\n    this.acknowledgedIssueNumbers = props.acknowledgedIssueNumbers;\n  }\n\n  /**\n   * Returns true iff we should show this notice.\n   */\n  apply(notice: Notice): boolean {\n    if (this.acknowledgedIssueNumbers.has(notice.issueNumber)) {\n      return false;\n    }\n\n    return this.applyVersion(notice, 'cli', this.props.cliVersion) ||\n      match(resolveAliases(notice.components), this.props.tree);\n  }\n\n  /**\n   * Returns true iff we should show the notice.\n   */\n  private applyVersion(notice: Notice, name: string, compareToVersion: string | undefined) {\n    if (compareToVersion === undefined) { return false; }\n\n    const affectedComponent = notice.components.find(component => component.name === name);\n    const affectedRange = affectedComponent?.version;\n    return affectedRange != null && semver.satisfies(compareToVersion, affectedRange);\n  }\n}\n\n/**\n * Some component names are aliases to actual component names. For example \"framework\"\n * is an alias for either the core library (v1) or the whole CDK library (v2).\n *\n * This function converts all aliases to their actual counterpart names, to be used to\n * match against the construct tree.\n *\n * @param components a list of components. Components whose name is an alias will be\n * transformed and all others will be left intact.\n */\nfunction resolveAliases(components: Component[]): Component[] {\n  return flatMap(components, component => {\n    if (component.name === 'framework') {\n      return [{\n        name: '@aws-cdk/core.',\n        version: component.version,\n      }, {\n        name: 'aws-cdk-lib.',\n        version: component.version,\n      }];\n    } else {\n      return [component];\n    }\n  });\n}\n\nfunction formatNotice(notice: Notice): string {\n  const componentsValue = notice.components.map(c => `${c.name}: ${c.version}`).join(', ');\n  return [\n    `${notice.issueNumber}\\t${notice.title}`,\n    formatOverview(notice.overview),\n    `\\tAffected versions: ${componentsValue}`,\n    `\\tMore information at: https://github.com/aws/aws-cdk/issues/${notice.issueNumber}`,\n  ].join('\\n\\n') + '\\n';\n}\n\nfunction formatOverview(text: string) {\n  const wrap = (s: string) => s.replace(/(?![^\\n]{1,60}$)([^\\n]{1,60})\\s/g, '$1\\n');\n\n  const heading = 'Overview: ';\n  const separator = `\\n\\t${' '.repeat(heading.length)}`;\n  const content = wrap(text)\n    .split('\\n')\n    .join(separator);\n\n  return '\\t' + heading + content;\n}\n\n/**\n * Whether any component in the tree matches any component in the query.\n * A match happens when:\n *\n * 1. The version of the node matches the version in the query, interpreted\n * as a semver range.\n *\n * 2. The name in the query is a prefix of the node name when the query ends in '.',\n * or the two names are exactly the same, otherwise.\n */\nfunction match(query: Component[], tree: ConstructTreeNode): boolean {\n  return some(tree, node => {\n    return query.some(component =>\n      compareNames(component.name, node.constructInfo?.fqn) &&\n      compareVersions(component.version, node.constructInfo?.version));\n  });\n\n  function compareNames(pattern: string, target: string | undefined): boolean {\n    if (target == null) { return false; }\n    return pattern.endsWith('.') ? target.startsWith(pattern) : pattern === target;\n  }\n\n  function compareVersions(pattern: string, target: string | undefined): boolean {\n    return semver.satisfies(target ?? '', pattern);\n  }\n}\n\nfunction loadTree(outdir: string) {\n  try {\n    return fs.readJSONSync(path.join(outdir, 'tree.json'));\n  } catch (e) {\n    debug(`Failed to get tree.json file: ${e}`);\n    return {};\n  }\n}\n\n/**\n * Source information on a construct (class fqn and version)\n */\ninterface ConstructInfo {\n  readonly fqn: string;\n  readonly version: string;\n}\n\n/**\n * A node in the construct tree.\n * @internal\n */\ninterface ConstructTreeNode {\n  readonly id: string;\n  readonly path: string;\n  readonly children?: { [key: string]: ConstructTreeNode };\n  readonly attributes?: { [key: string]: any };\n\n  /**\n   * Information on the construct class that led to this node, if available\n   */\n  readonly constructInfo?: ConstructInfo;\n}\n\nfunction some(node: ConstructTreeNode, predicate: (n: ConstructTreeNode) => boolean): boolean {\n  return node != null && (predicate(node) || findInChildren());\n\n  function findInChildren(): boolean {\n    if (node.children == null) { return false; }\n\n    for (const name in node.children) {\n      if (some(node.children[name], predicate)) {\n        return true;\n      }\n    }\n    return false;\n  }\n}\n"]}
283
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"notices.js","sourceRoot":"","sources":["notices.ts"],"names":[],"mappings":";;;AAAA,+BAA+B;AAC/B,6BAA6B;AAC7B,+BAA+B;AAC/B,iCAAiC;AACjC,uCAAyC;AACzC,iCAAiC;AACjC,oDAAiD;AACjD,uCAA0C;AAE1C,MAAM,eAAe,GAAG,IAAI,CAAC,IAAI,CAAC,yBAAW,EAAE,EAAE,cAAc,CAAC,CAAC;AAuB1D,KAAK,UAAU,cAAc;IAClC,MAAM,UAAU,GAAG,mBAAmB,CAAC,KAAK,CAAC,CAAC;IAC9C,OAAO,UAAU,CAAC,KAAK,EAAE,CAAC;AAC5B,CAAC;AAHD,wCAGC;AAEM,KAAK,UAAU,cAAc,CAAC,KAA0B;;IAC7D,MAAM,UAAU,GAAG,mBAAmB,OAAC,KAAK,CAAC,WAAW,mCAAI,KAAK,CAAC,CAAC;IACnE,eAAK,CAAC,MAAM,eAAe,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC;IAChD,OAAO,CAAC,CAAC;AACX,CAAC;AAJD,wCAIC;AAEM,KAAK,UAAU,eAAe,CAAC,UAA4B,EAAE,KAA0B;IAC5F,MAAM,IAAI,GAAG,MAAM,UAAU,CAAC,KAAK,EAAE,CAAC;IACtC,MAAM,kBAAkB,GAAG,aAAa,CAAC,aAAa,CAAC,IAAI,EAAE;QAC3D,MAAM,EAAE,KAAK,CAAC,MAAM;QACpB,wBAAwB,EAAE,IAAI,GAAG,CAAC,KAAK,CAAC,wBAAwB,CAAC;KAClE,CAAC,CAAC,CAAC;IAEJ,IAAI,kBAAkB,CAAC,MAAM,GAAG,CAAC,EAAE;QACjC,OAAO,YAAY,CAAC,kBAAkB,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC;KAC9D;IACD,OAAO,EAAE,CAAC;AACZ,CAAC;AAXD,0CAWC;AAED,SAAS,mBAAmB,CAAC,WAAoB;IAC/C,OAAO,IAAI,gBAAgB,CAAC,eAAe,EAAE,IAAI,uBAAuB,EAAE,EAAE,WAAW,CAAC,CAAC;AAC3F,CAAC;AAED,SAAS,YAAY,CAAC,kBAA4B,EAAE,aAAqB;IACvE,OAAO;QACL,WAAW;QACX,GAAG,kBAAkB;QACrB,wGAAwG,aAAa,IAAI;KAC1H,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;AACjB,CAAC;AASD,SAAgB,aAAa,CAAC,IAAc,EAAE,OAA4B;;IACxE,MAAM,MAAM,GAAG,IAAI,YAAY,CAAC;QAC9B,UAAU,QAAE,OAAO,CAAC,UAAU,mCAAI,uBAAa,EAAE;QACjD,wBAAwB,QAAE,OAAO,CAAC,wBAAwB,mCAAI,IAAI,GAAG,EAAE;QACvE,IAAI,EAAE,QAAQ,OAAC,OAAO,CAAC,MAAM,mCAAI,SAAS,CAAC,CAAC,IAAI;KACjD,CAAC,CAAC;IACH,OAAO,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC;AACrD,CAAC;AAPD,sCAOC;AAED,SAAgB,aAAa,CAAC,IAAc;IAC1C,OAAO,IAAI,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;AAChC,CAAC;AAFD,sCAEC;AAmBD,MAAa,uBAAuB;IAClC,KAAK;QACH,MAAM,OAAO,GAAG,IAAI,CAAC;QAErB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7B,IAAI;gBACF,MAAM,GAAG,GAAG,KAAK,CAAC,GAAG,CAAC,gDAAgD,EACpE,EAAE,OAAO,EAAE,EACX,GAAG,CAAC,EAAE;oBACJ,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;oBAC7B,IAAI,GAAG,CAAC,UAAU,KAAK,GAAG,EAAE;wBAC1B,GAAG,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;wBACxB,IAAI,OAAO,GAAG,EAAE,CAAC;wBACjB,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,KAAK,EAAE,EAAE;4BACvB,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,GAAG,OAAO,EAAE;gCACpC,OAAO,CAAC,EAAE,CAAC,CAAC;6BACb;4BACD,OAAO,IAAI,KAAK,CAAC;wBACnB,CAAC,CAAC,CAAC;wBACH,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE;4BACjB,IAAI;gCACF,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,OAAmB,CAAC;gCACrD,OAAO,CAAC,IAAI,aAAJ,IAAI,cAAJ,IAAI,GAAI,EAAE,CAAC,CAAC;6BACrB;4BAAC,OAAO,CAAC,EAAE;gCACV,eAAK,CAAC,4BAA4B,CAAC,EAAE,CAAC,CAAC;gCACvC,OAAO,CAAC,EAAE,CAAC,CAAC;6BACb;wBACH,CAAC,CAAC,CAAC;wBACH,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC,EAAE;4BAClB,eAAK,CAAC,4BAA4B,CAAC,EAAE,CAAC,CAAC;4BACvC,OAAO,CAAC,EAAE,CAAC,CAAC;wBACd,CAAC,CAAC,CAAC;qBACJ;yBAAM;wBACL,eAAK,CAAC,yCAAyC,GAAG,CAAC,UAAU,EAAE,CAAC,CAAC;wBACjE,OAAO,CAAC,EAAE,CAAC,CAAC;qBACb;gBACH,CAAC,CAAC,CAAC;gBACL,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC,EAAE;oBAClB,eAAK,CAAC,qBAAqB,CAAC,EAAE,CAAC,CAAC;oBAChC,OAAO,CAAC,EAAE,CAAC,CAAC;gBACd,CAAC,CAAC,CAAC;gBACH,GAAG,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,CAAC,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC;aACrC;YAAC,OAAO,CAAC,EAAE;gBACV,eAAK,CAAC,oCAAoC,CAAC,EAAE,CAAC,CAAC;gBAC/C,OAAO,CAAC,EAAE,CAAC,CAAC;aACb;QACH,CAAC,CAAC,CAAC;IACL,CAAC;CACF;AAhDD,0DAgDC;AAOD,MAAM,YAAY,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC,SAAS;AAE9C,MAAa,gBAAgB;IAC3B,YACmB,QAAgB,EAChB,UAA4B,EAC5B,SAAmB;QAFnB,aAAQ,GAAR,QAAQ,CAAQ;QAChB,eAAU,GAAV,UAAU,CAAkB;QAC5B,cAAS,GAAT,SAAS,CAAU;IACtC,CAAC;IAED,KAAK,CAAC,KAAK;;QACT,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC;QACrC,MAAM,IAAI,GAAG,UAAU,CAAC,OAAO,CAAC;QAChC,MAAM,UAAU,SAAG,UAAU,CAAC,UAAU,mCAAI,CAAC,CAAC;QAE9C,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,UAAU,IAAI,IAAI,CAAC,SAAS,EAAE;YAC7C,MAAM,SAAS,GAAG;gBAChB,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,YAAY;gBACrC,OAAO,EAAE,MAAM,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE;aACvC,CAAC;YACF,MAAM,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YAC3B,OAAO,SAAS,CAAC,OAAO,CAAC;SAC1B;aAAM;YACL,OAAO,IAAI,CAAC;SACb;IACH,CAAC;IAEO,KAAK,CAAC,IAAI;QAChB,MAAM,YAAY,GAAG;YACnB,UAAU,EAAE,CAAC;YACb,OAAO,EAAE,EAAE;SACZ,CAAC;QAEF,IAAI;YACF,OAAO,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC;gBACjC,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAkB;gBACnD,CAAC,CAAC,YAAY,CAAC;SAClB;QAAC,OAAO,CAAC,EAAE;YACV,eAAK,CAAC,sCAAsC,CAAC,EAAE,CAAC,CAAC;YACjD,OAAO,YAAY,CAAC;SACrB;IACH,CAAC;IAEO,KAAK,CAAC,IAAI,CAAC,MAAqB;QACtC,IAAI;YACF,MAAM,EAAE,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;SAC3C;QAAC,OAAO,CAAC,EAAE;YACV,eAAK,CAAC,yCAAyC,CAAC,EAAE,CAAC,CAAC;SACrD;IACH,CAAC;CACF;AA/CD,4CA+CC;AAQD,MAAa,YAAY;IAGvB,YAA6B,KAAwB;QAAxB,UAAK,GAAL,KAAK,CAAmB;QACnD,IAAI,CAAC,wBAAwB,GAAG,KAAK,CAAC,wBAAwB,CAAC;IACjE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAc;QAClB,IAAI,IAAI,CAAC,wBAAwB,CAAC,GAAG,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE;YACzD,OAAO,KAAK,CAAC;SACd;QAED,OAAO,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC;YAC5D,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC,UAAU,CAAC,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IAC9D,CAAC;IAED;;OAEG;IACK,YAAY,CAAC,MAAc,EAAE,IAAY,EAAE,gBAAoC;QACrF,IAAI,gBAAgB,KAAK,SAAS,EAAE;YAAE,OAAO,KAAK,CAAC;SAAE;QAErD,MAAM,iBAAiB,GAAG,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,SAAS,CAAC,IAAI,KAAK,IAAI,CAAC,CAAC;QACvF,MAAM,aAAa,GAAG,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,OAAO,CAAC;QACjD,OAAO,aAAa,IAAI,IAAI,IAAI,MAAM,CAAC,SAAS,CAAC,gBAAgB,EAAE,aAAa,CAAC,CAAC;IACpF,CAAC;CACF;AA7BD,oCA6BC;AAED;;;;;;;;;GASG;AACH,SAAS,cAAc,CAAC,UAAuB;IAC7C,OAAO,cAAO,CAAC,UAAU,EAAE,SAAS,CAAC,EAAE;QACrC,IAAI,SAAS,CAAC,IAAI,KAAK,WAAW,EAAE;YAClC,OAAO,CAAC;oBACN,IAAI,EAAE,gBAAgB;oBACtB,OAAO,EAAE,SAAS,CAAC,OAAO;iBAC3B,EAAE;oBACD,IAAI,EAAE,cAAc;oBACpB,OAAO,EAAE,SAAS,CAAC,OAAO;iBAC3B,CAAC,CAAC;SACJ;aAAM;YACL,OAAO,CAAC,SAAS,CAAC,CAAC;SACpB;IACH,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAS,YAAY,CAAC,MAAc;IAClC,MAAM,eAAe,GAAG,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,IAAI,KAAK,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACzF,OAAO;QACL,GAAG,MAAM,CAAC,WAAW,KAAK,MAAM,CAAC,KAAK,EAAE;QACxC,cAAc,CAAC,MAAM,CAAC,QAAQ,CAAC;QAC/B,wBAAwB,eAAe,EAAE;QACzC,gEAAgE,MAAM,CAAC,WAAW,EAAE;KACrF,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC;AACxB,CAAC;AAED,SAAS,cAAc,CAAC,IAAY;IAClC,MAAM,IAAI,GAAG,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,kCAAkC,EAAE,MAAM,CAAC,CAAC;IAElF,MAAM,OAAO,GAAG,YAAY,CAAC;IAC7B,MAAM,SAAS,GAAG,OAAO,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;IACtD,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC;SACvB,KAAK,CAAC,IAAI,CAAC;SACX,IAAI,CAAC,SAAS,CAAC,CAAC;IAEnB,OAAO,IAAI,GAAG,OAAO,GAAG,OAAO,CAAC;AAClC,CAAC;AAED;;;;;;;;;GASG;AACH,SAAS,KAAK,CAAC,KAAkB,EAAE,IAAuB;IACxD,OAAO,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,EAAE;QACvB,OAAO,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE;;YAC5B,OAAA,YAAY,CAAC,SAAS,CAAC,IAAI,QAAE,IAAI,CAAC,aAAa,0CAAE,GAAG,CAAC;gBACrD,eAAe,CAAC,SAAS,CAAC,OAAO,QAAE,IAAI,CAAC,aAAa,0CAAE,OAAO,CAAC,CAAA;SAAA,CAAC,CAAC;IACrE,CAAC,CAAC,CAAC;IAEH,SAAS,YAAY,CAAC,OAAe,EAAE,MAA0B;QAC/D,IAAI,MAAM,IAAI,IAAI,EAAE;YAAE,OAAO,KAAK,CAAC;SAAE;QACrC,OAAO,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,KAAK,MAAM,CAAC;IACjF,CAAC;IAED,SAAS,eAAe,CAAC,OAAe,EAAE,MAA0B;QAClE,OAAO,MAAM,CAAC,SAAS,CAAC,MAAM,aAAN,MAAM,cAAN,MAAM,GAAI,EAAE,EAAE,OAAO,CAAC,CAAC;IACjD,CAAC;AACH,CAAC;AAED,SAAS,QAAQ,CAAC,MAAc;IAC9B,IAAI;QACF,OAAO,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,CAAC;KACxD;IAAC,OAAO,CAAC,EAAE;QACV,eAAK,CAAC,iCAAiC,CAAC,EAAE,CAAC,CAAC;QAC5C,OAAO,EAAE,CAAC;KACX;AACH,CAAC;AA0BD,SAAS,IAAI,CAAC,IAAuB,EAAE,SAA4C;IACjF,OAAO,IAAI,IAAI,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,cAAc,EAAE,CAAC,CAAC;IAE7D,SAAS,cAAc;QACrB,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,EAAE;YAAE,OAAO,KAAK,CAAC;SAAE;QAE5C,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,QAAQ,EAAE;YAChC,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,SAAS,CAAC,EAAE;gBACxC,OAAO,IAAI,CAAC;aACb;SACF;QACD,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC","sourcesContent":["import * as https from 'https';\nimport * as path from 'path';\nimport * as fs from 'fs-extra';\nimport * as semver from 'semver';\nimport { debug, print } from './logging';\nimport { flatMap } from './util';\nimport { cdkCacheDir } from './util/directories';\nimport { versionNumber } from './version';\n\nconst CACHE_FILE_PATH = path.join(cdkCacheDir(), 'notices.json');\n\nexport interface DisplayNoticesProps {\n  /**\n   * The cloud assembly directory. Usually 'cdk.out'.\n   */\n  readonly outdir: string;\n\n  /**\n   * Issue numbers of notices that have been acknowledged by a user\n   * of the current CDK repository. These notices will be skipped.\n   */\n  readonly acknowledgedIssueNumbers: number[];\n\n  /**\n   * Whether cached notices should be ignored. Setting this property\n   * to true will force the CLI to download fresh data\n   *\n   * @default false\n   */\n  readonly ignoreCache?: boolean;\n}\n\nexport async function refreshNotices() {\n  const dataSource = dataSourceReference(false);\n  return dataSource.fetch();\n}\n\nexport async function displayNotices(props: DisplayNoticesProps) {\n  const dataSource = dataSourceReference(props.ignoreCache ?? false);\n  print(await generateMessage(dataSource, props));\n  return 0;\n}\n\nexport async function generateMessage(dataSource: NoticeDataSource, props: DisplayNoticesProps) {\n  const data = await dataSource.fetch();\n  const individualMessages = formatNotices(filterNotices(data, {\n    outdir: props.outdir,\n    acknowledgedIssueNumbers: new Set(props.acknowledgedIssueNumbers),\n  }));\n\n  if (individualMessages.length > 0) {\n    return finalMessage(individualMessages, data[0].issueNumber);\n  }\n  return '';\n}\n\nfunction dataSourceReference(ignoreCache: boolean): NoticeDataSource {\n  return new CachedDataSource(CACHE_FILE_PATH, new WebsiteNoticeDataSource(), ignoreCache);\n}\n\nfunction finalMessage(individualMessages: string[], exampleNumber: number): string {\n  return [\n    '\\nNOTICES',\n    ...individualMessages,\n    `If you don’t want to see a notice anymore, use \"cdk acknowledge <id>\". For example, \"cdk acknowledge ${exampleNumber}\".`,\n  ].join('\\n\\n');\n}\n\nexport interface FilterNoticeOptions {\n  outdir?: string,\n  cliVersion?: string,\n  frameworkVersion?: string,\n  acknowledgedIssueNumbers?: Set<number>,\n}\n\nexport function filterNotices(data: Notice[], options: FilterNoticeOptions): Notice[] {\n  const filter = new NoticeFilter({\n    cliVersion: options.cliVersion ?? versionNumber(),\n    acknowledgedIssueNumbers: options.acknowledgedIssueNumbers ?? new Set(),\n    tree: loadTree(options.outdir ?? 'cdk.out').tree,\n  });\n  return data.filter(notice => filter.apply(notice));\n}\n\nexport function formatNotices(data: Notice[]): string[] {\n  return data.map(formatNotice);\n}\n\nexport interface Component {\n  name: string;\n  version: string;\n}\n\nexport interface Notice {\n  title: string;\n  issueNumber: number;\n  overview: string;\n  components: Component[];\n  schemaVersion: string;\n}\n\nexport interface NoticeDataSource {\n  fetch(): Promise<Notice[]>,\n}\n\nexport class WebsiteNoticeDataSource implements NoticeDataSource {\n  fetch(): Promise<Notice[]> {\n    const timeout = 3000;\n\n    return new Promise((resolve) => {\n      try {\n        const req = https.get('https://cli.cdk.dev-tools.aws.dev/notices.json',\n          { timeout },\n          res => {\n            const startTime = Date.now();\n            if (res.statusCode === 200) {\n              res.setEncoding('utf8');\n              let rawData = '';\n              res.on('data', (chunk) => {\n                if (Date.now() - startTime > timeout) {\n                  resolve([]);\n                }\n                rawData += chunk;\n              });\n              res.on('end', () => {\n                try {\n                  const data = JSON.parse(rawData).notices as Notice[];\n                  resolve(data ?? []);\n                } catch (e) {\n                  debug(`Failed to parse notices: ${e}`);\n                  resolve([]);\n                }\n              });\n              res.on('error', e => {\n                debug(`Failed to fetch notices: ${e}`);\n                resolve([]);\n              });\n            } else {\n              debug(`Failed to fetch notices. Status code: ${res.statusCode}`);\n              resolve([]);\n            }\n          });\n        req.on('error', e => {\n          debug(`Error on request: ${e}`);\n          resolve([]);\n        });\n        req.on('timeout', _ => resolve([]));\n      } catch (e) {\n        debug(`HTTPS 'get' call threw an error: ${e}`);\n        resolve([]);\n      }\n    });\n  }\n}\n\ninterface CachedNotices {\n  expiration: number,\n  notices: Notice[],\n}\n\nconst TIME_TO_LIVE = 60 * 60 * 1000; // 1 hour\n\nexport class CachedDataSource implements NoticeDataSource {\n  constructor(\n    private readonly fileName: string,\n    private readonly dataSource: NoticeDataSource,\n    private readonly skipCache?: boolean) {\n  }\n\n  async fetch(): Promise<Notice[]> {\n    const cachedData = await this.load();\n    const data = cachedData.notices;\n    const expiration = cachedData.expiration ?? 0;\n\n    if (Date.now() > expiration || this.skipCache) {\n      const freshData = {\n        expiration: Date.now() + TIME_TO_LIVE,\n        notices: await this.dataSource.fetch(),\n      };\n      await this.save(freshData);\n      return freshData.notices;\n    } else {\n      return data;\n    }\n  }\n\n  private async load(): Promise<CachedNotices> {\n    const defaultValue = {\n      expiration: 0,\n      notices: [],\n    };\n\n    try {\n      return fs.existsSync(this.fileName)\n        ? await fs.readJSON(this.fileName) as CachedNotices\n        : defaultValue;\n    } catch (e) {\n      debug(`Failed to load notices from cache: ${e}`);\n      return defaultValue;\n    }\n  }\n\n  private async save(cached: CachedNotices): Promise<void> {\n    try {\n      await fs.writeJSON(this.fileName, cached);\n    } catch (e) {\n      debug(`Failed to store notices in the cache: ${e}`);\n    }\n  }\n}\n\nexport interface NoticeFilterProps {\n  cliVersion: string,\n  acknowledgedIssueNumbers: Set<number>,\n  tree: ConstructTreeNode,\n}\n\nexport class NoticeFilter {\n  private readonly acknowledgedIssueNumbers: Set<number>;\n\n  constructor(private readonly props: NoticeFilterProps) {\n    this.acknowledgedIssueNumbers = props.acknowledgedIssueNumbers;\n  }\n\n  /**\n   * Returns true iff we should show this notice.\n   */\n  apply(notice: Notice): boolean {\n    if (this.acknowledgedIssueNumbers.has(notice.issueNumber)) {\n      return false;\n    }\n\n    return this.applyVersion(notice, 'cli', this.props.cliVersion) ||\n      match(resolveAliases(notice.components), this.props.tree);\n  }\n\n  /**\n   * Returns true iff we should show the notice.\n   */\n  private applyVersion(notice: Notice, name: string, compareToVersion: string | undefined) {\n    if (compareToVersion === undefined) { return false; }\n\n    const affectedComponent = notice.components.find(component => component.name === name);\n    const affectedRange = affectedComponent?.version;\n    return affectedRange != null && semver.satisfies(compareToVersion, affectedRange);\n  }\n}\n\n/**\n * Some component names are aliases to actual component names. For example \"framework\"\n * is an alias for either the core library (v1) or the whole CDK library (v2).\n *\n * This function converts all aliases to their actual counterpart names, to be used to\n * match against the construct tree.\n *\n * @param components a list of components. Components whose name is an alias will be\n * transformed and all others will be left intact.\n */\nfunction resolveAliases(components: Component[]): Component[] {\n  return flatMap(components, component => {\n    if (component.name === 'framework') {\n      return [{\n        name: '@aws-cdk/core.',\n        version: component.version,\n      }, {\n        name: 'aws-cdk-lib.',\n        version: component.version,\n      }];\n    } else {\n      return [component];\n    }\n  });\n}\n\nfunction formatNotice(notice: Notice): string {\n  const componentsValue = notice.components.map(c => `${c.name}: ${c.version}`).join(', ');\n  return [\n    `${notice.issueNumber}\\t${notice.title}`,\n    formatOverview(notice.overview),\n    `\\tAffected versions: ${componentsValue}`,\n    `\\tMore information at: https://github.com/aws/aws-cdk/issues/${notice.issueNumber}`,\n  ].join('\\n\\n') + '\\n';\n}\n\nfunction formatOverview(text: string) {\n  const wrap = (s: string) => s.replace(/(?![^\\n]{1,60}$)([^\\n]{1,60})\\s/g, '$1\\n');\n\n  const heading = 'Overview: ';\n  const separator = `\\n\\t${' '.repeat(heading.length)}`;\n  const content = wrap(text)\n    .split('\\n')\n    .join(separator);\n\n  return '\\t' + heading + content;\n}\n\n/**\n * Whether any component in the tree matches any component in the query.\n * A match happens when:\n *\n * 1. The version of the node matches the version in the query, interpreted\n * as a semver range.\n *\n * 2. The name in the query is a prefix of the node name when the query ends in '.',\n * or the two names are exactly the same, otherwise.\n */\nfunction match(query: Component[], tree: ConstructTreeNode): boolean {\n  return some(tree, node => {\n    return query.some(component =>\n      compareNames(component.name, node.constructInfo?.fqn) &&\n      compareVersions(component.version, node.constructInfo?.version));\n  });\n\n  function compareNames(pattern: string, target: string | undefined): boolean {\n    if (target == null) { return false; }\n    return pattern.endsWith('.') ? target.startsWith(pattern) : pattern === target;\n  }\n\n  function compareVersions(pattern: string, target: string | undefined): boolean {\n    return semver.satisfies(target ?? '', pattern);\n  }\n}\n\nfunction loadTree(outdir: string) {\n  try {\n    return fs.readJSONSync(path.join(outdir, 'tree.json'));\n  } catch (e) {\n    debug(`Failed to get tree.json file: ${e}`);\n    return {};\n  }\n}\n\n/**\n * Source information on a construct (class fqn and version)\n */\ninterface ConstructInfo {\n  readonly fqn: string;\n  readonly version: string;\n}\n\n/**\n * A node in the construct tree.\n * @internal\n */\ninterface ConstructTreeNode {\n  readonly id: string;\n  readonly path: string;\n  readonly children?: { [key: string]: ConstructTreeNode };\n  readonly attributes?: { [key: string]: any };\n\n  /**\n   * Information on the construct class that led to this node, if available\n   */\n  readonly constructInfo?: ConstructInfo;\n}\n\nfunction some(node: ConstructTreeNode, predicate: (n: ConstructTreeNode) => boolean): boolean {\n  return node != null && (predicate(node) || findInChildren());\n\n  function findInChildren(): boolean {\n    if (node.children == null) { return false; }\n\n    for (const name in node.children) {\n      if (some(node.children[name], predicate)) {\n        return true;\n      }\n    }\n    return false;\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "aws-cdk",
3
3
  "description": "CDK Toolkit, the command line tool for CDK apps",
4
- "version": "2.14.0",
4
+ "version": "2.15.0",
5
5
  "bin": {
6
6
  "cdk": "bin/cdk"
7
7
  },
@@ -58,7 +58,7 @@
58
58
  },
59
59
  "license": "Apache-2.0",
60
60
  "devDependencies": {
61
- "@aws-cdk/core": "2.14.0",
61
+ "@aws-cdk/core": "2.15.0",
62
62
  "@octokit/rest": "^18.12.0",
63
63
  "@types/archiver": "^5.3.1",
64
64
  "@types/fs-extra": "^8.1.2",
@@ -75,27 +75,27 @@
75
75
  "@types/wrap-ansi": "^3.0.0",
76
76
  "@types/yargs": "^15.0.14",
77
77
  "aws-sdk-mock": "5.6.0",
78
- "@aws-cdk/cdk-build-tools": "2.14.0",
78
+ "@aws-cdk/cdk-build-tools": "2.15.0",
79
79
  "jest": "^27.5.1",
80
80
  "madge": "^5.0.1",
81
81
  "constructs": "^10.0.0",
82
82
  "make-runnable": "^1.3.10",
83
83
  "mockery": "^2.1.0",
84
84
  "nock": "^13.2.4",
85
- "@aws-cdk/pkglint": "2.14.0",
85
+ "@aws-cdk/pkglint": "2.15.0",
86
86
  "sinon": "^9.2.4",
87
87
  "ts-jest": "^27.1.3",
88
88
  "ts-mock-imports": "^1.3.8",
89
89
  "xml-js": "^1.6.11",
90
- "@aws-cdk/cloud-assembly-schema": "2.14.0",
91
- "@aws-cdk/cloudformation-diff": "2.14.0",
92
- "@aws-cdk/cx-api": "2.14.0",
93
- "@aws-cdk/region-info": "2.14.0",
90
+ "@aws-cdk/cloud-assembly-schema": "2.15.0",
91
+ "@aws-cdk/cloudformation-diff": "2.15.0",
92
+ "@aws-cdk/cx-api": "2.15.0",
93
+ "@aws-cdk/region-info": "2.15.0",
94
94
  "@jsii/check-node": "1.54.0",
95
95
  "archiver": "^5.3.0",
96
96
  "aws-sdk": "^2.979.0",
97
97
  "camelcase": "^6.3.0",
98
- "cdk-assets": "2.14.0",
98
+ "cdk-assets": "2.15.0",
99
99
  "chokidar": "^3.5.3",
100
100
  "chalk": "^4",
101
101
  "decamelize": "^5.0.1",
@@ -117,7 +117,7 @@ test('changes only to CDK::Metadata result in a noOp', async () => {
117
117
  setup.setCurrentCfnStackTemplate({
118
118
  Resources: {
119
119
  MetaData: {
120
- Type: 'AWS::CDK::MetaData',
120
+ Type: 'AWS::CDK::Metadata',
121
121
  Properties: {
122
122
  Prop: 'old-value',
123
123
  },
@@ -295,4 +295,40 @@ test('can correctly reference AWS::URLSuffix in hotswappable changes', async ()
295
295
  expect(hotswapMockSdkProvider.mockSdkProvider.sdk.removeCustomUserAgent)
296
296
  .toHaveBeenCalledWith('cdk-hotswap/success-lambda-function');
297
297
  });
298
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-deployments.test.js","sourceRoot":"","sources":["hotswap-deployments.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { Lambda, StepFunctions } from 'aws-sdk';\nimport * as setup from './hotswap-test-setup';\n\nlet hotswapMockSdkProvider: setup.HotswapMockSdkProvider;\nlet mockUpdateLambdaCode: (params: Lambda.Types.UpdateFunctionCodeRequest) => Lambda.Types.FunctionConfiguration;\nlet mockUpdateMachineDefinition: (params: StepFunctions.Types.UpdateStateMachineInput) => StepFunctions.Types.UpdateStateMachineOutput;\nlet mockGetEndpointSuffix: () => string;\n\nbeforeEach(() => {\n  hotswapMockSdkProvider = setup.setupHotswapTests();\n  mockUpdateLambdaCode = jest.fn().mockReturnValue({});\n  mockUpdateMachineDefinition = jest.fn();\n  mockGetEndpointSuffix = jest.fn(() => 'amazonaws.com');\n  hotswapMockSdkProvider.stubLambda({\n    updateFunctionCode: mockUpdateLambdaCode,\n  });\n  hotswapMockSdkProvider.setUpdateStateMachineMock(mockUpdateMachineDefinition);\n  hotswapMockSdkProvider.stubGetEndpointSuffix(mockGetEndpointSuffix);\n});\n\ntest('returns a deployStackResult with noOp=true when it receives an empty set of changes', async () => {\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(setup.cdkStackArtifactOf());\n\n  // THEN\n  expect(deployStackResult).not.toBeUndefined();\n  expect(deployStackResult?.noOp).toBeTruthy();\n  expect(deployStackResult?.stackArn).toEqual(setup.STACK_ID);\n});\n\ntest('A change to only a non-hotswappable resource results in a full deployment', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      SomethingElse: {\n        Type: 'AWS::CloudFormation::SomethingElse',\n        Properties: {\n          Prop: 'old-value',\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        SomethingElse: {\n          Type: 'AWS::CloudFormation::SomethingElse',\n          Properties: {\n            Prop: 'new-value',\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).toBeUndefined();\n  expect(mockUpdateMachineDefinition).not.toHaveBeenCalled();\n  expect(mockUpdateLambdaCode).not.toHaveBeenCalled();\n});\n\ntest('A change to both a hotswappable resource and a non-hotswappable resource results in a full deployment', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      Func: {\n        Type: 'AWS::Lambda::Function',\n        Properties: {\n          Code: {\n            S3Bucket: 'current-bucket',\n            S3Key: 'current-key',\n          },\n          FunctionName: 'my-function',\n        },\n        Metadata: {\n          'aws:asset:path': 'old-path',\n        },\n      },\n      SomethingElse: {\n        Type: 'AWS::CloudFormation::SomethingElse',\n        Properties: {\n          Prop: 'old-value',\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        Func: {\n          Type: 'AWS::Lambda::Function',\n          Properties: {\n            Code: {\n              S3Bucket: 'current-bucket',\n              S3Key: 'new-key',\n            },\n            FunctionName: 'my-function',\n          },\n          Metadata: {\n            'aws:asset:path': 'new-path',\n          },\n        },\n        SomethingElse: {\n          Type: 'AWS::CloudFormation::SomethingElse',\n          Properties: {\n            Prop: 'new-value',\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).toBeUndefined();\n  expect(mockUpdateMachineDefinition).not.toHaveBeenCalled();\n  expect(mockUpdateLambdaCode).not.toHaveBeenCalled();\n});\n\ntest('changes only to CDK::Metadata result in a noOp', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      MetaData: {\n        Type: 'AWS::CDK::MetaData',\n        Properties: {\n          Prop: 'old-value',\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        MetaData: {\n          Type: 'AWS::CDK::Metadata',\n          Properties: {\n            Prop: 'new-value',\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).not.toBeUndefined();\n  expect(deployStackResult?.noOp).toEqual(true);\n  expect(mockUpdateMachineDefinition).not.toHaveBeenCalled();\n  expect(mockUpdateLambdaCode).not.toHaveBeenCalled();\n});\n\ntest('resource deletions require full deployments', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      Machine: {\n        Type: 'AWS::StepFunctions::StateMachine',\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf();\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).toBeUndefined();\n  expect(mockUpdateLambdaCode).not.toHaveBeenCalled();\n  expect(mockUpdateMachineDefinition).not.toHaveBeenCalled();\n});\n\ntest('can correctly reference AWS::Partition in hotswappable changes', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      Func: {\n        Type: 'AWS::Lambda::Function',\n        Properties: {\n          Code: {\n            S3Bucket: 'current-bucket',\n            S3Key: 'current-key',\n          },\n          FunctionName: {\n            'Fn::Join': [\n              '',\n              [\n                { Ref: 'AWS::Partition' },\n                '-',\n                'my-function',\n              ],\n            ],\n          },\n        },\n        Metadata: {\n          'aws:asset:path': 'new-path',\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        Func: {\n          Type: 'AWS::Lambda::Function',\n          Properties: {\n            Code: {\n              S3Bucket: 'current-bucket',\n              S3Key: 'new-key',\n            },\n            FunctionName: {\n              'Fn::Join': [\n                '',\n                [\n                  { Ref: 'AWS::Partition' },\n                  '-',\n                  'my-function',\n                ],\n              ],\n            },\n          },\n          Metadata: {\n            'aws:asset:path': 'new-path',\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).not.toBeUndefined();\n  expect(mockUpdateLambdaCode).toHaveBeenCalledWith({\n    FunctionName: 'aws-my-function',\n    S3Bucket: 'current-bucket',\n    S3Key: 'new-key',\n  });\n});\n\ntest('can correctly reference AWS::URLSuffix in hotswappable changes', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      Func: {\n        Type: 'AWS::Lambda::Function',\n        Properties: {\n          Code: {\n            S3Bucket: 'current-bucket',\n            S3Key: 'current-key',\n          },\n          FunctionName: {\n            'Fn::Join': ['', [\n              'my-function-',\n              { Ref: 'AWS::URLSuffix' },\n              '-',\n              { Ref: 'AWS::URLSuffix' },\n            ]],\n          },\n        },\n        Metadata: {\n          'aws:asset:path': 'old-path',\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        Func: {\n          Type: 'AWS::Lambda::Function',\n          Properties: {\n            Code: {\n              S3Bucket: 'current-bucket',\n              S3Key: 'new-key',\n            },\n            FunctionName: {\n              'Fn::Join': ['', [\n                'my-function-',\n                { Ref: 'AWS::URLSuffix' },\n                '-',\n                { Ref: 'AWS::URLSuffix' },\n              ]],\n            },\n          },\n          Metadata: {\n            'aws:asset:path': 'new-path',\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).not.toBeUndefined();\n  expect(mockUpdateLambdaCode).toHaveBeenCalledWith({\n    FunctionName: 'my-function-amazonaws.com-amazonaws.com',\n    S3Bucket: 'current-bucket',\n    S3Key: 'new-key',\n  });\n  expect(mockGetEndpointSuffix).toHaveBeenCalledTimes(1);\n\n  // the User-Agent is set correctly\n  expect(hotswapMockSdkProvider.mockSdkProvider.sdk.appendCustomUserAgent)\n    .toHaveBeenCalledWith('cdk-hotswap/success-lambda-function');\n  expect(hotswapMockSdkProvider.mockSdkProvider.sdk.removeCustomUserAgent)\n    .toHaveBeenCalledWith('cdk-hotswap/success-lambda-function');\n});\n"]}
298
+ test('changing the type of a deployed resource always results in a full deployment', async () => {
299
+ // GIVEN
300
+ setup.setCurrentCfnStackTemplate({
301
+ Resources: {
302
+ SharedLogicalId: {
303
+ Type: 'AWS::Lambda::Function',
304
+ Properties: {
305
+ Code: {
306
+ S3Bucket: 'current-bucket',
307
+ S3Key: 'new-key',
308
+ },
309
+ FunctionName: 'my-function',
310
+ },
311
+ },
312
+ },
313
+ });
314
+ const cdkStackArtifact = setup.cdkStackArtifactOf({
315
+ template: {
316
+ Resources: {
317
+ SharedLogicalId: {
318
+ Type: 'AWS::StepFunctions::StateMachine',
319
+ Properties: {
320
+ DefinitionString: '{ Prop: "new-value" }',
321
+ StateMachineName: 'my-machine',
322
+ },
323
+ },
324
+ },
325
+ },
326
+ });
327
+ // WHEN
328
+ const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);
329
+ // THEN
330
+ expect(deployStackResult).toBeUndefined();
331
+ expect(mockUpdateMachineDefinition).not.toHaveBeenCalled();
332
+ expect(mockUpdateLambdaCode).not.toHaveBeenCalled();
333
+ });
334
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-deployments.test.js","sourceRoot":"","sources":["hotswap-deployments.test.ts"],"names":[],"mappings":";;AACA,8CAA8C;AAE9C,IAAI,sBAAoD,CAAC;AACzD,IAAI,oBAA4G,CAAC;AACjH,IAAI,2BAAkI,CAAC;AACvI,IAAI,qBAAmC,CAAC;AAExC,UAAU,CAAC,GAAG,EAAE;IACd,sBAAsB,GAAG,KAAK,CAAC,iBAAiB,EAAE,CAAC;IACnD,oBAAoB,GAAG,IAAI,CAAC,EAAE,EAAE,CAAC,eAAe,CAAC,EAAE,CAAC,CAAC;IACrD,2BAA2B,GAAG,IAAI,CAAC,EAAE,EAAE,CAAC;IACxC,qBAAqB,GAAG,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC,eAAe,CAAC,CAAC;IACvD,sBAAsB,CAAC,UAAU,CAAC;QAChC,kBAAkB,EAAE,oBAAoB;KACzC,CAAC,CAAC;IACH,sBAAsB,CAAC,yBAAyB,CAAC,2BAA2B,CAAC,CAAC;IAC9E,sBAAsB,CAAC,qBAAqB,CAAC,qBAAqB,CAAC,CAAC;AACtE,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,qFAAqF,EAAE,KAAK,IAAI,EAAE;IACrG,OAAO;IACP,MAAM,iBAAiB,GAAG,MAAM,sBAAsB,CAAC,oBAAoB,CAAC,KAAK,CAAC,kBAAkB,EAAE,CAAC,CAAC;IAExG,OAAO;IACP,MAAM,CAAC,iBAAiB,CAAC,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;IAC9C,MAAM,CAAC,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,IAAI,CAAC,CAAC,UAAU,EAAE,CAAC;IAC7C,MAAM,CAAC,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,QAAQ,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;AAC9D,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,2EAA2E,EAAE,KAAK,IAAI,EAAE;IAC3F,QAAQ;IACR,KAAK,CAAC,0BAA0B,CAAC;QAC/B,SAAS,EAAE;YACT,aAAa,EAAE;gBACb,IAAI,EAAE,oCAAoC;gBAC1C,UAAU,EAAE;oBACV,IAAI,EAAE,WAAW;iBAClB;aACF;SACF;KACF,CAAC,CAAC;IACH,MAAM,gBAAgB,GAAG,KAAK,CAAC,kBAAkB,CAAC;QAChD,QAAQ,EAAE;YACR,SAAS,EAAE;gBACT,aAAa,EAAE;oBACb,IAAI,EAAE,oCAAoC;oBAC1C,UAAU,EAAE;wBACV,IAAI,EAAE,WAAW;qBAClB;iBACF;aACF;SACF;KACF,CAAC,CAAC;IAEH,OAAO;IACP,MAAM,iBAAiB,GAAG,MAAM,sBAAsB,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,CAAC;IAE9F,OAAO;IACP,MAAM,CAAC,iBAAiB,CAAC,CAAC,aAAa,EAAE,CAAC;IAC1C,MAAM,CAAC,2BAA2B,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;IAC3D,MAAM,CAAC,oBAAoB,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;AACtD,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,uGAAuG,EAAE,KAAK,IAAI,EAAE;IACvH,QAAQ;IACR,KAAK,CAAC,0BAA0B,CAAC;QAC/B,SAAS,EAAE;YACT,IAAI,EAAE;gBACJ,IAAI,EAAE,uBAAuB;gBAC7B,UAAU,EAAE;oBACV,IAAI,EAAE;wBACJ,QAAQ,EAAE,gBAAgB;wBAC1B,KAAK,EAAE,aAAa;qBACrB;oBACD,YAAY,EAAE,aAAa;iBAC5B;gBACD,QAAQ,EAAE;oBACR,gBAAgB,EAAE,UAAU;iBAC7B;aACF;YACD,aAAa,EAAE;gBACb,IAAI,EAAE,oCAAoC;gBAC1C,UAAU,EAAE;oBACV,IAAI,EAAE,WAAW;iBAClB;aACF;SACF;KACF,CAAC,CAAC;IACH,MAAM,gBAAgB,GAAG,KAAK,CAAC,kBAAkB,CAAC;QAChD,QAAQ,EAAE;YACR,SAAS,EAAE;gBACT,IAAI,EAAE;oBACJ,IAAI,EAAE,uBAAuB;oBAC7B,UAAU,EAAE;wBACV,IAAI,EAAE;4BACJ,QAAQ,EAAE,gBAAgB;4BAC1B,KAAK,EAAE,SAAS;yBACjB;wBACD,YAAY,EAAE,aAAa;qBAC5B;oBACD,QAAQ,EAAE;wBACR,gBAAgB,EAAE,UAAU;qBAC7B;iBACF;gBACD,aAAa,EAAE;oBACb,IAAI,EAAE,oCAAoC;oBAC1C,UAAU,EAAE;wBACV,IAAI,EAAE,WAAW;qBAClB;iBACF;aACF;SACF;KACF,CAAC,CAAC;IAEH,OAAO;IACP,MAAM,iBAAiB,GAAG,MAAM,sBAAsB,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,CAAC;IAE9F,OAAO;IACP,MAAM,CAAC,iBAAiB,CAAC,CAAC,aAAa,EAAE,CAAC;IAC1C,MAAM,CAAC,2BAA2B,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;IAC3D,MAAM,CAAC,oBAAoB,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;AACtD,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,gDAAgD,EAAE,KAAK,IAAI,EAAE;IAChE,QAAQ;IACR,KAAK,CAAC,0BAA0B,CAAC;QAC/B,SAAS,EAAE;YACT,QAAQ,EAAE;gBACR,IAAI,EAAE,oBAAoB;gBAC1B,UAAU,EAAE;oBACV,IAAI,EAAE,WAAW;iBAClB;aACF;SACF;KACF,CAAC,CAAC;IACH,MAAM,gBAAgB,GAAG,KAAK,CAAC,kBAAkB,CAAC;QAChD,QAAQ,EAAE;YACR,SAAS,EAAE;gBACT,QAAQ,EAAE;oBACR,IAAI,EAAE,oBAAoB;oBAC1B,UAAU,EAAE;wBACV,IAAI,EAAE,WAAW;qBAClB;iBACF;aACF;SACF;KACF,CAAC,CAAC;IAEH,OAAO;IACP,MAAM,iBAAiB,GAAG,MAAM,sBAAsB,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,CAAC;IAE9F,OAAO;IACP,MAAM,CAAC,iBAAiB,CAAC,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;IAC9C,MAAM,CAAC,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,IAAI,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;IAC9C,MAAM,CAAC,2BAA2B,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;IAC3D,MAAM,CAAC,oBAAoB,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;AACtD,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,6CAA6C,EAAE,KAAK,IAAI,EAAE;IAC7D,QAAQ;IACR,KAAK,CAAC,0BAA0B,CAAC;QAC/B,SAAS,EAAE;YACT,OAAO,EAAE;gBACP,IAAI,EAAE,kCAAkC;aACzC;SACF;KACF,CAAC,CAAC;IACH,MAAM,gBAAgB,GAAG,KAAK,CAAC,kBAAkB,EAAE,CAAC;IAEpD,OAAO;IACP,MAAM,iBAAiB,GAAG,MAAM,sBAAsB,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,CAAC;IAE9F,OAAO;IACP,MAAM,CAAC,iBAAiB,CAAC,CAAC,aAAa,EAAE,CAAC;IAC1C,MAAM,CAAC,oBAAoB,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;IACpD,MAAM,CAAC,2BAA2B,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;AAC7D,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,gEAAgE,EAAE,KAAK,IAAI,EAAE;IAChF,QAAQ;IACR,KAAK,CAAC,0BAA0B,CAAC;QAC/B,SAAS,EAAE;YACT,IAAI,EAAE;gBACJ,IAAI,EAAE,uBAAuB;gBAC7B,UAAU,EAAE;oBACV,IAAI,EAAE;wBACJ,QAAQ,EAAE,gBAAgB;wBAC1B,KAAK,EAAE,aAAa;qBACrB;oBACD,YAAY,EAAE;wBACZ,UAAU,EAAE;4BACV,EAAE;4BACF;gCACE,EAAE,GAAG,EAAE,gBAAgB,EAAE;gCACzB,GAAG;gCACH,aAAa;6BACd;yBACF;qBACF;iBACF;gBACD,QAAQ,EAAE;oBACR,gBAAgB,EAAE,UAAU;iBAC7B;aACF;SACF;KACF,CAAC,CAAC;IACH,MAAM,gBAAgB,GAAG,KAAK,CAAC,kBAAkB,CAAC;QAChD,QAAQ,EAAE;YACR,SAAS,EAAE;gBACT,IAAI,EAAE;oBACJ,IAAI,EAAE,uBAAuB;oBAC7B,UAAU,EAAE;wBACV,IAAI,EAAE;4BACJ,QAAQ,EAAE,gBAAgB;4BAC1B,KAAK,EAAE,SAAS;yBACjB;wBACD,YAAY,EAAE;4BACZ,UAAU,EAAE;gCACV,EAAE;gCACF;oCACE,EAAE,GAAG,EAAE,gBAAgB,EAAE;oCACzB,GAAG;oCACH,aAAa;iCACd;6BACF;yBACF;qBACF;oBACD,QAAQ,EAAE;wBACR,gBAAgB,EAAE,UAAU;qBAC7B;iBACF;aACF;SACF;KACF,CAAC,CAAC;IAEH,OAAO;IACP,MAAM,iBAAiB,GAAG,MAAM,sBAAsB,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,CAAC;IAE9F,OAAO;IACP,MAAM,CAAC,iBAAiB,CAAC,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;IAC9C,MAAM,CAAC,oBAAoB,CAAC,CAAC,oBAAoB,CAAC;QAChD,YAAY,EAAE,iBAAiB;QAC/B,QAAQ,EAAE,gBAAgB;QAC1B,KAAK,EAAE,SAAS;KACjB,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,gEAAgE,EAAE,KAAK,IAAI,EAAE;IAChF,QAAQ;IACR,KAAK,CAAC,0BAA0B,CAAC;QAC/B,SAAS,EAAE;YACT,IAAI,EAAE;gBACJ,IAAI,EAAE,uBAAuB;gBAC7B,UAAU,EAAE;oBACV,IAAI,EAAE;wBACJ,QAAQ,EAAE,gBAAgB;wBAC1B,KAAK,EAAE,aAAa;qBACrB;oBACD,YAAY,EAAE;wBACZ,UAAU,EAAE,CAAC,EAAE,EAAE;gCACf,cAAc;gCACd,EAAE,GAAG,EAAE,gBAAgB,EAAE;gCACzB,GAAG;gCACH,EAAE,GAAG,EAAE,gBAAgB,EAAE;6BAC1B,CAAC;qBACH;iBACF;gBACD,QAAQ,EAAE;oBACR,gBAAgB,EAAE,UAAU;iBAC7B;aACF;SACF;KACF,CAAC,CAAC;IACH,MAAM,gBAAgB,GAAG,KAAK,CAAC,kBAAkB,CAAC;QAChD,QAAQ,EAAE;YACR,SAAS,EAAE;gBACT,IAAI,EAAE;oBACJ,IAAI,EAAE,uBAAuB;oBAC7B,UAAU,EAAE;wBACV,IAAI,EAAE;4BACJ,QAAQ,EAAE,gBAAgB;4BAC1B,KAAK,EAAE,SAAS;yBACjB;wBACD,YAAY,EAAE;4BACZ,UAAU,EAAE,CAAC,EAAE,EAAE;oCACf,cAAc;oCACd,EAAE,GAAG,EAAE,gBAAgB,EAAE;oCACzB,GAAG;oCACH,EAAE,GAAG,EAAE,gBAAgB,EAAE;iCAC1B,CAAC;yBACH;qBACF;oBACD,QAAQ,EAAE;wBACR,gBAAgB,EAAE,UAAU;qBAC7B;iBACF;aACF;SACF;KACF,CAAC,CAAC;IAEH,OAAO;IACP,MAAM,iBAAiB,GAAG,MAAM,sBAAsB,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,CAAC;IAE9F,OAAO;IACP,MAAM,CAAC,iBAAiB,CAAC,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;IAC9C,MAAM,CAAC,oBAAoB,CAAC,CAAC,oBAAoB,CAAC;QAChD,YAAY,EAAE,yCAAyC;QACvD,QAAQ,EAAE,gBAAgB;QAC1B,KAAK,EAAE,SAAS;KACjB,CAAC,CAAC;IACH,MAAM,CAAC,qBAAqB,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;IAEvD,kCAAkC;IAClC,MAAM,CAAC,sBAAsB,CAAC,eAAe,CAAC,GAAG,CAAC,qBAAqB,CAAC;SACrE,oBAAoB,CAAC,qCAAqC,CAAC,CAAC;IAC/D,MAAM,CAAC,sBAAsB,CAAC,eAAe,CAAC,GAAG,CAAC,qBAAqB,CAAC;SACrE,oBAAoB,CAAC,qCAAqC,CAAC,CAAC;AACjE,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,8EAA8E,EAAE,KAAK,IAAI,EAAE;IAC9F,QAAQ;IACR,KAAK,CAAC,0BAA0B,CAAC;QAC/B,SAAS,EAAE;YACT,eAAe,EAAE;gBACf,IAAI,EAAE,uBAAuB;gBAC7B,UAAU,EAAE;oBACV,IAAI,EAAE;wBACJ,QAAQ,EAAE,gBAAgB;wBAC1B,KAAK,EAAE,SAAS;qBACjB;oBACD,YAAY,EAAE,aAAa;iBAC5B;aACF;SACF;KACF,CAAC,CAAC;IACH,MAAM,gBAAgB,GAAG,KAAK,CAAC,kBAAkB,CAAC;QAChD,QAAQ,EAAE;YACR,SAAS,EAAE;gBACT,eAAe,EAAE;oBACf,IAAI,EAAE,kCAAkC;oBACxC,UAAU,EAAE;wBACV,gBAAgB,EAAE,uBAAuB;wBACzC,gBAAgB,EAAE,YAAY;qBAC/B;iBACF;aACF;SACF;KACF,CAAC,CAAC;IAEH,OAAO;IACP,MAAM,iBAAiB,GAAG,MAAM,sBAAsB,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,CAAC;IAE9F,OAAO;IACP,MAAM,CAAC,iBAAiB,CAAC,CAAC,aAAa,EAAE,CAAC;IAC1C,MAAM,CAAC,2BAA2B,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;IAC3D,MAAM,CAAC,oBAAoB,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;AACtD,CAAC,CAAC,CAAC","sourcesContent":["import { Lambda, StepFunctions } from 'aws-sdk';\nimport * as setup from './hotswap-test-setup';\n\nlet hotswapMockSdkProvider: setup.HotswapMockSdkProvider;\nlet mockUpdateLambdaCode: (params: Lambda.Types.UpdateFunctionCodeRequest) => Lambda.Types.FunctionConfiguration;\nlet mockUpdateMachineDefinition: (params: StepFunctions.Types.UpdateStateMachineInput) => StepFunctions.Types.UpdateStateMachineOutput;\nlet mockGetEndpointSuffix: () => string;\n\nbeforeEach(() => {\n  hotswapMockSdkProvider = setup.setupHotswapTests();\n  mockUpdateLambdaCode = jest.fn().mockReturnValue({});\n  mockUpdateMachineDefinition = jest.fn();\n  mockGetEndpointSuffix = jest.fn(() => 'amazonaws.com');\n  hotswapMockSdkProvider.stubLambda({\n    updateFunctionCode: mockUpdateLambdaCode,\n  });\n  hotswapMockSdkProvider.setUpdateStateMachineMock(mockUpdateMachineDefinition);\n  hotswapMockSdkProvider.stubGetEndpointSuffix(mockGetEndpointSuffix);\n});\n\ntest('returns a deployStackResult with noOp=true when it receives an empty set of changes', async () => {\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(setup.cdkStackArtifactOf());\n\n  // THEN\n  expect(deployStackResult).not.toBeUndefined();\n  expect(deployStackResult?.noOp).toBeTruthy();\n  expect(deployStackResult?.stackArn).toEqual(setup.STACK_ID);\n});\n\ntest('A change to only a non-hotswappable resource results in a full deployment', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      SomethingElse: {\n        Type: 'AWS::CloudFormation::SomethingElse',\n        Properties: {\n          Prop: 'old-value',\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        SomethingElse: {\n          Type: 'AWS::CloudFormation::SomethingElse',\n          Properties: {\n            Prop: 'new-value',\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).toBeUndefined();\n  expect(mockUpdateMachineDefinition).not.toHaveBeenCalled();\n  expect(mockUpdateLambdaCode).not.toHaveBeenCalled();\n});\n\ntest('A change to both a hotswappable resource and a non-hotswappable resource results in a full deployment', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      Func: {\n        Type: 'AWS::Lambda::Function',\n        Properties: {\n          Code: {\n            S3Bucket: 'current-bucket',\n            S3Key: 'current-key',\n          },\n          FunctionName: 'my-function',\n        },\n        Metadata: {\n          'aws:asset:path': 'old-path',\n        },\n      },\n      SomethingElse: {\n        Type: 'AWS::CloudFormation::SomethingElse',\n        Properties: {\n          Prop: 'old-value',\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        Func: {\n          Type: 'AWS::Lambda::Function',\n          Properties: {\n            Code: {\n              S3Bucket: 'current-bucket',\n              S3Key: 'new-key',\n            },\n            FunctionName: 'my-function',\n          },\n          Metadata: {\n            'aws:asset:path': 'new-path',\n          },\n        },\n        SomethingElse: {\n          Type: 'AWS::CloudFormation::SomethingElse',\n          Properties: {\n            Prop: 'new-value',\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).toBeUndefined();\n  expect(mockUpdateMachineDefinition).not.toHaveBeenCalled();\n  expect(mockUpdateLambdaCode).not.toHaveBeenCalled();\n});\n\ntest('changes only to CDK::Metadata result in a noOp', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      MetaData: {\n        Type: 'AWS::CDK::Metadata',\n        Properties: {\n          Prop: 'old-value',\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        MetaData: {\n          Type: 'AWS::CDK::Metadata',\n          Properties: {\n            Prop: 'new-value',\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).not.toBeUndefined();\n  expect(deployStackResult?.noOp).toEqual(true);\n  expect(mockUpdateMachineDefinition).not.toHaveBeenCalled();\n  expect(mockUpdateLambdaCode).not.toHaveBeenCalled();\n});\n\ntest('resource deletions require full deployments', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      Machine: {\n        Type: 'AWS::StepFunctions::StateMachine',\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf();\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).toBeUndefined();\n  expect(mockUpdateLambdaCode).not.toHaveBeenCalled();\n  expect(mockUpdateMachineDefinition).not.toHaveBeenCalled();\n});\n\ntest('can correctly reference AWS::Partition in hotswappable changes', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      Func: {\n        Type: 'AWS::Lambda::Function',\n        Properties: {\n          Code: {\n            S3Bucket: 'current-bucket',\n            S3Key: 'current-key',\n          },\n          FunctionName: {\n            'Fn::Join': [\n              '',\n              [\n                { Ref: 'AWS::Partition' },\n                '-',\n                'my-function',\n              ],\n            ],\n          },\n        },\n        Metadata: {\n          'aws:asset:path': 'new-path',\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        Func: {\n          Type: 'AWS::Lambda::Function',\n          Properties: {\n            Code: {\n              S3Bucket: 'current-bucket',\n              S3Key: 'new-key',\n            },\n            FunctionName: {\n              'Fn::Join': [\n                '',\n                [\n                  { Ref: 'AWS::Partition' },\n                  '-',\n                  'my-function',\n                ],\n              ],\n            },\n          },\n          Metadata: {\n            'aws:asset:path': 'new-path',\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).not.toBeUndefined();\n  expect(mockUpdateLambdaCode).toHaveBeenCalledWith({\n    FunctionName: 'aws-my-function',\n    S3Bucket: 'current-bucket',\n    S3Key: 'new-key',\n  });\n});\n\ntest('can correctly reference AWS::URLSuffix in hotswappable changes', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      Func: {\n        Type: 'AWS::Lambda::Function',\n        Properties: {\n          Code: {\n            S3Bucket: 'current-bucket',\n            S3Key: 'current-key',\n          },\n          FunctionName: {\n            'Fn::Join': ['', [\n              'my-function-',\n              { Ref: 'AWS::URLSuffix' },\n              '-',\n              { Ref: 'AWS::URLSuffix' },\n            ]],\n          },\n        },\n        Metadata: {\n          'aws:asset:path': 'old-path',\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        Func: {\n          Type: 'AWS::Lambda::Function',\n          Properties: {\n            Code: {\n              S3Bucket: 'current-bucket',\n              S3Key: 'new-key',\n            },\n            FunctionName: {\n              'Fn::Join': ['', [\n                'my-function-',\n                { Ref: 'AWS::URLSuffix' },\n                '-',\n                { Ref: 'AWS::URLSuffix' },\n              ]],\n            },\n          },\n          Metadata: {\n            'aws:asset:path': 'new-path',\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).not.toBeUndefined();\n  expect(mockUpdateLambdaCode).toHaveBeenCalledWith({\n    FunctionName: 'my-function-amazonaws.com-amazonaws.com',\n    S3Bucket: 'current-bucket',\n    S3Key: 'new-key',\n  });\n  expect(mockGetEndpointSuffix).toHaveBeenCalledTimes(1);\n\n  // the User-Agent is set correctly\n  expect(hotswapMockSdkProvider.mockSdkProvider.sdk.appendCustomUserAgent)\n    .toHaveBeenCalledWith('cdk-hotswap/success-lambda-function');\n  expect(hotswapMockSdkProvider.mockSdkProvider.sdk.removeCustomUserAgent)\n    .toHaveBeenCalledWith('cdk-hotswap/success-lambda-function');\n});\n\ntest('changing the type of a deployed resource always results in a full deployment', async () => {\n  // GIVEN\n  setup.setCurrentCfnStackTemplate({\n    Resources: {\n      SharedLogicalId: {\n        Type: 'AWS::Lambda::Function',\n        Properties: {\n          Code: {\n            S3Bucket: 'current-bucket',\n            S3Key: 'new-key',\n          },\n          FunctionName: 'my-function',\n        },\n      },\n    },\n  });\n  const cdkStackArtifact = setup.cdkStackArtifactOf({\n    template: {\n      Resources: {\n        SharedLogicalId: {\n          Type: 'AWS::StepFunctions::StateMachine',\n          Properties: {\n            DefinitionString: '{ Prop: \"new-value\" }',\n            StateMachineName: 'my-machine',\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployStackResult = await hotswapMockSdkProvider.tryHotswapDeployment(cdkStackArtifact);\n\n  // THEN\n  expect(deployStackResult).toBeUndefined();\n  expect(mockUpdateMachineDefinition).not.toHaveBeenCalled();\n  expect(mockUpdateLambdaCode).not.toHaveBeenCalled();\n});\n"]}
@@ -1,5 +1,4 @@
1
1
  import * as cxapi from '@aws-cdk/cx-api';
2
- import { CloudFormation } from 'aws-sdk';
3
2
  import * as AWS from 'aws-sdk';
4
3
  import * as codebuild from 'aws-sdk/clients/codebuild';
5
4
  import * as lambda from 'aws-sdk/clients/lambda';
@@ -10,13 +9,16 @@ import { TestStackArtifact } from '../../util';
10
9
  import { MockSdkProvider, SyncHandlerSubsetOf } from '../../util/mock-sdk';
11
10
  export declare const STACK_ID = "stackId";
12
11
  export declare function setupHotswapTests(): HotswapMockSdkProvider;
12
+ export declare function setupHotswapNestedStackTests(rootStackName: string): HotswapMockSdkProvider;
13
13
  export declare function cdkStackArtifactOf(testStackArtifact?: Partial<TestStackArtifact>): cxapi.CloudFormationStackArtifact;
14
- export declare function pushStackResourceSummaries(...items: CloudFormation.StackResourceSummary[]): void;
14
+ export declare function pushStackResourceSummaries(...items: AWS.CloudFormation.StackResourceSummary[]): void;
15
+ export declare function pushNestedStackResourceSummaries(stackName: string, ...items: AWS.CloudFormation.StackResourceSummary[]): void;
15
16
  export declare function setCurrentCfnStackTemplate(template: Template): void;
16
- export declare function stackSummaryOf(logicalId: string, resourceType: string, physicalResourceId: string): CloudFormation.StackResourceSummary;
17
+ export declare function addTemplateToCloudFormationLookupMock(stackArtifact: cxapi.CloudFormationStackArtifact): void;
18
+ export declare function stackSummaryOf(logicalId: string, resourceType: string, physicalResourceId: string): AWS.CloudFormation.StackResourceSummary;
17
19
  export declare class HotswapMockSdkProvider {
18
20
  readonly mockSdkProvider: MockSdkProvider;
19
- constructor();
21
+ constructor(rootStackName?: string);
20
22
  setUpdateStateMachineMock(mockUpdateMachineDefinition: (input: stepfunctions.UpdateStateMachineInput) => stepfunctions.UpdateStateMachineOutput): void;
21
23
  stubLambda(stubs: SyncHandlerSubsetOf<AWS.Lambda>, serviceStubs?: SyncHandlerSubsetOf<AWS.Service>, additionalProperties?: {
22
24
  [key: string]: any;
@@ -1,7 +1,8 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.HotswapMockSdkProvider = exports.stackSummaryOf = exports.setCurrentCfnStackTemplate = exports.pushStackResourceSummaries = exports.cdkStackArtifactOf = exports.setupHotswapTests = exports.STACK_ID = void 0;
3
+ exports.HotswapMockSdkProvider = exports.stackSummaryOf = exports.addTemplateToCloudFormationLookupMock = exports.setCurrentCfnStackTemplate = exports.pushNestedStackResourceSummaries = exports.pushStackResourceSummaries = exports.cdkStackArtifactOf = exports.setupHotswapNestedStackTests = exports.setupHotswapTests = exports.STACK_ID = void 0;
4
4
  const deployments = require("../../../lib/api/hotswap-deployments");
5
+ const cloudformation_1 = require("../../../lib/api/util/cloudformation");
5
6
  const util_1 = require("../../util");
6
7
  const mock_sdk_1 = require("../../util/mock-sdk");
7
8
  const fake_cloudformation_stack_1 = require("../fake-cloudformation-stack");
@@ -10,6 +11,8 @@ exports.STACK_ID = 'stackId';
10
11
  let hotswapMockSdkProvider;
11
12
  let currentCfnStack;
12
13
  const currentCfnStackResources = [];
14
+ let stackTemplates;
15
+ let currentNestedCfnStackResources;
13
16
  function setupHotswapTests() {
14
17
  jest.resetAllMocks();
15
18
  // clear the array
@@ -19,9 +22,28 @@ function setupHotswapTests() {
19
22
  stackName: STACK_NAME,
20
23
  stackId: exports.STACK_ID,
21
24
  });
25
+ cloudformation_1.CloudFormationStack.lookup = async (_, _stackName) => {
26
+ return currentCfnStack;
27
+ };
22
28
  return hotswapMockSdkProvider;
23
29
  }
24
30
  exports.setupHotswapTests = setupHotswapTests;
31
+ function setupHotswapNestedStackTests(rootStackName) {
32
+ jest.resetAllMocks();
33
+ currentNestedCfnStackResources = {};
34
+ hotswapMockSdkProvider = new HotswapMockSdkProvider(rootStackName);
35
+ currentCfnStack = new fake_cloudformation_stack_1.FakeCloudformationStack({
36
+ stackName: rootStackName,
37
+ stackId: exports.STACK_ID,
38
+ });
39
+ stackTemplates = {};
40
+ cloudformation_1.CloudFormationStack.lookup = async (_, stackName) => {
41
+ currentCfnStack.template = async () => stackTemplates[stackName];
42
+ return currentCfnStack;
43
+ };
44
+ return hotswapMockSdkProvider;
45
+ }
46
+ exports.setupHotswapNestedStackTests = setupHotswapNestedStackTests;
25
47
  function cdkStackArtifactOf(testStackArtifact = {}) {
26
48
  return util_1.testStack({
27
49
  stackName: STACK_NAME,
@@ -33,11 +55,23 @@ function pushStackResourceSummaries(...items) {
33
55
  currentCfnStackResources.push(...items);
34
56
  }
35
57
  exports.pushStackResourceSummaries = pushStackResourceSummaries;
58
+ function pushNestedStackResourceSummaries(stackName, ...items) {
59
+ if (!currentNestedCfnStackResources[stackName]) {
60
+ currentNestedCfnStackResources[stackName] = [];
61
+ }
62
+ currentNestedCfnStackResources[stackName].push(...items);
63
+ }
64
+ exports.pushNestedStackResourceSummaries = pushNestedStackResourceSummaries;
36
65
  function setCurrentCfnStackTemplate(template) {
37
66
  const templateDeepCopy = JSON.parse(JSON.stringify(template)); // deep copy the template, so our tests can mutate one template instead of creating two
38
67
  currentCfnStack.setTemplate(templateDeepCopy);
39
68
  }
40
69
  exports.setCurrentCfnStackTemplate = setCurrentCfnStackTemplate;
70
+ function addTemplateToCloudFormationLookupMock(stackArtifact) {
71
+ const templateDeepCopy = JSON.parse(JSON.stringify(stackArtifact.template)); // deep copy the template, so our tests can mutate one template instead of creating two
72
+ stackTemplates[stackArtifact.stackName] = templateDeepCopy;
73
+ }
74
+ exports.addTemplateToCloudFormationLookupMock = addTemplateToCloudFormationLookupMock;
41
75
  function stackSummaryOf(logicalId, resourceType, physicalResourceId) {
42
76
  return {
43
77
  LogicalResourceId: logicalId,
@@ -49,15 +83,21 @@ function stackSummaryOf(logicalId, resourceType, physicalResourceId) {
49
83
  }
50
84
  exports.stackSummaryOf = stackSummaryOf;
51
85
  class HotswapMockSdkProvider {
52
- constructor() {
86
+ constructor(rootStackName) {
53
87
  this.mockSdkProvider = new mock_sdk_1.MockSdkProvider({ realSdk: false });
54
88
  this.mockSdkProvider.stubCloudFormation({
55
89
  listStackResources: ({ StackName: stackName }) => {
56
- if (stackName !== STACK_NAME) {
57
- throw new Error(`Expected Stack name in listStackResources() call to be: '${STACK_NAME}', but received: ${stackName}'`);
90
+ if (rootStackName) {
91
+ const knownStackNames = Object.keys(currentNestedCfnStackResources);
92
+ if (stackName !== rootStackName && !knownStackNames.includes(stackName)) {
93
+ throw new Error(`Expected Stack name in listStackResources() call to be a member of ['${rootStackName}, ${knownStackNames}'], but received: '${stackName}'`);
94
+ }
95
+ }
96
+ else if (stackName !== STACK_NAME) {
97
+ throw new Error(`Expected Stack name in listStackResources() call to be: '${STACK_NAME}', but received: '${stackName}'`);
58
98
  }
59
99
  return {
60
- StackResourceSummaries: currentCfnStackResources,
100
+ StackResourceSummaries: rootStackName ? currentNestedCfnStackResources[stackName] : currentCfnStackResources,
61
101
  };
62
102
  },
63
103
  });
@@ -110,4 +150,4 @@ class HotswapMockSdkProvider {
110
150
  }
111
151
  }
112
152
  exports.HotswapMockSdkProvider = HotswapMockSdkProvider;
113
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-test-setup.js","sourceRoot":"","sources":["hotswap-test-setup.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport { CloudFormation } from 'aws-sdk';\nimport * as AWS from 'aws-sdk';\nimport * as codebuild from 'aws-sdk/clients/codebuild';\nimport * as lambda from 'aws-sdk/clients/lambda';\nimport * as stepfunctions from 'aws-sdk/clients/stepfunctions';\nimport { DeployStackResult } from '../../../lib/api';\nimport * as deployments from '../../../lib/api/hotswap-deployments';\nimport { Template } from '../../../lib/api/util/cloudformation';\nimport { testStack, TestStackArtifact } from '../../util';\nimport { MockSdkProvider, SyncHandlerSubsetOf } from '../../util/mock-sdk';\nimport { FakeCloudformationStack } from '../fake-cloudformation-stack';\n\nconst STACK_NAME = 'withouterrors';\nexport const STACK_ID = 'stackId';\n\nlet hotswapMockSdkProvider: HotswapMockSdkProvider;\nlet currentCfnStack: FakeCloudformationStack;\nconst currentCfnStackResources: CloudFormation.StackResourceSummary[] = [];\n\nexport function setupHotswapTests() {\n  jest.resetAllMocks();\n  // clear the array\n  currentCfnStackResources.splice(0);\n  hotswapMockSdkProvider = new HotswapMockSdkProvider();\n  currentCfnStack = new FakeCloudformationStack({\n    stackName: STACK_NAME,\n    stackId: STACK_ID,\n  });\n\n  return hotswapMockSdkProvider;\n}\n\nexport function cdkStackArtifactOf(testStackArtifact: Partial<TestStackArtifact> = {}): cxapi.CloudFormationStackArtifact {\n  return testStack({\n    stackName: STACK_NAME,\n    ...testStackArtifact,\n  });\n}\n\nexport function pushStackResourceSummaries(...items: CloudFormation.StackResourceSummary[]) {\n  currentCfnStackResources.push(...items);\n}\n\nexport function setCurrentCfnStackTemplate(template: Template) {\n  const templateDeepCopy = JSON.parse(JSON.stringify(template)); // deep copy the template, so our tests can mutate one template instead of creating two\n  currentCfnStack.setTemplate(templateDeepCopy);\n}\n\nexport function stackSummaryOf(logicalId: string, resourceType: string, physicalResourceId: string): CloudFormation.StackResourceSummary {\n  return {\n    LogicalResourceId: logicalId,\n    PhysicalResourceId: physicalResourceId,\n    ResourceType: resourceType,\n    ResourceStatus: 'CREATE_COMPLETE',\n    LastUpdatedTimestamp: new Date(),\n  };\n}\n\nexport class HotswapMockSdkProvider {\n  public readonly mockSdkProvider: MockSdkProvider;\n\n  constructor() {\n    this.mockSdkProvider = new MockSdkProvider({ realSdk: false });\n\n    this.mockSdkProvider.stubCloudFormation({\n      listStackResources: ({ StackName: stackName }) => {\n        if (stackName !== STACK_NAME) {\n          throw new Error(`Expected Stack name in listStackResources() call to be: '${STACK_NAME}', but received: ${stackName}'`);\n        }\n        return {\n          StackResourceSummaries: currentCfnStackResources,\n        };\n      },\n    });\n  }\n\n  public setUpdateStateMachineMock(\n    mockUpdateMachineDefinition: (input: stepfunctions.UpdateStateMachineInput) => stepfunctions.UpdateStateMachineOutput,\n  ) {\n    this.mockSdkProvider.stubStepFunctions({\n      updateStateMachine: mockUpdateMachineDefinition,\n    });\n  }\n\n  public stubLambda(\n    stubs: SyncHandlerSubsetOf<AWS.Lambda>,\n    serviceStubs?: SyncHandlerSubsetOf<AWS.Service>,\n    additionalProperties: { [key: string]: any } = {},\n  ): void {\n    this.mockSdkProvider.stubLambda(stubs, {\n      api: {\n        waiters: {},\n      },\n      makeRequest() {\n        return {\n          promise: () => Promise.resolve({}),\n          response: {},\n          addListeners: () => {},\n        };\n      },\n      ...serviceStubs,\n      ...additionalProperties,\n    });\n  }\n\n  public getLambdaApiWaiters(): { [key: string]: any } {\n    return (this.mockSdkProvider.sdk.lambda() as any).api.waiters;\n  }\n\n  public setUpdateProjectMock(mockUpdateProject: (input: codebuild.UpdateProjectInput) => codebuild.UpdateProjectOutput) {\n    this.mockSdkProvider.stubCodeBuild({\n      updateProject: mockUpdateProject,\n    });\n  }\n\n  public stubAppSync(stubs: SyncHandlerSubsetOf<AWS.AppSync>) {\n    this.mockSdkProvider.stubAppSync(stubs);\n  }\n\n  public setInvokeLambdaMock(mockInvokeLambda: (input: lambda.InvocationRequest) => lambda.InvocationResponse) {\n    this.mockSdkProvider.stubLambda({\n      invoke: mockInvokeLambda,\n    });\n  }\n\n  public stubEcs(stubs: SyncHandlerSubsetOf<AWS.ECS>, additionalProperties: { [key: string]: any } = {}): void {\n    this.mockSdkProvider.stubEcs(stubs, additionalProperties);\n  }\n\n  public stubGetEndpointSuffix(stub: () => string) {\n    this.mockSdkProvider.stubGetEndpointSuffix(stub);\n  }\n\n  public tryHotswapDeployment(\n    stackArtifact: cxapi.CloudFormationStackArtifact,\n    assetParams: { [key: string]: string } = {},\n  ): Promise<DeployStackResult | undefined> {\n    return deployments.tryHotswapDeployment(this.mockSdkProvider, assetParams, currentCfnStack, stackArtifact);\n  }\n}\n"]}
153
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-test-setup.js","sourceRoot":"","sources":["hotswap-test-setup.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport * as AWS from 'aws-sdk';\nimport * as codebuild from 'aws-sdk/clients/codebuild';\nimport * as lambda from 'aws-sdk/clients/lambda';\nimport * as stepfunctions from 'aws-sdk/clients/stepfunctions';\nimport { DeployStackResult } from '../../../lib/api';\nimport * as deployments from '../../../lib/api/hotswap-deployments';\nimport { CloudFormationStack, Template } from '../../../lib/api/util/cloudformation';\nimport { testStack, TestStackArtifact } from '../../util';\nimport { MockSdkProvider, SyncHandlerSubsetOf } from '../../util/mock-sdk';\nimport { FakeCloudformationStack } from '../fake-cloudformation-stack';\n\nconst STACK_NAME = 'withouterrors';\nexport const STACK_ID = 'stackId';\n\nlet hotswapMockSdkProvider: HotswapMockSdkProvider;\nlet currentCfnStack: FakeCloudformationStack;\nconst currentCfnStackResources: AWS.CloudFormation.StackResourceSummary[] = [];\nlet stackTemplates: { [stackName: string]: any };\nlet currentNestedCfnStackResources: { [stackName: string]: AWS.CloudFormation.StackResourceSummary[] };\n\nexport function setupHotswapTests(): HotswapMockSdkProvider {\n  jest.resetAllMocks();\n  // clear the array\n  currentCfnStackResources.splice(0);\n  hotswapMockSdkProvider = new HotswapMockSdkProvider();\n  currentCfnStack = new FakeCloudformationStack({\n    stackName: STACK_NAME,\n    stackId: STACK_ID,\n  });\n  CloudFormationStack.lookup = async (_: AWS.CloudFormation, _stackName: string) => {\n    return currentCfnStack;\n  };\n\n  return hotswapMockSdkProvider;\n}\n\nexport function setupHotswapNestedStackTests(rootStackName: string) {\n  jest.resetAllMocks();\n  currentNestedCfnStackResources = {};\n  hotswapMockSdkProvider = new HotswapMockSdkProvider(rootStackName);\n  currentCfnStack = new FakeCloudformationStack({\n    stackName: rootStackName,\n    stackId: STACK_ID,\n  });\n  stackTemplates = {};\n  CloudFormationStack.lookup = async (_: AWS.CloudFormation, stackName: string) => {\n    currentCfnStack.template = async () => stackTemplates[stackName];\n    return currentCfnStack;\n  };\n\n  return hotswapMockSdkProvider;\n}\n\nexport function cdkStackArtifactOf(testStackArtifact: Partial<TestStackArtifact> = {}): cxapi.CloudFormationStackArtifact {\n  return testStack({\n    stackName: STACK_NAME,\n    ...testStackArtifact,\n  });\n}\n\nexport function pushStackResourceSummaries(...items: AWS.CloudFormation.StackResourceSummary[]) {\n  currentCfnStackResources.push(...items);\n}\n\nexport function pushNestedStackResourceSummaries(stackName: string, ...items: AWS.CloudFormation.StackResourceSummary[]) {\n  if (!currentNestedCfnStackResources[stackName]) {\n    currentNestedCfnStackResources[stackName] = [];\n  }\n  currentNestedCfnStackResources[stackName].push(...items);\n}\n\nexport function setCurrentCfnStackTemplate(template: Template) {\n  const templateDeepCopy = JSON.parse(JSON.stringify(template)); // deep copy the template, so our tests can mutate one template instead of creating two\n  currentCfnStack.setTemplate(templateDeepCopy);\n}\n\nexport function addTemplateToCloudFormationLookupMock(stackArtifact: cxapi.CloudFormationStackArtifact) {\n  const templateDeepCopy = JSON.parse(JSON.stringify(stackArtifact.template)); // deep copy the template, so our tests can mutate one template instead of creating two\n  stackTemplates[stackArtifact.stackName] = templateDeepCopy;\n}\n\nexport function stackSummaryOf(logicalId: string, resourceType: string, physicalResourceId: string): AWS.CloudFormation.StackResourceSummary {\n  return {\n    LogicalResourceId: logicalId,\n    PhysicalResourceId: physicalResourceId,\n    ResourceType: resourceType,\n    ResourceStatus: 'CREATE_COMPLETE',\n    LastUpdatedTimestamp: new Date(),\n  };\n}\n\nexport class HotswapMockSdkProvider {\n  public readonly mockSdkProvider: MockSdkProvider;\n\n  constructor(rootStackName?: string) {\n    this.mockSdkProvider = new MockSdkProvider({ realSdk: false });\n\n    this.mockSdkProvider.stubCloudFormation({\n      listStackResources: ({ StackName: stackName }) => {\n        if (rootStackName) {\n          const knownStackNames = Object.keys(currentNestedCfnStackResources);\n          if (stackName !== rootStackName && !knownStackNames.includes(stackName)) {\n            throw new Error(`Expected Stack name in listStackResources() call to be a member of ['${rootStackName}, ${knownStackNames}'], but received: '${stackName}'`);\n          }\n        } else if (stackName !== STACK_NAME) {\n          throw new Error(`Expected Stack name in listStackResources() call to be: '${STACK_NAME}', but received: '${stackName}'`);\n        }\n        return {\n          StackResourceSummaries: rootStackName ? currentNestedCfnStackResources[stackName] : currentCfnStackResources,\n        };\n      },\n    });\n  }\n\n  public setUpdateStateMachineMock(\n    mockUpdateMachineDefinition: (input: stepfunctions.UpdateStateMachineInput) => stepfunctions.UpdateStateMachineOutput,\n  ) {\n    this.mockSdkProvider.stubStepFunctions({\n      updateStateMachine: mockUpdateMachineDefinition,\n    });\n  }\n\n  public stubLambda(\n    stubs: SyncHandlerSubsetOf<AWS.Lambda>,\n    serviceStubs?: SyncHandlerSubsetOf<AWS.Service>,\n    additionalProperties: { [key: string]: any } = {},\n  ): void {\n    this.mockSdkProvider.stubLambda(stubs, {\n      api: {\n        waiters: {},\n      },\n      makeRequest() {\n        return {\n          promise: () => Promise.resolve({}),\n          response: {},\n          addListeners: () => {},\n        };\n      },\n      ...serviceStubs,\n      ...additionalProperties,\n    });\n  }\n\n  public getLambdaApiWaiters(): { [key: string]: any } {\n    return (this.mockSdkProvider.sdk.lambda() as any).api.waiters;\n  }\n\n  public setUpdateProjectMock(mockUpdateProject: (input: codebuild.UpdateProjectInput) => codebuild.UpdateProjectOutput) {\n    this.mockSdkProvider.stubCodeBuild({\n      updateProject: mockUpdateProject,\n    });\n  }\n\n  public stubAppSync(stubs: SyncHandlerSubsetOf<AWS.AppSync>) {\n    this.mockSdkProvider.stubAppSync(stubs);\n  }\n\n  public setInvokeLambdaMock(mockInvokeLambda: (input: lambda.InvocationRequest) => lambda.InvocationResponse) {\n    this.mockSdkProvider.stubLambda({\n      invoke: mockInvokeLambda,\n    });\n  }\n\n  public stubEcs(stubs: SyncHandlerSubsetOf<AWS.ECS>, additionalProperties: { [key: string]: any } = {}): void {\n    this.mockSdkProvider.stubEcs(stubs, additionalProperties);\n  }\n\n  public stubGetEndpointSuffix(stub: () => string) {\n    this.mockSdkProvider.stubGetEndpointSuffix(stub);\n  }\n\n  public tryHotswapDeployment(\n    stackArtifact: cxapi.CloudFormationStackArtifact,\n    assetParams: { [key: string]: string } = {},\n  ): Promise<DeployStackResult | undefined> {\n    return deployments.tryHotswapDeployment(this.mockSdkProvider, assetParams, currentCfnStack, stackArtifact);\n  }\n}\n"]}
@@ -0,0 +1 @@
1
+ export {};