aws-cdk 2.20.0 → 2.22.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 (33) hide show
  1. package/THIRD_PARTY_LICENSES +8 -28
  2. package/build-info.json +2 -2
  3. package/does-not-exist.json +1 -1
  4. package/lib/api/aws-auth/account-cache.d.ts +1 -1
  5. package/lib/api/aws-auth/account-cache.js +2 -2
  6. package/lib/api/deploy-stack.js +18 -6
  7. package/lib/api/util/cloudformation/stack-activity-monitor.d.ts +5 -0
  8. package/lib/api/util/cloudformation/stack-activity-monitor.js +19 -3
  9. package/lib/api/util/cloudformation.d.ts +6 -3
  10. package/lib/api/util/cloudformation.js +27 -8
  11. package/lib/api/util/display.d.ts +2 -1
  12. package/lib/api/util/display.js +14 -19
  13. package/lib/cli.js +2 -3
  14. package/lib/index.js +8351 -5344
  15. package/lib/init-templates/v1/app/csharp/README.md +1 -1
  16. package/lib/init-templates/v1/app/fsharp/README.md +1 -1
  17. package/lib/init-templates/v1/app/go/README.md +1 -1
  18. package/lib/init-templates/v1/app/java/README.md +1 -1
  19. package/lib/init-templates/v1/app/javascript/README.md +1 -1
  20. package/lib/init-templates/v1/app/python/README.template.md +1 -1
  21. package/lib/init-templates/v1/app/typescript/README.md +1 -1
  22. package/lib/init-templates/v2/app/csharp/README.md +1 -1
  23. package/lib/init-templates/v2/app/fsharp/README.md +1 -1
  24. package/lib/init-templates/v2/app/go/README.md +1 -1
  25. package/lib/init-templates/v2/app/java/README.md +1 -1
  26. package/lib/init-templates/v2/app/javascript/README.md +1 -1
  27. package/lib/init-templates/v2/app/python/README.template.md +1 -1
  28. package/lib/init-templates/v2/app/typescript/README.md +1 -1
  29. package/lib/notices.d.ts +1 -0
  30. package/lib/notices.js +43 -21
  31. package/package.json +11 -11
  32. package/test/api/util/display.test.js +21 -1
  33. package/test/notices.test.js +35 -21
@@ -1,6 +1,6 @@
1
1
  # Welcome to your CDK C# project!
2
2
 
3
- This is a blank project for C# development with CDK.
3
+ This is a blank project for CDK development with C#.
4
4
 
5
5
  The `cdk.json` file tells the CDK Toolkit how to execute your app.
6
6
 
@@ -1,6 +1,6 @@
1
1
  ## Welcome to your CDK F# project!
2
2
 
3
- This is a blank project for F# development with CDK.
3
+ This is a blank project for CDK development with F#.
4
4
 
5
5
  The `cdk.json` file tells the CDK Toolkit how to execute your app.
6
6
 
@@ -1,6 +1,6 @@
1
1
  # Welcome to your CDK Go project!
2
2
 
3
- This is a blank project for Go development with CDK.
3
+ This is a blank project for CDK development with Go.
4
4
 
5
5
  **NOTICE**: Go support is still in Developer Preview. This implies that APIs may
6
6
  change while we address early feedback from the community. We would love to hear
@@ -1,6 +1,6 @@
1
1
  # Welcome to your CDK Java project!
2
2
 
3
- This is a blank project for Java development with CDK.
3
+ This is a blank project for CDK development with Java.
4
4
 
5
5
  The `cdk.json` file tells the CDK Toolkit how to execute your app.
6
6
 
@@ -1,6 +1,6 @@
1
1
  # Welcome to your CDK JavaScript project
2
2
 
3
- This is a blank project for JavaScript development with CDK.
3
+ This is a blank project for CDK development with JavaScript.
4
4
 
5
5
  The `cdk.json` file tells the CDK Toolkit how to execute your app. The build step is not required when using JavaScript.
6
6
 
@@ -1,7 +1,7 @@
1
1
 
2
2
  # Welcome to your CDK Python project!
3
3
 
4
- This is a blank project for Python development with CDK.
4
+ This is a blank project for CDK development with Python.
5
5
 
6
6
  The `cdk.json` file tells the CDK Toolkit how to execute your app.
7
7
 
@@ -1,6 +1,6 @@
1
1
  # Welcome to your CDK TypeScript project
2
2
 
3
- This is a blank project for TypeScript development with CDK.
3
+ This is a blank project for CDK development with TypeScript.
4
4
 
5
5
  The `cdk.json` file tells the CDK Toolkit how to execute your app.
6
6
 
@@ -1,6 +1,6 @@
1
1
  # Welcome to your CDK C# project!
2
2
 
3
- This is a blank project for C# development with CDK.
3
+ This is a blank project for CDK development with C#.
4
4
 
5
5
  The `cdk.json` file tells the CDK Toolkit how to execute your app.
6
6
 
@@ -1,6 +1,6 @@
1
1
  ## Welcome to your CDK F# project!
2
2
 
3
- This is a blank project for F# development with CDK.
3
+ This is a blank project for CDK development with F#.
4
4
 
5
5
  The `cdk.json` file tells the CDK Toolkit how to execute your app.
6
6
 
@@ -1,6 +1,6 @@
1
1
  # Welcome to your CDK Go project!
2
2
 
3
- This is a blank project for Go development with CDK.
3
+ This is a blank project for CDK development with Go.
4
4
 
5
5
  **NOTICE**: Go support is still in Developer Preview. This implies that APIs may
6
6
  change while we address early feedback from the community. We would love to hear
@@ -1,6 +1,6 @@
1
1
  # Welcome to your CDK Java project!
2
2
 
3
- This is a blank project for Java development with CDK.
3
+ This is a blank project for CDK development with Java.
4
4
 
5
5
  The `cdk.json` file tells the CDK Toolkit how to execute your app.
6
6
 
@@ -1,6 +1,6 @@
1
1
  # Welcome to your CDK JavaScript project
2
2
 
3
- This is a blank project for JavaScript development with CDK.
3
+ This is a blank project for CDK development with JavaScript.
4
4
 
5
5
  The `cdk.json` file tells the CDK Toolkit how to execute your app. The build step is not required when using JavaScript.
6
6
 
@@ -1,7 +1,7 @@
1
1
 
2
2
  # Welcome to your CDK Python project!
3
3
 
4
- This is a blank project for Python development with CDK.
4
+ This is a blank project for CDK development with Python.
5
5
 
6
6
  The `cdk.json` file tells the CDK Toolkit how to execute your app.
7
7
 
@@ -1,6 +1,6 @@
1
1
  # Welcome to your CDK TypeScript project
2
2
 
3
- This is a blank project for TypeScript development with CDK.
3
+ This is a blank project for CDK development with TypeScript.
4
4
 
5
5
  The `cdk.json` file tells the CDK Toolkit how to execute your app.
6
6
 
package/lib/notices.d.ts CHANGED
@@ -50,6 +50,7 @@ export declare class CachedDataSource implements NoticeDataSource {
50
50
  private readonly skipCache?;
51
51
  constructor(fileName: string, dataSource: NoticeDataSource, skipCache?: boolean | undefined);
52
52
  fetch(): Promise<Notice[]>;
53
+ private fetchInner;
53
54
  private load;
54
55
  private save;
55
56
  }
package/lib/notices.js CHANGED
@@ -62,8 +62,7 @@ exports.formatNotices = formatNotices;
62
62
  class WebsiteNoticeDataSource {
63
63
  fetch() {
64
64
  const timeout = 3000;
65
- return new Promise((resolve) => {
66
- setTimeout(() => resolve([]), timeout);
65
+ return new Promise((resolve, reject) => {
67
66
  try {
68
67
  const req = https.get('https://cli.cdk.dev-tools.aws.dev/notices.json', { timeout }, res => {
69
68
  if (res.statusCode === 200) {
@@ -75,38 +74,48 @@ class WebsiteNoticeDataSource {
75
74
  res.on('end', () => {
76
75
  try {
77
76
  const data = JSON.parse(rawData).notices;
77
+ if (!data) {
78
+ throw new Error("'notices' key is missing");
79
+ }
80
+ logging_1.debug('Notices refreshed');
78
81
  resolve(data !== null && data !== void 0 ? data : []);
79
82
  }
80
83
  catch (e) {
81
- logging_1.debug(`Failed to parse notices: ${e}`);
82
- resolve([]);
84
+ reject(new Error(`Failed to parse notices: ${e.message}`));
83
85
  }
84
86
  });
85
87
  res.on('error', e => {
86
- logging_1.debug(`Failed to fetch notices: ${e}`);
87
- resolve([]);
88
+ reject(new Error(`Failed to fetch notices: ${e.message}`));
88
89
  });
89
90
  }
90
91
  else {
91
- logging_1.debug(`Failed to fetch notices. Status code: ${res.statusCode}`);
92
- resolve([]);
92
+ reject(new Error(`Failed to fetch notices. Status code: ${res.statusCode}`));
93
93
  }
94
94
  });
95
- req.on('error', e => {
96
- logging_1.debug(`Error on request: ${e}`);
97
- resolve([]);
95
+ req.on('error', reject);
96
+ req.on('timeout', () => {
97
+ // The 'timeout' event doesn't stop anything by itself, it just
98
+ // notifies that it has been long time since we saw bytes.
99
+ // In our case, we want to give up.
100
+ req.destroy(new Error('Request timed out'));
98
101
  });
99
- req.on('timeout', _ => resolve([]));
102
+ // It's not like I don't *trust* the 'timeout' event... but I don't trust it.
103
+ // Add a backup timer that will destroy the request after all.
104
+ // (This is at least necessary to make the tests pass, but that's probably because of 'nock'.
105
+ // It's not clear whether users will hit this).
106
+ setTimeout(() => {
107
+ req.destroy(new Error('Request timed out. You should never see this message; if you do, please let us know at https://github.com/aws/aws-cdk/issues'));
108
+ }, timeout + 200);
100
109
  }
101
110
  catch (e) {
102
- logging_1.debug(`HTTPS 'get' call threw an error: ${e}`);
103
- resolve([]);
111
+ reject(new Error(`HTTPS 'get' call threw an error: ${e.message}`));
104
112
  }
105
113
  });
106
114
  }
107
115
  }
108
116
  exports.WebsiteNoticeDataSource = WebsiteNoticeDataSource;
109
- const TIME_TO_LIVE = 60 * 60 * 1000; // 1 hour
117
+ const TIME_TO_LIVE_SUCCESS = 60 * 60 * 1000; // 1 hour
118
+ const TIME_TO_LIVE_ERROR = 1 * 60 * 1000; // 1 minute
110
119
  class CachedDataSource {
111
120
  constructor(fileName, dataSource, skipCache) {
112
121
  this.fileName = fileName;
@@ -119,17 +128,30 @@ class CachedDataSource {
119
128
  const data = cachedData.notices;
120
129
  const expiration = (_a = cachedData.expiration) !== null && _a !== void 0 ? _a : 0;
121
130
  if (Date.now() > expiration || this.skipCache) {
122
- const freshData = {
123
- expiration: Date.now() + TIME_TO_LIVE,
124
- notices: await this.dataSource.fetch(),
125
- };
131
+ const freshData = await this.fetchInner();
126
132
  await this.save(freshData);
127
133
  return freshData.notices;
128
134
  }
129
135
  else {
136
+ logging_1.debug(`Reading cached notices from ${this.fileName}`);
130
137
  return data;
131
138
  }
132
139
  }
140
+ async fetchInner() {
141
+ try {
142
+ return {
143
+ expiration: Date.now() + TIME_TO_LIVE_SUCCESS,
144
+ notices: await this.dataSource.fetch(),
145
+ };
146
+ }
147
+ catch (e) {
148
+ logging_1.debug(`Could not refresh notices: ${e}`);
149
+ return {
150
+ expiration: Date.now() + TIME_TO_LIVE_ERROR,
151
+ notices: [],
152
+ };
153
+ }
154
+ }
133
155
  async load() {
134
156
  const defaultValue = {
135
157
  expiration: 0,
@@ -260,7 +282,7 @@ function loadTree(outdir) {
260
282
  return fs.readJSONSync(path.join(outdir, 'tree.json'));
261
283
  }
262
284
  catch (e) {
263
- logging_1.debug(`Failed to get tree.json file: ${e}`);
285
+ logging_1.trace(`Failed to get tree.json file: ${e}. Proceeding with empty tree.`);
264
286
  return {};
265
287
  }
266
288
  }
@@ -278,4 +300,4 @@ function some(node, predicate) {
278
300
  return false;
279
301
  }
280
302
  }
281
- //# 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 filteredNotices = filterNotices(data, {\n    outdir: props.outdir,\n    acknowledgedIssueNumbers: new Set(props.acknowledgedIssueNumbers),\n  });\n\n  if (filteredNotices.length > 0) {\n    const individualMessages = formatNotices(filteredNotices);\n    return finalMessage(individualMessages, filteredNotices[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      setTimeout(() => resolve([]), timeout);\n      try {\n        const req = https.get('https://cli.cdk.dev-tools.aws.dev/notices.json',\n          { timeout },\n          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        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"]}
303
+ //# 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, trace } 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 filteredNotices = filterNotices(data, {\n    outdir: props.outdir,\n    acknowledgedIssueNumbers: new Set(props.acknowledgedIssueNumbers),\n  });\n\n  if (filteredNotices.length > 0) {\n    const individualMessages = formatNotices(filteredNotices);\n    return finalMessage(individualMessages, filteredNotices[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    return new Promise((resolve, reject) => {\n      try {\n        const req = https.get('https://cli.cdk.dev-tools.aws.dev/notices.json',\n          { timeout },\n          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                  if (!data) {\n                    throw new Error(\"'notices' key is missing\");\n                  }\n                  debug('Notices refreshed');\n                  resolve(data ?? []);\n                } catch (e) {\n                  reject(new Error(`Failed to parse notices: ${e.message}`));\n                }\n              });\n              res.on('error', e => {\n                reject(new Error(`Failed to fetch notices: ${e.message}`));\n              });\n            } else {\n              reject(new Error(`Failed to fetch notices. Status code: ${res.statusCode}`));\n            }\n          });\n        req.on('error', reject);\n        req.on('timeout', () => {\n          // The 'timeout' event doesn't stop anything by itself, it just\n          // notifies that it has been long time since we saw bytes.\n          // In our case, we want to give up.\n          req.destroy(new Error('Request timed out'));\n        });\n\n        // It's not like I don't *trust* the 'timeout' event... but I don't trust it.\n        // Add a backup timer that will destroy the request after all.\n        // (This is at least necessary to make the tests pass, but that's probably because of 'nock'.\n        // It's not clear whether users will hit this).\n        setTimeout(() => {\n          req.destroy(new Error('Request timed out. You should never see this message; if you do, please let us know at https://github.com/aws/aws-cdk/issues'));\n        }, timeout + 200);\n      } catch (e) {\n        reject(new Error(`HTTPS 'get' call threw an error: ${e.message}`));\n      }\n    });\n  }\n}\n\ninterface CachedNotices {\n  expiration: number,\n  notices: Notice[],\n}\n\nconst TIME_TO_LIVE_SUCCESS = 60 * 60 * 1000; // 1 hour\nconst TIME_TO_LIVE_ERROR = 1 * 60 * 1000; // 1 minute\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 = await this.fetchInner();\n      await this.save(freshData);\n      return freshData.notices;\n    } else {\n      debug(`Reading cached notices from ${this.fileName}`);\n      return data;\n    }\n  }\n\n  private async fetchInner(): Promise<CachedNotices> {\n    try {\n      return {\n        expiration: Date.now() + TIME_TO_LIVE_SUCCESS,\n        notices: await this.dataSource.fetch(),\n      };\n    } catch (e) {\n      debug(`Could not refresh notices: ${e}`);\n      return {\n        expiration: Date.now() + TIME_TO_LIVE_ERROR,\n        notices: [],\n      };\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    trace(`Failed to get tree.json file: ${e}. Proceeding with empty tree.`);\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.20.0",
4
+ "version": "2.22.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.20.0",
61
+ "@aws-cdk/core": "2.22.0",
62
62
  "@octokit/rest": "^18.12.0",
63
63
  "@types/archiver": "^5.3.1",
64
64
  "@types/fs-extra": "^8.1.2",
@@ -75,28 +75,28 @@
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.20.0",
78
+ "@aws-cdk/cdk-build-tools": "2.22.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.20.0",
85
+ "@aws-cdk/pkglint": "2.22.0",
86
86
  "sinon": "^9.2.4",
87
87
  "ts-jest": "^27.1.4",
88
88
  "ts-mock-imports": "^1.3.8",
89
89
  "xml-js": "^1.6.11",
90
90
  "axios": "^0.26.1",
91
- "@aws-cdk/cloud-assembly-schema": "2.20.0",
92
- "@aws-cdk/cloudformation-diff": "2.20.0",
93
- "@aws-cdk/cx-api": "2.20.0",
94
- "@aws-cdk/region-info": "2.20.0",
95
- "@jsii/check-node": "1.55.1",
91
+ "@aws-cdk/cloud-assembly-schema": "2.22.0",
92
+ "@aws-cdk/cloudformation-diff": "2.22.0",
93
+ "@aws-cdk/cx-api": "2.22.0",
94
+ "@aws-cdk/region-info": "2.22.0",
95
+ "@jsii/check-node": "1.56.0",
96
96
  "archiver": "^5.3.0",
97
97
  "aws-sdk": "^2.1093.0",
98
98
  "camelcase": "^6.3.0",
99
- "cdk-assets": "2.20.0",
99
+ "cdk-assets": "2.22.0",
100
100
  "chokidar": "^3.5.3",
101
101
  "chalk": "^4",
102
102
  "decamelize": "^5.0.1",
@@ -106,7 +106,7 @@
106
106
  "minimatch": ">=3.1",
107
107
  "promptly": "^3.2.0",
108
108
  "proxy-agent": "^5.0.0",
109
- "semver": "^7.3.5",
109
+ "semver": "^7.3.6",
110
110
  "source-map-support": "^0.5.21",
111
111
  "strip-ansi": "^6.0.1",
112
112
  "table": "^6.8.0",
@@ -30,5 +30,25 @@ describe('Rewritable Block Tests', () => {
30
30
  });
31
31
  expect(output.length).toEqual(6);
32
32
  });
33
+ test('display accounts for newlines in output', () => {
34
+ const output = console_listener_1.stderr.inspectSync(() => {
35
+ block.displayLines(['before\nafter']);
36
+ });
37
+ expect(output.length).toEqual(3); // cursorup + 2 lines
38
+ });
39
+ test('removeEmptyLines only removes trailing lines', () => {
40
+ console_listener_1.stderr.inspectSync(() => {
41
+ block.displayLines(Array.from(Array(5).keys()).map(x => `${x}`));
42
+ });
43
+ console_listener_1.stderr.inspectSync(() => {
44
+ // Leaves 3 empty lines
45
+ block.displayLines(Array.from(Array(2).keys()).map(x => `${x}`));
46
+ });
47
+ const output = console_listener_1.stderr.inspectSync(() => {
48
+ block.removeEmptyLines();
49
+ });
50
+ const expectedEmptyLines = 3;
51
+ expect(JSON.stringify(output)).toEqual(JSON.stringify([`\u001b[${expectedEmptyLines}A`]));
52
+ });
33
53
  });
34
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGlzcGxheS50ZXN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiZGlzcGxheS50ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQUEsMkRBQWdFO0FBQ2hFLDBEQUE2QztBQUc3QyxRQUFRLENBQUMsd0JBQXdCLEVBQUUsR0FBRyxFQUFFO0lBQ3RDLElBQUksS0FBc0IsQ0FBQztJQUMzQixVQUFVLENBQUMsR0FBRyxFQUFFO1FBQ2QsS0FBSyxHQUFHLElBQUkseUJBQWUsQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDNUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxJQUFJLEdBQUcsRUFBRSxDQUFDO0lBQzNCLENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLG1GQUFtRixFQUFFLEdBQUcsRUFBRTtRQUM3RixNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDO1FBQ3pFLE1BQU0sTUFBTSxHQUFHLHlCQUFNLENBQUMsV0FBVyxDQUFDLEdBQUcsRUFBRTtZQUNyQyxLQUFLLENBQUMsWUFBWSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzVCLENBQUMsQ0FBQyxDQUFDO1FBRUgsTUFBTSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLE1BQU8sQ0FBQyxDQUFDO0lBQy9DLENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLDJGQUEyRixFQUFFLEdBQUcsRUFBRTtRQUNyRyxNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDO1FBQ3hFLE1BQU0sTUFBTSxHQUFHLHlCQUFNLENBQUMsV0FBVyxDQUFDLEdBQUcsRUFBRTtZQUNyQyxLQUFLLENBQUMsWUFBWSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzVCLENBQUMsQ0FBQyxDQUFDO1FBRUgsTUFBTSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDLENBQUM7SUFDcEMsQ0FBQyxDQUFDLENBQUM7SUFFSCxJQUFJLENBQUMsOEVBQThFLEVBQUUsR0FBRyxFQUFFO1FBQ3hGLE1BQU0sS0FBSyxHQUFHLEtBQUssQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksRUFBRSxDQUFDLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUM7UUFDdkUsT0FBTyxDQUFDLE1BQU0sQ0FBQyxJQUFJLEdBQUcsU0FBUyxDQUFDO1FBQ2hDLE1BQU0sTUFBTSxHQUFHLHlCQUFNLENBQUMsV0FBVyxDQUFDLEdBQUcsRUFBRTtZQUNyQyxLQUFLLENBQUMsWUFBWSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzVCLENBQUMsQ0FBQyxDQUFDO1FBRUgsTUFBTSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDbkMsQ0FBQyxDQUFDLENBQUM7QUFDTCxDQUFDLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFJld3JpdGFibGVCbG9jayB9IGZyb20gJy4uLy4uLy4uL2xpYi9hcGkvdXRpbC9kaXNwbGF5JztcbmltcG9ydCB7IHN0ZGVyciB9IGZyb20gJy4uL2NvbnNvbGUtbGlzdGVuZXInO1xuXG5cbmRlc2NyaWJlKCdSZXdyaXRhYmxlIEJsb2NrIFRlc3RzJywgKCkgPT4ge1xuICBsZXQgYmxvY2s6IFJld3JpdGFibGVCbG9jaztcbiAgYmVmb3JlRWFjaCgoKSA9PiB7XG4gICAgYmxvY2sgPSBuZXcgUmV3cml0YWJsZUJsb2NrKHByb2Nlc3Muc3RkZXJyKTtcbiAgICBwcm9jZXNzLnN0ZGVyci5yb3dzID0gODA7XG4gIH0pO1xuXG4gIHRlc3QoJ2Rpc3BsYXlMaW5lcyB3cml0ZXMgbWF4aW11bSBsaW5lcyBiYXNlZCBvbiByb3dzIGlmIHRoZXJlIGFyZSBtb3JlIGxpbmVzIHRoYW4gcm93cycsICgpID0+IHtcbiAgICBjb25zdCBsaW5lcyA9IEFycmF5LmZyb20oQXJyYXkoMTAwKS5rZXlzKCkpLm1hcChsaW5lID0+IGxpbmUudG9TdHJpbmcoKSk7XG4gICAgY29uc3Qgb3V0cHV0ID0gc3RkZXJyLmluc3BlY3RTeW5jKCgpID0+IHtcbiAgICAgIGJsb2NrLmRpc3BsYXlMaW5lcyhsaW5lcyk7XG4gICAgfSk7XG5cbiAgICBleHBlY3Qob3V0cHV0Lmxlbmd0aCkudG9FcXVhbChibG9jay5oZWlnaHQhKTtcbiAgfSk7XG5cbiAgdGVzdCgnZGlzcGxheUxpbmVzIHdyaXRlcyBtYXhpbXVtIGxpbmVzIGJhc2VkIG9uIGxpbmVzIGxlbmd0aCBpZiB0aGVyZSBhcmUgbGVzcyBsaW5lcyB0aGFuIHJvd3MnLCAoKSA9PiB7XG4gICAgY29uc3QgbGluZXMgPSBBcnJheS5mcm9tKEFycmF5KDQ1KS5rZXlzKCkpLm1hcChsaW5lID0+IGxpbmUudG9TdHJpbmcoKSk7XG4gICAgY29uc3Qgb3V0cHV0ID0gc3RkZXJyLmluc3BlY3RTeW5jKCgpID0+IHtcbiAgICAgIGJsb2NrLmRpc3BsYXlMaW5lcyhsaW5lcyk7XG4gICAgfSk7XG5cbiAgICBleHBlY3Qob3V0cHV0Lmxlbmd0aCkudG9FcXVhbCg0Nik7XG4gIH0pO1xuXG4gIHRlc3QoJ2Rpc3BsYXlMaW5lcyB3cml0ZXMgbWF4aW11bSBsaW5lcyBiYXNlZCBvbiBsaW5lcyBsZW5ndGggaWYgcm93cyBpcyB1bmRlZmluZWQnLCAoKSA9PiB7XG4gICAgY29uc3QgbGluZXMgPSBBcnJheS5mcm9tKEFycmF5KDUpLmtleXMoKSkubWFwKGxpbmUgPT4gbGluZS50b1N0cmluZygpKTtcbiAgICBwcm9jZXNzLnN0ZGVyci5yb3dzID0gdW5kZWZpbmVkO1xuICAgIGNvbnN0IG91dHB1dCA9IHN0ZGVyci5pbnNwZWN0U3luYygoKSA9PiB7XG4gICAgICBibG9jay5kaXNwbGF5TGluZXMobGluZXMpO1xuICAgIH0pO1xuXG4gICAgZXhwZWN0KG91dHB1dC5sZW5ndGgpLnRvRXF1YWwoNik7XG4gIH0pO1xufSk7Il19
54
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGlzcGxheS50ZXN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiZGlzcGxheS50ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQUEsMkRBQWdFO0FBQ2hFLDBEQUE2QztBQUc3QyxRQUFRLENBQUMsd0JBQXdCLEVBQUUsR0FBRyxFQUFFO0lBQ3RDLElBQUksS0FBc0IsQ0FBQztJQUMzQixVQUFVLENBQUMsR0FBRyxFQUFFO1FBQ2QsS0FBSyxHQUFHLElBQUkseUJBQWUsQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDNUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxJQUFJLEdBQUcsRUFBRSxDQUFDO0lBQzNCLENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLG1GQUFtRixFQUFFLEdBQUcsRUFBRTtRQUM3RixNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDO1FBQ3pFLE1BQU0sTUFBTSxHQUFHLHlCQUFNLENBQUMsV0FBVyxDQUFDLEdBQUcsRUFBRTtZQUNyQyxLQUFLLENBQUMsWUFBWSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzVCLENBQUMsQ0FBQyxDQUFDO1FBRUgsTUFBTSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLE1BQU8sQ0FBQyxDQUFDO0lBQy9DLENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLDJGQUEyRixFQUFFLEdBQUcsRUFBRTtRQUNyRyxNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDO1FBQ3hFLE1BQU0sTUFBTSxHQUFHLHlCQUFNLENBQUMsV0FBVyxDQUFDLEdBQUcsRUFBRTtZQUNyQyxLQUFLLENBQUMsWUFBWSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzVCLENBQUMsQ0FBQyxDQUFDO1FBRUgsTUFBTSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDLENBQUM7SUFDcEMsQ0FBQyxDQUFDLENBQUM7SUFFSCxJQUFJLENBQUMsOEVBQThFLEVBQUUsR0FBRyxFQUFFO1FBQ3hGLE1BQU0sS0FBSyxHQUFHLEtBQUssQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksRUFBRSxDQUFDLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUM7UUFDdkUsT0FBTyxDQUFDLE1BQU0sQ0FBQyxJQUFJLEdBQUcsU0FBUyxDQUFDO1FBQ2hDLE1BQU0sTUFBTSxHQUFHLHlCQUFNLENBQUMsV0FBVyxDQUFDLEdBQUcsRUFBRTtZQUNyQyxLQUFLLENBQUMsWUFBWSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzVCLENBQUMsQ0FBQyxDQUFDO1FBRUgsTUFBTSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDbkMsQ0FBQyxDQUFDLENBQUM7SUFFSCxJQUFJLENBQUMseUNBQXlDLEVBQUUsR0FBRyxFQUFFO1FBQ25ELE1BQU0sTUFBTSxHQUFHLHlCQUFNLENBQUMsV0FBVyxDQUFDLEdBQUcsRUFBRTtZQUNyQyxLQUFLLENBQUMsWUFBWSxDQUFDLENBQUMsZUFBZSxDQUFDLENBQUMsQ0FBQztRQUN4QyxDQUFDLENBQUMsQ0FBQztRQUNILE1BQU0sQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMscUJBQXFCO0lBQ3pELENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLDhDQUE4QyxFQUFFLEdBQUcsRUFBRTtRQUN4RCx5QkFBTSxDQUFDLFdBQVcsQ0FBQyxHQUFHLEVBQUU7WUFDdEIsS0FBSyxDQUFDLFlBQVksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBQ25FLENBQUMsQ0FBQyxDQUFDO1FBQ0gseUJBQU0sQ0FBQyxXQUFXLENBQUMsR0FBRyxFQUFFO1lBQ3RCLHVCQUF1QjtZQUN2QixLQUFLLENBQUMsWUFBWSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksRUFBRSxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDbkUsQ0FBQyxDQUFDLENBQUM7UUFFSCxNQUFNLE1BQU0sR0FBRyx5QkFBTSxDQUFDLFdBQVcsQ0FBQyxHQUFHLEVBQUU7WUFDckMsS0FBSyxDQUFDLGdCQUFnQixFQUFFLENBQUM7UUFDM0IsQ0FBQyxDQUFDLENBQUM7UUFDSCxNQUFNLGtCQUFrQixHQUFHLENBQUMsQ0FBQztRQUM3QixNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUMsVUFBVSxrQkFBa0IsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQzVGLENBQUMsQ0FBQyxDQUFDO0FBQ0wsQ0FBQyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZXdyaXRhYmxlQmxvY2sgfSBmcm9tICcuLi8uLi8uLi9saWIvYXBpL3V0aWwvZGlzcGxheSc7XG5pbXBvcnQgeyBzdGRlcnIgfSBmcm9tICcuLi9jb25zb2xlLWxpc3RlbmVyJztcblxuXG5kZXNjcmliZSgnUmV3cml0YWJsZSBCbG9jayBUZXN0cycsICgpID0+IHtcbiAgbGV0IGJsb2NrOiBSZXdyaXRhYmxlQmxvY2s7XG4gIGJlZm9yZUVhY2goKCkgPT4ge1xuICAgIGJsb2NrID0gbmV3IFJld3JpdGFibGVCbG9jayhwcm9jZXNzLnN0ZGVycik7XG4gICAgcHJvY2Vzcy5zdGRlcnIucm93cyA9IDgwO1xuICB9KTtcblxuICB0ZXN0KCdkaXNwbGF5TGluZXMgd3JpdGVzIG1heGltdW0gbGluZXMgYmFzZWQgb24gcm93cyBpZiB0aGVyZSBhcmUgbW9yZSBsaW5lcyB0aGFuIHJvd3MnLCAoKSA9PiB7XG4gICAgY29uc3QgbGluZXMgPSBBcnJheS5mcm9tKEFycmF5KDEwMCkua2V5cygpKS5tYXAobGluZSA9PiBsaW5lLnRvU3RyaW5nKCkpO1xuICAgIGNvbnN0IG91dHB1dCA9IHN0ZGVyci5pbnNwZWN0U3luYygoKSA9PiB7XG4gICAgICBibG9jay5kaXNwbGF5TGluZXMobGluZXMpO1xuICAgIH0pO1xuXG4gICAgZXhwZWN0KG91dHB1dC5sZW5ndGgpLnRvRXF1YWwoYmxvY2suaGVpZ2h0ISk7XG4gIH0pO1xuXG4gIHRlc3QoJ2Rpc3BsYXlMaW5lcyB3cml0ZXMgbWF4aW11bSBsaW5lcyBiYXNlZCBvbiBsaW5lcyBsZW5ndGggaWYgdGhlcmUgYXJlIGxlc3MgbGluZXMgdGhhbiByb3dzJywgKCkgPT4ge1xuICAgIGNvbnN0IGxpbmVzID0gQXJyYXkuZnJvbShBcnJheSg0NSkua2V5cygpKS5tYXAobGluZSA9PiBsaW5lLnRvU3RyaW5nKCkpO1xuICAgIGNvbnN0IG91dHB1dCA9IHN0ZGVyci5pbnNwZWN0U3luYygoKSA9PiB7XG4gICAgICBibG9jay5kaXNwbGF5TGluZXMobGluZXMpO1xuICAgIH0pO1xuXG4gICAgZXhwZWN0KG91dHB1dC5sZW5ndGgpLnRvRXF1YWwoNDYpO1xuICB9KTtcblxuICB0ZXN0KCdkaXNwbGF5TGluZXMgd3JpdGVzIG1heGltdW0gbGluZXMgYmFzZWQgb24gbGluZXMgbGVuZ3RoIGlmIHJvd3MgaXMgdW5kZWZpbmVkJywgKCkgPT4ge1xuICAgIGNvbnN0IGxpbmVzID0gQXJyYXkuZnJvbShBcnJheSg1KS5rZXlzKCkpLm1hcChsaW5lID0+IGxpbmUudG9TdHJpbmcoKSk7XG4gICAgcHJvY2Vzcy5zdGRlcnIucm93cyA9IHVuZGVmaW5lZDtcbiAgICBjb25zdCBvdXRwdXQgPSBzdGRlcnIuaW5zcGVjdFN5bmMoKCkgPT4ge1xuICAgICAgYmxvY2suZGlzcGxheUxpbmVzKGxpbmVzKTtcbiAgICB9KTtcblxuICAgIGV4cGVjdChvdXRwdXQubGVuZ3RoKS50b0VxdWFsKDYpO1xuICB9KTtcblxuICB0ZXN0KCdkaXNwbGF5IGFjY291bnRzIGZvciBuZXdsaW5lcyBpbiBvdXRwdXQnLCAoKSA9PiB7XG4gICAgY29uc3Qgb3V0cHV0ID0gc3RkZXJyLmluc3BlY3RTeW5jKCgpID0+IHtcbiAgICAgIGJsb2NrLmRpc3BsYXlMaW5lcyhbJ2JlZm9yZVxcbmFmdGVyJ10pO1xuICAgIH0pO1xuICAgIGV4cGVjdChvdXRwdXQubGVuZ3RoKS50b0VxdWFsKDMpOyAvLyBjdXJzb3J1cCArIDIgbGluZXNcbiAgfSk7XG5cbiAgdGVzdCgncmVtb3ZlRW1wdHlMaW5lcyBvbmx5IHJlbW92ZXMgdHJhaWxpbmcgbGluZXMnLCAoKSA9PiB7XG4gICAgc3RkZXJyLmluc3BlY3RTeW5jKCgpID0+IHtcbiAgICAgIGJsb2NrLmRpc3BsYXlMaW5lcyhBcnJheS5mcm9tKEFycmF5KDUpLmtleXMoKSkubWFwKHggPT4gYCR7eH1gKSk7XG4gICAgfSk7XG4gICAgc3RkZXJyLmluc3BlY3RTeW5jKCgpID0+IHtcbiAgICAgIC8vIExlYXZlcyAzIGVtcHR5IGxpbmVzXG4gICAgICBibG9jay5kaXNwbGF5TGluZXMoQXJyYXkuZnJvbShBcnJheSgyKS5rZXlzKCkpLm1hcCh4ID0+IGAke3h9YCkpO1xuICAgIH0pO1xuXG4gICAgY29uc3Qgb3V0cHV0ID0gc3RkZXJyLmluc3BlY3RTeW5jKCgpID0+IHtcbiAgICAgIGJsb2NrLnJlbW92ZUVtcHR5TGluZXMoKTtcbiAgICB9KTtcbiAgICBjb25zdCBleHBlY3RlZEVtcHR5TGluZXMgPSAzO1xuICAgIGV4cGVjdChKU09OLnN0cmluZ2lmeShvdXRwdXQpKS50b0VxdWFsKEpTT04uc3RyaW5naWZ5KFtgXFx1MDAxYlske2V4cGVjdGVkRW1wdHlMaW5lc31BYF0pKTtcbiAgfSk7XG59KTsiXX0=