aws-cdk 2.1001.0 → 2.1002.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 (54) hide show
  1. package/README.md +15 -0
  2. package/THIRD_PARTY_LICENSES +41 -41
  3. package/build-info.json +2 -2
  4. package/lib/api/bootstrap/bootstrap-template.yaml +3 -1
  5. package/lib/api/deployments/deploy-stack.d.ts +0 -20
  6. package/lib/api/deployments/deploy-stack.js +25 -20
  7. package/lib/api/deployments/deployments.d.ts +0 -27
  8. package/lib/api/deployments/deployments.js +13 -13
  9. package/lib/api/resource-import/importer.d.ts +0 -8
  10. package/lib/api/resource-import/importer.js +1 -1
  11. package/lib/api/resource-import/migrator.js +1 -2
  12. package/lib/api/stack-events/stack-activity-monitor.d.ts +87 -165
  13. package/lib/api/stack-events/stack-activity-monitor.js +61 -445
  14. package/lib/api/stack-events/stack-event-poller.d.ts +6 -0
  15. package/lib/api/stack-events/stack-event-poller.js +1 -1
  16. package/lib/api/stack-events/stack-progress-monitor.d.ts +61 -0
  17. package/lib/api/stack-events/stack-progress-monitor.js +94 -0
  18. package/lib/api/work-graph/work-graph-builder.js +4 -4
  19. package/lib/cli/activity-printer/base.d.ts +51 -0
  20. package/lib/cli/activity-printer/base.js +115 -0
  21. package/lib/cli/activity-printer/current.d.ts +25 -0
  22. package/lib/cli/activity-printer/current.js +122 -0
  23. package/lib/cli/activity-printer/history.d.ts +31 -0
  24. package/lib/cli/activity-printer/history.js +109 -0
  25. package/lib/cli/activity-printer/index.d.ts +3 -0
  26. package/lib/cli/activity-printer/index.js +20 -0
  27. package/lib/cli/cdk-toolkit.d.ts +1 -1
  28. package/lib/cli/cdk-toolkit.js +10 -9
  29. package/lib/cli/cli-config.js +5 -4
  30. package/lib/cli/cli.js +3 -1
  31. package/lib/cli/convert-to-user-input.js +18 -16
  32. package/lib/cli/parse-command-line-arguments.js +7 -1
  33. package/lib/cli/user-input.d.ts +8 -0
  34. package/lib/cli/user-input.js +1 -1
  35. package/lib/commands/deploy.d.ts +13 -0
  36. package/lib/commands/deploy.js +18 -0
  37. package/lib/context-providers/cc-api-provider.js +2 -2
  38. package/lib/index.js +18101 -16933
  39. package/lib/init.d.ts +5 -1
  40. package/lib/init.js +11 -8
  41. package/lib/legacy-exports-source.d.ts +1 -1
  42. package/lib/legacy-exports-source.js +2 -2
  43. package/lib/notices.js +2 -2
  44. package/lib/toolkit/cli-io-host.d.ts +28 -0
  45. package/lib/toolkit/cli-io-host.js +74 -2
  46. package/lib/toolkit/error.d.ts +1 -44
  47. package/lib/toolkit/error.js +16 -76
  48. package/lib/util/cloudformation.d.ts +12 -0
  49. package/lib/util/cloudformation.js +27 -1
  50. package/lib/util/string-manipulation.d.ts +5 -1
  51. package/lib/util/string-manipulation.js +11 -5
  52. package/package.json +25 -24
  53. /package/lib/{api/stack-events → cli/activity-printer}/display.d.ts +0 -0
  54. /package/lib/{api/stack-events → cli/activity-printer}/display.js +0 -0
@@ -1,108 +1,83 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.CurrentActivityPrinter = exports.HistoryActivityPrinter = exports.StackActivityMonitor = exports.StackActivityProgress = void 0;
3
+ exports.StackActivityMonitor = void 0;
4
4
  const util = require("util");
5
5
  const cloud_assembly_schema_1 = require("@aws-cdk/cloud-assembly-schema");
6
- const chalk = require("chalk");
6
+ const uuid = require("uuid");
7
7
  const stack_event_poller_1 = require("./stack-event-poller");
8
- const logging_1 = require("../../logging");
9
- const cli_io_host_1 = require("../../toolkit/cli-io-host");
10
- const display_1 = require("./display");
11
- /**
12
- * Supported display modes for stack deployment activity
13
- */
14
- var StackActivityProgress;
15
- (function (StackActivityProgress) {
16
- /**
17
- * Displays a progress bar with only the events for the resource currently being deployed
18
- */
19
- StackActivityProgress["BAR"] = "bar";
20
- /**
21
- * Displays complete history with all CloudFormation stack events
22
- */
23
- StackActivityProgress["EVENTS"] = "events";
24
- })(StackActivityProgress || (exports.StackActivityProgress = StackActivityProgress = {}));
8
+ const messages_1 = require("../../cli/messages");
9
+ const util_1 = require("../../util");
10
+ const stack_progress_monitor_1 = require("./stack-progress-monitor");
25
11
  class StackActivityMonitor {
26
- /**
27
- * Create a Stack Activity Monitor using a default printer, based on context clues
28
- */
29
- static withDefaultPrinter(cfn, stackName, stackArtifact, options = {}) {
30
- var _a, _b;
31
- const stream = options.ci ? process.stdout : process.stderr;
32
- const props = {
33
- resourceTypeColumnWidth: calcMaxResourceTypeLength(stackArtifact.template),
34
- resourcesTotal: options.resourcesTotal,
35
- stream,
36
- };
37
- const isWindows = process.platform === 'win32';
38
- const verbose = (_a = options.logLevel) !== null && _a !== void 0 ? _a : 'info';
39
- // On some CI systems (such as CircleCI) output still reports as a TTY so we also
40
- // need an individual check for whether we're running on CI.
41
- // see: https://discuss.circleci.com/t/circleci-terminal-is-a-tty-but-term-is-not-set/9965
42
- const fancyOutputAvailable = !isWindows && stream.isTTY && !options.ci;
43
- const progress = (_b = options.progress) !== null && _b !== void 0 ? _b : StackActivityProgress.BAR;
44
- const printer = fancyOutputAvailable && !verbose && progress === StackActivityProgress.BAR
45
- ? new CurrentActivityPrinter(props)
46
- : new HistoryActivityPrinter(props);
47
- return new StackActivityMonitor(cfn, stackName, printer, stackArtifact, options.changeSetCreationTime);
48
- }
49
- constructor(cfn, stackName, printer, stack, changeSetCreationTime) {
12
+ constructor({ cfn, ioHost, action, stack, stackName, resourcesTotal, changeSetCreationTime, pollingInterval = 2000, }) {
50
13
  var _a;
51
- this.stackName = stackName;
52
- this.printer = printer;
53
- this.stack = stack;
54
14
  this.errors = [];
55
- this.active = false;
15
+ this.ioHost = ioHost;
16
+ this.action = action;
17
+ this.stack = stack;
18
+ this.stackName = stackName;
19
+ this.progressMonitor = new stack_progress_monitor_1.StackProgressMonitor(resourcesTotal);
20
+ this.pollingInterval = pollingInterval;
56
21
  this.poller = new stack_event_poller_1.StackEventPoller(cfn, {
57
22
  stackName,
58
23
  startTime: (_a = changeSetCreationTime === null || changeSetCreationTime === void 0 ? void 0 : changeSetCreationTime.getTime()) !== null && _a !== void 0 ? _a : Date.now(),
59
24
  });
60
25
  }
61
- start() {
62
- this.active = true;
63
- this.printer.start();
26
+ async start() {
27
+ this.monitorId = uuid.v4();
28
+ await this.ioHost.notify((0, messages_1.debug)(this.action, `Deploying ${this.stackName}`, 'CDK_TOOLKIT_I5501', {
29
+ deployment: this.monitorId,
30
+ stack: this.stack,
31
+ stackName: this.stackName,
32
+ resourcesTotal: this.progressMonitor.total,
33
+ }));
64
34
  this.scheduleNextTick();
65
35
  return this;
66
36
  }
67
37
  async stop() {
68
- this.active = false;
38
+ const oldMonitorId = this.monitorId;
39
+ this.monitorId = undefined;
69
40
  if (this.tickTimer) {
70
41
  clearTimeout(this.tickTimer);
71
42
  }
72
43
  // Do a final poll for all events. This is to handle the situation where DescribeStackStatus
73
44
  // already returned an error, but the monitor hasn't seen all the events yet and we'd end
74
45
  // up not printing the failure reason to users.
75
- await this.finalPollToEnd();
76
- this.printer.stop();
46
+ await this.finalPollToEnd(oldMonitorId);
47
+ await this.ioHost.notify((0, messages_1.debug)(this.action, `Completed ${this.stackName}`, 'CDK_TOOLKIT_I5503', {
48
+ deployment: oldMonitorId,
49
+ stack: this.stack,
50
+ stackName: this.stackName,
51
+ resourcesTotal: this.progressMonitor.total,
52
+ }));
77
53
  }
78
54
  scheduleNextTick() {
79
- if (!this.active) {
55
+ if (!this.monitorId) {
80
56
  return;
81
57
  }
82
- this.tickTimer = setTimeout(() => void this.tick(), this.printer.updateSleep);
58
+ this.tickTimer = setTimeout(() => void this.tick(), this.pollingInterval);
83
59
  }
84
60
  async tick() {
85
- if (!this.active) {
61
+ if (!this.monitorId) {
86
62
  return;
87
63
  }
88
64
  try {
89
- this.readPromise = this.readNewEvents();
65
+ this.readPromise = this.readNewEvents(this.monitorId);
90
66
  await this.readPromise;
91
67
  this.readPromise = undefined;
92
68
  // We might have been stop()ped while the network call was in progress.
93
- if (!this.active) {
69
+ if (!this.monitorId) {
94
70
  return;
95
71
  }
96
- this.printer.print();
97
72
  }
98
73
  catch (e) {
99
- (0, logging_1.error)('Error occurred while monitoring stack: %s', e);
74
+ await this.ioHost.notify((0, messages_1.error)(this.action, util.format('Error occurred while monitoring stack: %s', e), 'CDK_TOOLKIT_E5500', { error: e }));
100
75
  }
101
76
  this.scheduleNextTick();
102
77
  }
103
78
  findMetadataFor(logicalId) {
104
- var _a, _b;
105
- const metadata = (_b = (_a = this.stack) === null || _a === void 0 ? void 0 : _a.manifest) === null || _b === void 0 ? void 0 : _b.metadata;
79
+ var _a;
80
+ const metadata = (_a = this.stack.manifest) === null || _a === void 0 ? void 0 : _a.metadata;
106
81
  if (!logicalId || !metadata) {
107
82
  return undefined;
108
83
  }
@@ -126,15 +101,18 @@ class StackActivityMonitor {
126
101
  * see a next page and the last event in the page is new to us (and within the time window).
127
102
  * haven't seen the final event
128
103
  */
129
- async readNewEvents() {
104
+ async readNewEvents(monitorId) {
130
105
  const pollEvents = await this.poller.poll();
131
- const activities = pollEvents.map((event) => ({
132
- ...event,
133
- metadata: this.findMetadataFor(event.event.LogicalResourceId),
134
- }));
135
- for (const activity of activities) {
106
+ for (const resourceEvent of pollEvents) {
107
+ this.progressMonitor.process(resourceEvent.event);
108
+ const activity = {
109
+ deployment: monitorId,
110
+ event: resourceEvent.event,
111
+ metadata: this.findMetadataFor(resourceEvent.event.LogicalResourceId),
112
+ progress: this.progressMonitor.progress,
113
+ };
136
114
  this.checkForErrors(activity);
137
- this.printer.addActivity(activity);
115
+ await this.ioHost.notify((0, messages_1.info)(this.action, this.formatActivity(activity, true), 'CDK_TOOLKIT_I5502', activity));
138
116
  }
139
117
  }
140
118
  /**
@@ -143,7 +121,7 @@ class StackActivityMonitor {
143
121
  * Finish any poll currently in progress, then do a final one until we've
144
122
  * reached the last page.
145
123
  */
146
- async finalPollToEnd() {
124
+ async finalPollToEnd(monitorId) {
147
125
  // If we were doing a poll, finish that first. It was started before
148
126
  // the moment we were sure we weren't going to get any new events anymore
149
127
  // so we need to do a new one anyway. Need to wait for this one though
@@ -151,11 +129,21 @@ class StackActivityMonitor {
151
129
  if (this.readPromise) {
152
130
  await this.readPromise;
153
131
  }
154
- await this.readNewEvents();
132
+ await this.readNewEvents(monitorId);
133
+ }
134
+ /**
135
+ * Formats a stack activity into a basic string
136
+ */
137
+ formatActivity(activity, progress) {
138
+ const event = activity.event;
139
+ const metadata = activity.metadata;
140
+ const resourceName = metadata ? metadata.constructPath : event.LogicalResourceId || '';
141
+ const logicalId = resourceName !== event.LogicalResourceId ? `(${event.LogicalResourceId}) ` : '';
142
+ return util.format('%s | %s%s | %s | %s | %s %s%s%s', event.StackName, progress !== false ? `${activity.progress.formatted} | ` : '', new Date(event.Timestamp).toLocaleTimeString(), event.ResourceStatus || '', event.ResourceType, resourceName, logicalId, event.ResourceStatusReason ? event.ResourceStatusReason : '', (metadata === null || metadata === void 0 ? void 0 : metadata.entry.trace) ? `\n\t${metadata.entry.trace.join('\n\t\\_ ')}` : '');
155
143
  }
156
144
  checkForErrors(activity) {
157
145
  var _a, _b, _c;
158
- if (hasErrorMessage((_a = activity.event.ResourceStatus) !== null && _a !== void 0 ? _a : '')) {
146
+ if ((0, util_1.stackEventHasErrorMessage)((_a = activity.event.ResourceStatus) !== null && _a !== void 0 ? _a : '')) {
159
147
  const isCancelled = ((_b = activity.event.ResourceStatusReason) !== null && _b !== void 0 ? _b : '').indexOf('cancelled') > -1;
160
148
  // Cancelled is not an interesting failure reason, nor is the stack message (stack
161
149
  // message will just say something like "stack failed to update")
@@ -175,376 +163,4 @@ class StackActivityMonitor {
175
163
  }
176
164
  }
177
165
  exports.StackActivityMonitor = StackActivityMonitor;
178
- function padRight(n, x) {
179
- return x + ' '.repeat(Math.max(0, n - x.length));
180
- }
181
- /**
182
- * Infamous padLeft()
183
- */
184
- function padLeft(n, x) {
185
- return ' '.repeat(Math.max(0, n - x.length)) + x;
186
- }
187
- function calcMaxResourceTypeLength(template) {
188
- const resources = (template && template.Resources) || {};
189
- let maxWidth = 0;
190
- for (const id of Object.keys(resources)) {
191
- const type = resources[id].Type || '';
192
- if (type.length > maxWidth) {
193
- maxWidth = type.length;
194
- }
195
- }
196
- return maxWidth;
197
- }
198
- class ActivityPrinterBase {
199
- constructor(props) {
200
- this.props = props;
201
- /**
202
- * Fetch new activity every 5 seconds
203
- */
204
- this.updateSleep = 5000;
205
- /**
206
- * A list of resource IDs which are currently being processed
207
- */
208
- this.resourcesInProgress = {};
209
- /**
210
- * Previous completion state observed by logical ID
211
- *
212
- * We use this to detect that if we see a DELETE_COMPLETE after a
213
- * CREATE_COMPLETE, it's actually a rollback and we should DECREASE
214
- * resourcesDone instead of increase it
215
- */
216
- this.resourcesPrevCompleteState = {};
217
- /**
218
- * Count of resources that have reported a _COMPLETE status
219
- */
220
- this.resourcesDone = 0;
221
- /**
222
- * How many digits we need to represent the total count (for lining up the status reporting)
223
- */
224
- this.resourceDigits = 0;
225
- this.rollingBack = false;
226
- this.failures = new Array();
227
- this.hookFailureMap = new Map();
228
- // +1 because the stack also emits a "COMPLETE" event at the end, and that wasn't
229
- // counted yet. This makes it line up with the amount of events we expect.
230
- this.resourcesTotal = props.resourcesTotal ? props.resourcesTotal + 1 : undefined;
231
- // How many digits does this number take to represent?
232
- this.resourceDigits = this.resourcesTotal ? Math.ceil(Math.log10(this.resourcesTotal)) : 0;
233
- }
234
- failureReason(activity) {
235
- var _a, _b;
236
- const resourceStatusReason = (_a = activity.event.ResourceStatusReason) !== null && _a !== void 0 ? _a : '';
237
- const logicalResourceId = (_b = activity.event.LogicalResourceId) !== null && _b !== void 0 ? _b : '';
238
- const hookFailureReasonMap = this.hookFailureMap.get(logicalResourceId);
239
- if (hookFailureReasonMap !== undefined) {
240
- for (const hookType of hookFailureReasonMap.keys()) {
241
- if (resourceStatusReason.includes(hookType)) {
242
- return resourceStatusReason + ' : ' + hookFailureReasonMap.get(hookType);
243
- }
244
- }
245
- }
246
- return resourceStatusReason;
247
- }
248
- addActivity(activity) {
249
- var _a, _b, _c, _d, _e;
250
- const status = activity.event.ResourceStatus;
251
- const hookStatus = activity.event.HookStatus;
252
- const hookType = activity.event.HookType;
253
- if (!status || !activity.event.LogicalResourceId) {
254
- return;
255
- }
256
- if (status === 'ROLLBACK_IN_PROGRESS' || status === 'UPDATE_ROLLBACK_IN_PROGRESS') {
257
- // Only triggered on the stack once we've started doing a rollback
258
- this.rollingBack = true;
259
- }
260
- if (status.endsWith('_IN_PROGRESS')) {
261
- this.resourcesInProgress[activity.event.LogicalResourceId] = activity;
262
- }
263
- if (hasErrorMessage(status)) {
264
- const isCancelled = ((_a = activity.event.ResourceStatusReason) !== null && _a !== void 0 ? _a : '').indexOf('cancelled') > -1;
265
- // Cancelled is not an interesting failure reason
266
- if (!isCancelled) {
267
- this.failures.push(activity);
268
- }
269
- }
270
- if (status.endsWith('_COMPLETE') || status.endsWith('_FAILED')) {
271
- delete this.resourcesInProgress[activity.event.LogicalResourceId];
272
- }
273
- if (status.endsWith('_COMPLETE_CLEANUP_IN_PROGRESS')) {
274
- this.resourcesDone++;
275
- }
276
- if (status.endsWith('_COMPLETE')) {
277
- const prevState = this.resourcesPrevCompleteState[activity.event.LogicalResourceId];
278
- if (!prevState) {
279
- this.resourcesDone++;
280
- }
281
- else {
282
- // If we completed this before and we're completing it AGAIN, means we're rolling back.
283
- // Protect against silly underflow.
284
- this.resourcesDone--;
285
- if (this.resourcesDone < 0) {
286
- this.resourcesDone = 0;
287
- }
288
- }
289
- this.resourcesPrevCompleteState[activity.event.LogicalResourceId] = status;
290
- }
291
- if (hookStatus !== undefined &&
292
- hookStatus.endsWith('_COMPLETE_FAILED') &&
293
- activity.event.LogicalResourceId !== undefined &&
294
- hookType !== undefined) {
295
- if (this.hookFailureMap.has(activity.event.LogicalResourceId)) {
296
- (_b = this.hookFailureMap.get(activity.event.LogicalResourceId)) === null || _b === void 0 ? void 0 : _b.set(hookType, (_c = activity.event.HookStatusReason) !== null && _c !== void 0 ? _c : '');
297
- }
298
- else {
299
- this.hookFailureMap.set(activity.event.LogicalResourceId, new Map());
300
- (_d = this.hookFailureMap.get(activity.event.LogicalResourceId)) === null || _d === void 0 ? void 0 : _d.set(hookType, (_e = activity.event.HookStatusReason) !== null && _e !== void 0 ? _e : '');
301
- }
302
- }
303
- }
304
- start() {
305
- // Empty on purpose
306
- }
307
- stop() {
308
- // Empty on purpose
309
- }
310
- }
311
- /**
312
- * Activity Printer which shows a full log of all CloudFormation events
313
- *
314
- * When there hasn't been activity for a while, it will print the resources
315
- * that are currently in progress, to show what's holding up the deployment.
316
- */
317
- class HistoryActivityPrinter extends ActivityPrinterBase {
318
- constructor(props) {
319
- super(props);
320
- /**
321
- * Last time we printed something to the console.
322
- *
323
- * Used to measure timeout for progress reporting.
324
- */
325
- this.lastPrintTime = Date.now();
326
- /**
327
- * Number of ms of change absence before we tell the user about the resources that are currently in progress.
328
- */
329
- this.inProgressDelay = 30000;
330
- this.printable = new Array();
331
- }
332
- addActivity(activity) {
333
- super.addActivity(activity);
334
- this.printable.push(activity);
335
- this.print();
336
- }
337
- print() {
338
- for (const activity of this.printable) {
339
- this.printOne(activity);
340
- }
341
- this.printable.splice(0, this.printable.length);
342
- this.printInProgress();
343
- }
344
- stop() {
345
- // Print failures at the end
346
- if (this.failures.length > 0) {
347
- (0, logging_1.info)('\nFailed resources:');
348
- for (const failure of this.failures) {
349
- // Root stack failures are not interesting
350
- if (failure.isStackEvent) {
351
- continue;
352
- }
353
- this.printOne(failure, false);
354
- }
355
- }
356
- }
357
- printOne(activity, progress) {
358
- const event = activity.event;
359
- const color = colorFromStatusResult(event.ResourceStatus);
360
- let reasonColor = chalk.cyan;
361
- let stackTrace = '';
362
- const metadata = activity.metadata;
363
- if (event.ResourceStatus && event.ResourceStatus.indexOf('FAILED') !== -1) {
364
- if (progress == undefined || progress) {
365
- event.ResourceStatusReason = event.ResourceStatusReason ? this.failureReason(activity) : '';
366
- }
367
- if (metadata) {
368
- stackTrace = metadata.entry.trace ? `\n\t${metadata.entry.trace.join('\n\t\\_ ')}` : '';
369
- }
370
- reasonColor = chalk.red;
371
- }
372
- const resourceName = metadata ? metadata.constructPath : event.LogicalResourceId || '';
373
- const logicalId = resourceName !== event.LogicalResourceId ? `(${event.LogicalResourceId}) ` : '';
374
- (0, logging_1.info)(util.format('%s | %s%s | %s | %s | %s %s%s%s', event.StackName, progress !== false ? `${this.progress()} | ` : '', new Date(event.Timestamp).toLocaleTimeString(), color(padRight(STATUS_WIDTH, (event.ResourceStatus || '').slice(0, STATUS_WIDTH))), // pad left and trim
375
- padRight(this.props.resourceTypeColumnWidth, event.ResourceType || ''), color(chalk.bold(resourceName)), logicalId, reasonColor(chalk.bold(event.ResourceStatusReason ? event.ResourceStatusReason : '')), reasonColor(stackTrace)));
376
- this.lastPrintTime = Date.now();
377
- }
378
- /**
379
- * Report the current progress as a [34/42] string, or just [34] if the total is unknown
380
- */
381
- progress() {
382
- if (this.resourcesTotal == null) {
383
- // Don't have total, show simple count and hope the human knows
384
- return padLeft(3, util.format('%s', this.resourcesDone)); // max 500 resources
385
- }
386
- return util.format('%s/%s', padLeft(this.resourceDigits, this.resourcesDone.toString()), padLeft(this.resourceDigits, this.resourcesTotal != null ? this.resourcesTotal.toString() : '?'));
387
- }
388
- /**
389
- * If some resources are taking a while to create, notify the user about what's currently in progress
390
- */
391
- printInProgress() {
392
- if (Date.now() < this.lastPrintTime + this.inProgressDelay) {
393
- return;
394
- }
395
- if (Object.keys(this.resourcesInProgress).length > 0) {
396
- (0, logging_1.info)(util.format('%s Currently in progress: %s', this.progress(), chalk.bold(Object.keys(this.resourcesInProgress).join(', '))));
397
- }
398
- // We cheat a bit here. To prevent printInProgress() from repeatedly triggering,
399
- // we set the timestamp into the future. It will be reset whenever a regular print
400
- // occurs, after which we can be triggered again.
401
- this.lastPrintTime = +Infinity;
402
- }
403
- }
404
- exports.HistoryActivityPrinter = HistoryActivityPrinter;
405
- /**
406
- * Activity Printer which shows the resources currently being updated
407
- *
408
- * It will continuously reupdate the terminal and show only the resources
409
- * that are currently being updated, in addition to a progress bar which
410
- * shows how far along the deployment is.
411
- *
412
- * Resources that have failed will always be shown, and will be recapitulated
413
- * along with their stack trace when the monitoring ends.
414
- *
415
- * Resources that failed deployment because they have been cancelled are
416
- * not included.
417
- */
418
- class CurrentActivityPrinter extends ActivityPrinterBase {
419
- constructor(props) {
420
- super(props);
421
- /**
422
- * This looks very disorienting sleeping for 5 seconds. Update quicker.
423
- */
424
- this.updateSleep = 2000;
425
- this.oldLogThreshold = cli_io_host_1.CliIoHost.instance().logLevel;
426
- this.stream = props.stream;
427
- this.block = new display_1.RewritableBlock(this.stream);
428
- }
429
- print() {
430
- var _a;
431
- const lines = [];
432
- // Add a progress bar at the top
433
- const progressWidth = Math.max(Math.min(((_a = this.block.width) !== null && _a !== void 0 ? _a : 80) - PROGRESSBAR_EXTRA_SPACE - 1, MAX_PROGRESSBAR_WIDTH), MIN_PROGRESSBAR_WIDTH);
434
- const prog = this.progressBar(progressWidth);
435
- if (prog) {
436
- lines.push(' ' + prog, '');
437
- }
438
- // Normally we'd only print "resources in progress", but it's also useful
439
- // to keep an eye on the failures and know about the specific errors asquickly
440
- // as possible (while the stack is still rolling back), so add those in.
441
- const toPrint = [...this.failures, ...Object.values(this.resourcesInProgress)];
442
- toPrint.sort((a, b) => a.event.Timestamp.getTime() - b.event.Timestamp.getTime());
443
- lines.push(...toPrint.map((res) => {
444
- var _a, _b, _c;
445
- const color = colorFromStatusActivity(res.event.ResourceStatus);
446
- const resourceName = (_c = (_b = (_a = res.metadata) === null || _a === void 0 ? void 0 : _a.constructPath) !== null && _b !== void 0 ? _b : res.event.LogicalResourceId) !== null && _c !== void 0 ? _c : '';
447
- return util.format('%s | %s | %s | %s%s', padLeft(TIMESTAMP_WIDTH, new Date(res.event.Timestamp).toLocaleTimeString()), color(padRight(STATUS_WIDTH, (res.event.ResourceStatus || '').slice(0, STATUS_WIDTH))), padRight(this.props.resourceTypeColumnWidth, res.event.ResourceType || ''), color(chalk.bold(shorten(40, resourceName))), this.failureReasonOnNextLine(res));
448
- }));
449
- this.block.displayLines(lines);
450
- }
451
- start() {
452
- // Need to prevent the waiter from printing 'stack not stable' every 5 seconds, it messes
453
- // with the output calculations.
454
- this.oldLogThreshold = cli_io_host_1.CliIoHost.instance().logLevel;
455
- cli_io_host_1.CliIoHost.instance().logLevel = 'info';
456
- }
457
- stop() {
458
- var _a, _b, _c;
459
- cli_io_host_1.CliIoHost.instance().logLevel = this.oldLogThreshold;
460
- // Print failures at the end
461
- const lines = new Array();
462
- for (const failure of this.failures) {
463
- // Root stack failures are not interesting
464
- if (failure.isStackEvent) {
465
- continue;
466
- }
467
- lines.push(util.format(chalk.red('%s | %s | %s | %s%s') + '\n', padLeft(TIMESTAMP_WIDTH, new Date(failure.event.Timestamp).toLocaleTimeString()), padRight(STATUS_WIDTH, (failure.event.ResourceStatus || '').slice(0, STATUS_WIDTH)), padRight(this.props.resourceTypeColumnWidth, failure.event.ResourceType || ''), shorten(40, (_a = failure.event.LogicalResourceId) !== null && _a !== void 0 ? _a : ''), this.failureReasonOnNextLine(failure)));
468
- const trace = (_c = (_b = failure.metadata) === null || _b === void 0 ? void 0 : _b.entry) === null || _c === void 0 ? void 0 : _c.trace;
469
- if (trace) {
470
- lines.push(chalk.red(`\t${trace.join('\n\t\\_ ')}\n`));
471
- }
472
- }
473
- // Display in the same block space, otherwise we're going to have silly empty lines.
474
- this.block.displayLines(lines);
475
- this.block.removeEmptyLines();
476
- }
477
- progressBar(width) {
478
- if (!this.resourcesTotal) {
479
- return '';
480
- }
481
- const fraction = Math.min(this.resourcesDone / this.resourcesTotal, 1);
482
- const innerWidth = Math.max(1, width - 2);
483
- const chars = innerWidth * fraction;
484
- const remainder = chars - Math.floor(chars);
485
- const fullChars = FULL_BLOCK.repeat(Math.floor(chars));
486
- const partialChar = PARTIAL_BLOCK[Math.floor(remainder * PARTIAL_BLOCK.length)];
487
- const filler = '·'.repeat(innerWidth - Math.floor(chars) - (partialChar ? 1 : 0));
488
- const color = this.rollingBack ? chalk.yellow : chalk.green;
489
- return '[' + color(fullChars + partialChar) + filler + `] (${this.resourcesDone}/${this.resourcesTotal})`;
490
- }
491
- failureReasonOnNextLine(activity) {
492
- var _a, _b;
493
- return hasErrorMessage((_a = activity.event.ResourceStatus) !== null && _a !== void 0 ? _a : '')
494
- ? `\n${' '.repeat(TIMESTAMP_WIDTH + STATUS_WIDTH + 6)}${chalk.red((_b = this.failureReason(activity)) !== null && _b !== void 0 ? _b : '')}`
495
- : '';
496
- }
497
- }
498
- exports.CurrentActivityPrinter = CurrentActivityPrinter;
499
- const FULL_BLOCK = '█';
500
- const PARTIAL_BLOCK = ['', '▏', '▎', '▍', '▌', '▋', '▊', '▉'];
501
- const MAX_PROGRESSBAR_WIDTH = 60;
502
- const MIN_PROGRESSBAR_WIDTH = 10;
503
- const PROGRESSBAR_EXTRA_SPACE = 2 /* leading spaces */ + 2 /* brackets */ + 4 /* progress number decoration */ + 6; /* 2 progress numbers up to 999 */
504
- function hasErrorMessage(status) {
505
- return status.endsWith('_FAILED') || status === 'ROLLBACK_IN_PROGRESS' || status === 'UPDATE_ROLLBACK_IN_PROGRESS';
506
- }
507
- function colorFromStatusResult(status) {
508
- if (!status) {
509
- return chalk.reset;
510
- }
511
- if (status.indexOf('FAILED') !== -1) {
512
- return chalk.red;
513
- }
514
- if (status.indexOf('ROLLBACK') !== -1) {
515
- return chalk.yellow;
516
- }
517
- if (status.indexOf('COMPLETE') !== -1) {
518
- return chalk.green;
519
- }
520
- return chalk.reset;
521
- }
522
- function colorFromStatusActivity(status) {
523
- if (!status) {
524
- return chalk.reset;
525
- }
526
- if (status.endsWith('_FAILED')) {
527
- return chalk.red;
528
- }
529
- if (status.startsWith('CREATE_') || status.startsWith('UPDATE_') || status.startsWith('IMPORT_')) {
530
- return chalk.green;
531
- }
532
- // For stacks, it may also be 'UPDDATE_ROLLBACK_IN_PROGRESS'
533
- if (status.indexOf('ROLLBACK_') !== -1) {
534
- return chalk.yellow;
535
- }
536
- if (status.startsWith('DELETE_')) {
537
- return chalk.yellow;
538
- }
539
- return chalk.reset;
540
- }
541
- function shorten(maxWidth, p) {
542
- if (p.length <= maxWidth) {
543
- return p;
544
- }
545
- const half = Math.floor((maxWidth - 3) / 2);
546
- return p.slice(0, half) + '...' + p.slice(-half);
547
- }
548
- const TIMESTAMP_WIDTH = 12;
549
- const STATUS_WIDTH = 20;
550
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-activity-monitor.js","sourceRoot":"","sources":["stack-activity-monitor.ts"],"names":[],"mappings":"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","sourcesContent":["import * as util from 'util';\nimport { ArtifactMetadataEntryType, type MetadataEntry } from '@aws-cdk/cloud-assembly-schema';\nimport type { CloudFormationStackArtifact } from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport { ResourceEvent, StackEventPoller } from './stack-event-poller';\nimport { error, info } from '../../logging';\nimport { CliIoHost, IoMessageLevel } from '../../toolkit/cli-io-host';\nimport type { ICloudFormationClient } from '../aws-auth';\nimport { RewritableBlock } from './display';\n\nexport interface StackActivity extends ResourceEvent {\n  readonly metadata?: ResourceMetadata;\n}\n\nexport interface ResourceMetadata {\n  entry: MetadataEntry;\n  constructPath: string;\n}\n\n/**\n * Supported display modes for stack deployment activity\n */\nexport enum StackActivityProgress {\n  /**\n   * Displays a progress bar with only the events for the resource currently being deployed\n   */\n  BAR = 'bar',\n\n  /**\n   * Displays complete history with all CloudFormation stack events\n   */\n  EVENTS = 'events',\n}\n\nexport interface WithDefaultPrinterProps {\n  /**\n   * Total number of resources to update\n   *\n   * Used to calculate a progress bar.\n   *\n   * @default - No progress reporting.\n   */\n  readonly resourcesTotal?: number;\n\n  /**\n   * The log level that was requested in the CLI\n   *\n   * If verbose or trace is requested, we'll always use the full history printer.\n   *\n   * @default - Use value from logging.logLevel\n   */\n  readonly logLevel?: IoMessageLevel;\n\n  /**\n   * Whether to display all stack events or to display only the events for the\n   * resource currently being deployed\n   *\n   * If not set, the stack history with all stack events will be displayed\n   *\n   * @default false\n   */\n  progress?: StackActivityProgress;\n\n  /**\n   * Whether we are on a CI system\n   *\n   * If so, disable the \"optimized\" stack monitor.\n   *\n   * @default false\n   */\n  readonly ci?: boolean;\n\n  /**\n   * Creation time of the change set\n   *\n   * This will be used to filter events, only showing those from after the change\n   * set creation time.\n   *\n   * It is recommended to use this, otherwise the filtering will be subject\n   * to clock drift between local and cloud machines.\n   *\n   * @default - local machine's current time\n   */\n  readonly changeSetCreationTime?: Date;\n}\n\nexport class StackActivityMonitor {\n  /**\n   * Create a Stack Activity Monitor using a default printer, based on context clues\n   */\n  public static withDefaultPrinter(\n    cfn: ICloudFormationClient,\n    stackName: string,\n    stackArtifact: CloudFormationStackArtifact,\n    options: WithDefaultPrinterProps = {},\n  ) {\n    const stream = options.ci ? process.stdout : process.stderr;\n\n    const props: PrinterProps = {\n      resourceTypeColumnWidth: calcMaxResourceTypeLength(stackArtifact.template),\n      resourcesTotal: options.resourcesTotal,\n      stream,\n    };\n\n    const isWindows = process.platform === 'win32';\n    const verbose = options.logLevel ?? 'info';\n    // On some CI systems (such as CircleCI) output still reports as a TTY so we also\n    // need an individual check for whether we're running on CI.\n    // see: https://discuss.circleci.com/t/circleci-terminal-is-a-tty-but-term-is-not-set/9965\n    const fancyOutputAvailable = !isWindows && stream.isTTY && !options.ci;\n    const progress = options.progress ?? StackActivityProgress.BAR;\n\n    const printer =\n      fancyOutputAvailable && !verbose && progress === StackActivityProgress.BAR\n        ? new CurrentActivityPrinter(props)\n        : new HistoryActivityPrinter(props);\n\n    return new StackActivityMonitor(cfn, stackName, printer, stackArtifact, options.changeSetCreationTime);\n  }\n\n  /**\n   * The poller used to read stack events\n   */\n  public readonly poller: StackEventPoller;\n\n  public readonly errors: string[] = [];\n\n  private active = false;\n\n  /**\n   * Current tick timer\n   */\n  private tickTimer?: ReturnType<typeof setTimeout>;\n\n  /**\n   * Set to the activity of reading the current events\n   */\n  private readPromise?: Promise<any>;\n\n  constructor(\n    cfn: ICloudFormationClient,\n    private readonly stackName: string,\n    private readonly printer: IActivityPrinter,\n    private readonly stack?: CloudFormationStackArtifact,\n    changeSetCreationTime?: Date,\n  ) {\n    this.poller = new StackEventPoller(cfn, {\n      stackName,\n      startTime: changeSetCreationTime?.getTime() ?? Date.now(),\n    });\n  }\n\n  public start() {\n    this.active = true;\n    this.printer.start();\n    this.scheduleNextTick();\n    return this;\n  }\n\n  public async stop() {\n    this.active = false;\n    if (this.tickTimer) {\n      clearTimeout(this.tickTimer);\n    }\n\n    // Do a final poll for all events. This is to handle the situation where DescribeStackStatus\n    // already returned an error, but the monitor hasn't seen all the events yet and we'd end\n    // up not printing the failure reason to users.\n    await this.finalPollToEnd();\n\n    this.printer.stop();\n  }\n\n  private scheduleNextTick() {\n    if (!this.active) {\n      return;\n    }\n\n    this.tickTimer = setTimeout(() => void this.tick(), this.printer.updateSleep);\n  }\n\n  private async tick() {\n    if (!this.active) {\n      return;\n    }\n\n    try {\n      this.readPromise = this.readNewEvents();\n      await this.readPromise;\n      this.readPromise = undefined;\n\n      // We might have been stop()ped while the network call was in progress.\n      if (!this.active) {\n        return;\n      }\n\n      this.printer.print();\n    } catch (e) {\n      error('Error occurred while monitoring stack: %s', e);\n    }\n    this.scheduleNextTick();\n  }\n\n  private findMetadataFor(logicalId: string | undefined): ResourceMetadata | undefined {\n    const metadata = this.stack?.manifest?.metadata;\n    if (!logicalId || !metadata) {\n      return undefined;\n    }\n    for (const path of Object.keys(metadata)) {\n      const entry = metadata[path]\n        .filter((e) => e.type === ArtifactMetadataEntryType.LOGICAL_ID)\n        .find((e) => e.data === logicalId);\n      if (entry) {\n        return {\n          entry,\n          constructPath: this.simplifyConstructPath(path),\n        };\n      }\n    }\n    return undefined;\n  }\n\n  /**\n   * Reads all new events from the stack history\n   *\n   * The events are returned in reverse chronological order; we continue to the next page if we\n   * see a next page and the last event in the page is new to us (and within the time window).\n   * haven't seen the final event\n   */\n  private async readNewEvents(): Promise<void> {\n    const pollEvents = await this.poller.poll();\n\n    const activities: StackActivity[] = pollEvents.map((event) => ({\n      ...event,\n      metadata: this.findMetadataFor(event.event.LogicalResourceId),\n    }));\n\n    for (const activity of activities) {\n      this.checkForErrors(activity);\n      this.printer.addActivity(activity);\n    }\n  }\n\n  /**\n   * Perform a final poll to the end and flush out all events to the printer\n   *\n   * Finish any poll currently in progress, then do a final one until we've\n   * reached the last page.\n   */\n  private async finalPollToEnd() {\n    // If we were doing a poll, finish that first. It was started before\n    // the moment we were sure we weren't going to get any new events anymore\n    // so we need to do a new one anyway. Need to wait for this one though\n    // because our state is single-threaded.\n    if (this.readPromise) {\n      await this.readPromise;\n    }\n\n    await this.readNewEvents();\n  }\n\n  private checkForErrors(activity: StackActivity) {\n    if (hasErrorMessage(activity.event.ResourceStatus ?? '')) {\n      const isCancelled = (activity.event.ResourceStatusReason ?? '').indexOf('cancelled') > -1;\n\n      // Cancelled is not an interesting failure reason, nor is the stack message (stack\n      // message will just say something like \"stack failed to update\")\n      if (!isCancelled && activity.event.StackName !== activity.event.LogicalResourceId) {\n        this.errors.push(activity.event.ResourceStatusReason ?? '');\n      }\n    }\n  }\n\n  private simplifyConstructPath(path: string) {\n    path = path.replace(/\\/Resource$/, '');\n    path = path.replace(/^\\//, ''); // remove \"/\" prefix\n\n    // remove \"<stack-name>/\" prefix\n    if (path.startsWith(this.stackName + '/')) {\n      path = path.slice(this.stackName.length + 1);\n    }\n    return path;\n  }\n}\n\nfunction padRight(n: number, x: string): string {\n  return x + ' '.repeat(Math.max(0, n - x.length));\n}\n\n/**\n * Infamous padLeft()\n */\nfunction padLeft(n: number, x: string): string {\n  return ' '.repeat(Math.max(0, n - x.length)) + x;\n}\n\nfunction calcMaxResourceTypeLength(template: any) {\n  const resources = (template && template.Resources) || {};\n  let maxWidth = 0;\n  for (const id of Object.keys(resources)) {\n    const type = resources[id].Type || '';\n    if (type.length > maxWidth) {\n      maxWidth = type.length;\n    }\n  }\n  return maxWidth;\n}\n\ninterface PrinterProps {\n  /**\n   * Total resources to deploy\n   */\n  readonly resourcesTotal?: number;\n\n  /**\n   * The with of the \"resource type\" column.\n   */\n  readonly resourceTypeColumnWidth: number;\n\n  /**\n   * Stream to write to\n   */\n  readonly stream: NodeJS.WriteStream;\n}\n\nexport interface IActivityPrinter {\n  readonly updateSleep: number;\n\n  addActivity(activity: StackActivity): void;\n  print(): void;\n  start(): void;\n  stop(): void;\n}\n\nabstract class ActivityPrinterBase implements IActivityPrinter {\n  /**\n   * Fetch new activity every 5 seconds\n   */\n  public readonly updateSleep: number = 5_000;\n\n  /**\n   * A list of resource IDs which are currently being processed\n   */\n  protected resourcesInProgress: Record<string, StackActivity> = {};\n\n  /**\n   * Previous completion state observed by logical ID\n   *\n   * We use this to detect that if we see a DELETE_COMPLETE after a\n   * CREATE_COMPLETE, it's actually a rollback and we should DECREASE\n   * resourcesDone instead of increase it\n   */\n  protected resourcesPrevCompleteState: Record<string, string> = {};\n\n  /**\n   * Count of resources that have reported a _COMPLETE status\n   */\n  protected resourcesDone: number = 0;\n\n  /**\n   * How many digits we need to represent the total count (for lining up the status reporting)\n   */\n  protected readonly resourceDigits: number = 0;\n\n  protected readonly resourcesTotal?: number;\n\n  protected rollingBack = false;\n\n  protected readonly failures = new Array<StackActivity>();\n\n  protected hookFailureMap = new Map<string, Map<string, string>>();\n\n  constructor(protected readonly props: PrinterProps) {\n    // +1 because the stack also emits a \"COMPLETE\" event at the end, and that wasn't\n    // counted yet. This makes it line up with the amount of events we expect.\n    this.resourcesTotal = props.resourcesTotal ? props.resourcesTotal + 1 : undefined;\n\n    // How many digits does this number take to represent?\n    this.resourceDigits = this.resourcesTotal ? Math.ceil(Math.log10(this.resourcesTotal)) : 0;\n  }\n\n  public failureReason(activity: StackActivity) {\n    const resourceStatusReason = activity.event.ResourceStatusReason ?? '';\n    const logicalResourceId = activity.event.LogicalResourceId ?? '';\n    const hookFailureReasonMap = this.hookFailureMap.get(logicalResourceId);\n\n    if (hookFailureReasonMap !== undefined) {\n      for (const hookType of hookFailureReasonMap.keys()) {\n        if (resourceStatusReason.includes(hookType)) {\n          return resourceStatusReason + ' : ' + hookFailureReasonMap.get(hookType);\n        }\n      }\n    }\n    return resourceStatusReason;\n  }\n\n  public addActivity(activity: StackActivity) {\n    const status = activity.event.ResourceStatus;\n    const hookStatus = activity.event.HookStatus;\n    const hookType = activity.event.HookType;\n    if (!status || !activity.event.LogicalResourceId) {\n      return;\n    }\n\n    if (status === 'ROLLBACK_IN_PROGRESS' || status === 'UPDATE_ROLLBACK_IN_PROGRESS') {\n      // Only triggered on the stack once we've started doing a rollback\n      this.rollingBack = true;\n    }\n\n    if (status.endsWith('_IN_PROGRESS')) {\n      this.resourcesInProgress[activity.event.LogicalResourceId] = activity;\n    }\n\n    if (hasErrorMessage(status)) {\n      const isCancelled = (activity.event.ResourceStatusReason ?? '').indexOf('cancelled') > -1;\n\n      // Cancelled is not an interesting failure reason\n      if (!isCancelled) {\n        this.failures.push(activity);\n      }\n    }\n\n    if (status.endsWith('_COMPLETE') || status.endsWith('_FAILED')) {\n      delete this.resourcesInProgress[activity.event.LogicalResourceId];\n    }\n\n    if (status.endsWith('_COMPLETE_CLEANUP_IN_PROGRESS')) {\n      this.resourcesDone++;\n    }\n\n    if (status.endsWith('_COMPLETE')) {\n      const prevState = this.resourcesPrevCompleteState[activity.event.LogicalResourceId];\n      if (!prevState) {\n        this.resourcesDone++;\n      } else {\n        // If we completed this before and we're completing it AGAIN, means we're rolling back.\n        // Protect against silly underflow.\n        this.resourcesDone--;\n        if (this.resourcesDone < 0) {\n          this.resourcesDone = 0;\n        }\n      }\n      this.resourcesPrevCompleteState[activity.event.LogicalResourceId] = status;\n    }\n\n    if (\n      hookStatus !== undefined &&\n      hookStatus.endsWith('_COMPLETE_FAILED') &&\n      activity.event.LogicalResourceId !== undefined &&\n      hookType !== undefined\n    ) {\n      if (this.hookFailureMap.has(activity.event.LogicalResourceId)) {\n        this.hookFailureMap.get(activity.event.LogicalResourceId)?.set(hookType, activity.event.HookStatusReason ?? '');\n      } else {\n        this.hookFailureMap.set(activity.event.LogicalResourceId, new Map<string, string>());\n        this.hookFailureMap.get(activity.event.LogicalResourceId)?.set(hookType, activity.event.HookStatusReason ?? '');\n      }\n    }\n  }\n\n  public abstract print(): void;\n\n  public start() {\n    // Empty on purpose\n  }\n\n  public stop() {\n    // Empty on purpose\n  }\n}\n\n/**\n * Activity Printer which shows a full log of all CloudFormation events\n *\n * When there hasn't been activity for a while, it will print the resources\n * that are currently in progress, to show what's holding up the deployment.\n */\nexport class HistoryActivityPrinter extends ActivityPrinterBase {\n  /**\n   * Last time we printed something to the console.\n   *\n   * Used to measure timeout for progress reporting.\n   */\n  private lastPrintTime = Date.now();\n\n  /**\n   * Number of ms of change absence before we tell the user about the resources that are currently in progress.\n   */\n  private readonly inProgressDelay = 30_000;\n\n  private readonly printable = new Array<StackActivity>();\n\n  constructor(props: PrinterProps) {\n    super(props);\n  }\n\n  public addActivity(activity: StackActivity) {\n    super.addActivity(activity);\n    this.printable.push(activity);\n    this.print();\n  }\n\n  public print() {\n    for (const activity of this.printable) {\n      this.printOne(activity);\n    }\n    this.printable.splice(0, this.printable.length);\n    this.printInProgress();\n  }\n\n  public stop() {\n    // Print failures at the end\n    if (this.failures.length > 0) {\n      info('\\nFailed resources:');\n      for (const failure of this.failures) {\n        // Root stack failures are not interesting\n        if (failure.isStackEvent) {\n          continue;\n        }\n\n        this.printOne(failure, false);\n      }\n    }\n  }\n\n  private printOne(activity: StackActivity, progress?: boolean) {\n    const event = activity.event;\n    const color = colorFromStatusResult(event.ResourceStatus);\n    let reasonColor = chalk.cyan;\n\n    let stackTrace = '';\n    const metadata = activity.metadata;\n\n    if (event.ResourceStatus && event.ResourceStatus.indexOf('FAILED') !== -1) {\n      if (progress == undefined || progress) {\n        event.ResourceStatusReason = event.ResourceStatusReason ? this.failureReason(activity) : '';\n      }\n      if (metadata) {\n        stackTrace = metadata.entry.trace ? `\\n\\t${metadata.entry.trace.join('\\n\\t\\\\_ ')}` : '';\n      }\n      reasonColor = chalk.red;\n    }\n\n    const resourceName = metadata ? metadata.constructPath : event.LogicalResourceId || '';\n\n    const logicalId = resourceName !== event.LogicalResourceId ? `(${event.LogicalResourceId}) ` : '';\n\n    info(\n      util.format(\n        '%s | %s%s | %s | %s | %s %s%s%s',\n        event.StackName,\n        progress !== false ? `${this.progress()} | ` : '',\n        new Date(event.Timestamp!).toLocaleTimeString(),\n        color(padRight(STATUS_WIDTH, (event.ResourceStatus || '').slice(0, STATUS_WIDTH))), // pad left and trim\n        padRight(this.props.resourceTypeColumnWidth, event.ResourceType || ''),\n        color(chalk.bold(resourceName)),\n        logicalId,\n        reasonColor(chalk.bold(event.ResourceStatusReason ? event.ResourceStatusReason : '')),\n        reasonColor(stackTrace),\n      ),\n    );\n\n    this.lastPrintTime = Date.now();\n  }\n\n  /**\n   * Report the current progress as a [34/42] string, or just [34] if the total is unknown\n   */\n  private progress(): string {\n    if (this.resourcesTotal == null) {\n      // Don't have total, show simple count and hope the human knows\n      return padLeft(3, util.format('%s', this.resourcesDone)); // max 500 resources\n    }\n\n    return util.format(\n      '%s/%s',\n      padLeft(this.resourceDigits, this.resourcesDone.toString()),\n      padLeft(this.resourceDigits, this.resourcesTotal != null ? this.resourcesTotal.toString() : '?'),\n    );\n  }\n\n  /**\n   * If some resources are taking a while to create, notify the user about what's currently in progress\n   */\n  private printInProgress() {\n    if (Date.now() < this.lastPrintTime + this.inProgressDelay) {\n      return;\n    }\n\n    if (Object.keys(this.resourcesInProgress).length > 0) {\n      info(\n        util.format(\n          '%s Currently in progress: %s',\n          this.progress(),\n          chalk.bold(Object.keys(this.resourcesInProgress).join(', ')),\n        ),\n      );\n    }\n\n    // We cheat a bit here. To prevent printInProgress() from repeatedly triggering,\n    // we set the timestamp into the future. It will be reset whenever a regular print\n    // occurs, after which we can be triggered again.\n    this.lastPrintTime = +Infinity;\n  }\n}\n\n/**\n * Activity Printer which shows the resources currently being updated\n *\n * It will continuously reupdate the terminal and show only the resources\n * that are currently being updated, in addition to a progress bar which\n * shows how far along the deployment is.\n *\n * Resources that have failed will always be shown, and will be recapitulated\n * along with their stack trace when the monitoring ends.\n *\n * Resources that failed deployment because they have been cancelled are\n * not included.\n */\nexport class CurrentActivityPrinter extends ActivityPrinterBase {\n  /**\n   * This looks very disorienting sleeping for 5 seconds. Update quicker.\n   */\n  public readonly updateSleep: number = 2_000;\n\n  private oldLogThreshold: IoMessageLevel;\n  private readonly stream: NodeJS.WriteStream;\n  private block: RewritableBlock;\n\n  constructor(props: PrinterProps) {\n    super(props);\n    this.oldLogThreshold = CliIoHost.instance().logLevel;\n    this.stream = props.stream;\n    this.block = new RewritableBlock(this.stream);\n  }\n\n  public print(): void {\n    const lines = [];\n\n    // Add a progress bar at the top\n    const progressWidth = Math.max(\n      Math.min((this.block.width ?? 80) - PROGRESSBAR_EXTRA_SPACE - 1, MAX_PROGRESSBAR_WIDTH),\n      MIN_PROGRESSBAR_WIDTH,\n    );\n    const prog = this.progressBar(progressWidth);\n    if (prog) {\n      lines.push('  ' + prog, '');\n    }\n\n    // Normally we'd only print \"resources in progress\", but it's also useful\n    // to keep an eye on the failures and know about the specific errors asquickly\n    // as possible (while the stack is still rolling back), so add those in.\n    const toPrint: StackActivity[] = [...this.failures, ...Object.values(this.resourcesInProgress)];\n    toPrint.sort((a, b) => a.event.Timestamp!.getTime() - b.event.Timestamp!.getTime());\n\n    lines.push(\n      ...toPrint.map((res) => {\n        const color = colorFromStatusActivity(res.event.ResourceStatus);\n        const resourceName = res.metadata?.constructPath ?? res.event.LogicalResourceId ?? '';\n\n        return util.format(\n          '%s | %s | %s | %s%s',\n          padLeft(TIMESTAMP_WIDTH, new Date(res.event.Timestamp!).toLocaleTimeString()),\n          color(padRight(STATUS_WIDTH, (res.event.ResourceStatus || '').slice(0, STATUS_WIDTH))),\n          padRight(this.props.resourceTypeColumnWidth, res.event.ResourceType || ''),\n          color(chalk.bold(shorten(40, resourceName))),\n          this.failureReasonOnNextLine(res),\n        );\n      }),\n    );\n\n    this.block.displayLines(lines);\n  }\n\n  public start() {\n    // Need to prevent the waiter from printing 'stack not stable' every 5 seconds, it messes\n    // with the output calculations.\n    this.oldLogThreshold = CliIoHost.instance().logLevel;\n    CliIoHost.instance().logLevel = 'info';\n  }\n\n  public stop() {\n    CliIoHost.instance().logLevel = this.oldLogThreshold;\n\n    // Print failures at the end\n    const lines = new Array<string>();\n    for (const failure of this.failures) {\n      // Root stack failures are not interesting\n      if (failure.isStackEvent) {\n        continue;\n      }\n\n      lines.push(\n        util.format(\n          chalk.red('%s | %s | %s | %s%s') + '\\n',\n          padLeft(TIMESTAMP_WIDTH, new Date(failure.event.Timestamp!).toLocaleTimeString()),\n          padRight(STATUS_WIDTH, (failure.event.ResourceStatus || '').slice(0, STATUS_WIDTH)),\n          padRight(this.props.resourceTypeColumnWidth, failure.event.ResourceType || ''),\n          shorten(40, failure.event.LogicalResourceId ?? ''),\n          this.failureReasonOnNextLine(failure),\n        ),\n      );\n\n      const trace = failure.metadata?.entry?.trace;\n      if (trace) {\n        lines.push(chalk.red(`\\t${trace.join('\\n\\t\\\\_ ')}\\n`));\n      }\n    }\n\n    // Display in the same block space, otherwise we're going to have silly empty lines.\n    this.block.displayLines(lines);\n    this.block.removeEmptyLines();\n  }\n\n  private progressBar(width: number) {\n    if (!this.resourcesTotal) {\n      return '';\n    }\n    const fraction = Math.min(this.resourcesDone / this.resourcesTotal, 1);\n    const innerWidth = Math.max(1, width - 2);\n    const chars = innerWidth * fraction;\n    const remainder = chars - Math.floor(chars);\n\n    const fullChars = FULL_BLOCK.repeat(Math.floor(chars));\n    const partialChar = PARTIAL_BLOCK[Math.floor(remainder * PARTIAL_BLOCK.length)];\n    const filler = '·'.repeat(innerWidth - Math.floor(chars) - (partialChar ? 1 : 0));\n\n    const color = this.rollingBack ? chalk.yellow : chalk.green;\n\n    return '[' + color(fullChars + partialChar) + filler + `] (${this.resourcesDone}/${this.resourcesTotal})`;\n  }\n\n  private failureReasonOnNextLine(activity: StackActivity) {\n    return hasErrorMessage(activity.event.ResourceStatus ?? '')\n      ? `\\n${' '.repeat(TIMESTAMP_WIDTH + STATUS_WIDTH + 6)}${chalk.red(this.failureReason(activity) ?? '')}`\n      : '';\n  }\n}\n\nconst FULL_BLOCK = '█';\nconst PARTIAL_BLOCK = ['', '▏', '▎', '▍', '▌', '▋', '▊', '▉'];\nconst MAX_PROGRESSBAR_WIDTH = 60;\nconst MIN_PROGRESSBAR_WIDTH = 10;\nconst PROGRESSBAR_EXTRA_SPACE =\n  2 /* leading spaces */ + 2 /* brackets */ + 4 /* progress number decoration */ + 6; /* 2 progress numbers up to 999 */\n\nfunction hasErrorMessage(status: string) {\n  return status.endsWith('_FAILED') || status === 'ROLLBACK_IN_PROGRESS' || status === 'UPDATE_ROLLBACK_IN_PROGRESS';\n}\n\nfunction colorFromStatusResult(status?: string) {\n  if (!status) {\n    return chalk.reset;\n  }\n\n  if (status.indexOf('FAILED') !== -1) {\n    return chalk.red;\n  }\n  if (status.indexOf('ROLLBACK') !== -1) {\n    return chalk.yellow;\n  }\n  if (status.indexOf('COMPLETE') !== -1) {\n    return chalk.green;\n  }\n\n  return chalk.reset;\n}\n\nfunction colorFromStatusActivity(status?: string) {\n  if (!status) {\n    return chalk.reset;\n  }\n\n  if (status.endsWith('_FAILED')) {\n    return chalk.red;\n  }\n\n  if (status.startsWith('CREATE_') || status.startsWith('UPDATE_') || status.startsWith('IMPORT_')) {\n    return chalk.green;\n  }\n  // For stacks, it may also be 'UPDDATE_ROLLBACK_IN_PROGRESS'\n  if (status.indexOf('ROLLBACK_') !== -1) {\n    return chalk.yellow;\n  }\n  if (status.startsWith('DELETE_')) {\n    return chalk.yellow;\n  }\n\n  return chalk.reset;\n}\n\nfunction shorten(maxWidth: number, p: string) {\n  if (p.length <= maxWidth) {\n    return p;\n  }\n  const half = Math.floor((maxWidth - 3) / 2);\n  return p.slice(0, half) + '...' + p.slice(-half);\n}\n\nconst TIMESTAMP_WIDTH = 12;\nconst STATUS_WIDTH = 20;\n"]}
166
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-activity-monitor.js","sourceRoot":"","sources":["stack-activity-monitor.ts"],"names":[],"mappings":"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","sourcesContent":["\nimport * as util from 'util';\nimport { ArtifactMetadataEntryType, type MetadataEntry } from '@aws-cdk/cloud-assembly-schema';\nimport type { CloudFormationStackArtifact } from '@aws-cdk/cx-api';\nimport { StackEvent } from '@aws-sdk/client-cloudformation';\nimport * as uuid from 'uuid';\nimport { StackEventPoller } from './stack-event-poller';\nimport { debug, error, info } from '../../cli/messages';\nimport { IoMessaging } from '../../toolkit/cli-io-host';\nimport { stackEventHasErrorMessage } from '../../util';\nimport type { ICloudFormationClient } from '../aws-auth';\nimport { StackProgress, StackProgressMonitor } from './stack-progress-monitor';\n\n/**\n * Payload when stack monitoring is starting or stopping for a given stack deployment.\n */\nexport interface StackMonitoringControlEvent {\n  /**\n   * A unique identifier for a specific stack deployment.\n   *\n   * Use this value to attribute stack activities received for concurrent deployments.\n   */\n  readonly deployment: string;\n\n  /**\n   * The stack artifact that is getting deployed\n   */\n  readonly stack: CloudFormationStackArtifact;\n\n  /**\n   * The name of the Stack that is getting deployed\n   */\n  readonly stackName: string;\n\n  /**\n   * Total number of resources taking part in this deployment\n   *\n   * The number might not always be known or accurate.\n   * Only use for informational purposes and handle the case when it's unavailable.\n   */\n  readonly resourcesTotal?: number;\n}\n\nexport interface StackActivity {\n  /**\n   * A unique identifier for a specific stack deployment.\n   *\n   * Use this value to attribute stack activities received for concurrent deployments.\n   */\n  readonly deployment: string;\n\n  /**\n   * The Stack Event as received from CloudFormation\n   */\n  readonly event: StackEvent;\n\n  /**\n   * Additional resource metadata\n   */\n  readonly metadata?: ResourceMetadata;\n\n  /**\n   * The stack progress\n   */\n  readonly progress: StackProgress;\n}\n\nexport interface ResourceMetadata {\n  entry: MetadataEntry;\n  constructPath: string;\n}\n\nexport interface StackActivityMonitorProps {\n  /**\n   * The CloudFormation client\n   */\n  readonly cfn: ICloudFormationClient;\n\n  /**\n   * The IoHost used for messaging\n   */\n  readonly ioHost: IoMessaging['ioHost'];\n\n  /**\n   * The current ToolkitAction\n   */\n  readonly action: IoMessaging['action'];\n\n  /**\n   * The stack artifact that is getting deployed\n   */\n  readonly stack: CloudFormationStackArtifact;\n\n  /**\n   * The name of the Stack that is getting deployed\n   */\n  readonly stackName: string;\n\n  /**\n   * Total number of resources to update\n   *\n   * Used to calculate a progress bar.\n   *\n   * @default - No progress reporting.\n   */\n  readonly resourcesTotal?: number;\n\n  /**\n   * Creation time of the change set\n   *\n   * This will be used to filter events, only showing those from after the change\n   * set creation time.\n   *\n   * It is recommended to use this, otherwise the filtering will be subject\n   * to clock drift between local and cloud machines.\n   *\n   * @default - local machine's current time\n   */\n  readonly changeSetCreationTime?: Date;\n\n  /**\n   * Time to wait between fetching new activities.\n   *\n   * Must wait a reasonable amount of time between polls, since we need to consider CloudFormation API limits\n   *\n   * @default 2_000\n   */\n  readonly pollingInterval?: number;\n}\n\nexport class StackActivityMonitor {\n  /**\n   * The poller used to read stack events\n   */\n  private readonly poller: StackEventPoller;\n\n  /**\n   * Fetch new activity every 1 second\n   * Printers can decide to update a view less frequently if desired\n   */\n  private readonly pollingInterval: number;\n\n  public readonly errors: string[] = [];\n\n  private monitorId?: string;\n\n  private readonly progressMonitor: StackProgressMonitor;\n\n  /**\n   * Current tick timer\n   */\n  private tickTimer?: ReturnType<typeof setTimeout>;\n\n  /**\n   * Set to the activity of reading the current events\n   */\n  private readPromise?: Promise<any>;\n\n  private readonly ioHost: IoMessaging['ioHost'];\n  private readonly action: IoMessaging['action'];\n  private readonly stackName: string;\n  private readonly stack: CloudFormationStackArtifact;\n\n  constructor({\n    cfn,\n    ioHost,\n    action,\n    stack,\n    stackName,\n    resourcesTotal,\n    changeSetCreationTime,\n    pollingInterval = 2_000,\n  }: StackActivityMonitorProps) {\n    this.ioHost = ioHost;\n    this.action = action;\n    this.stack = stack;\n    this.stackName = stackName;\n\n    this.progressMonitor = new StackProgressMonitor(resourcesTotal);\n    this.pollingInterval = pollingInterval;\n    this.poller = new StackEventPoller(cfn, {\n      stackName,\n      startTime: changeSetCreationTime?.getTime() ?? Date.now(),\n    });\n  }\n\n  public async start() {\n    this.monitorId = uuid.v4();\n    await this.ioHost.notify(debug(this.action, `Deploying ${this.stackName}`, 'CDK_TOOLKIT_I5501', {\n      deployment: this.monitorId,\n      stack: this.stack,\n      stackName: this.stackName,\n      resourcesTotal: this.progressMonitor.total,\n    } as StackMonitoringControlEvent));\n    this.scheduleNextTick();\n    return this;\n  }\n\n  public async stop() {\n    const oldMonitorId = this.monitorId!;\n    this.monitorId = undefined;\n    if (this.tickTimer) {\n      clearTimeout(this.tickTimer);\n    }\n\n    // Do a final poll for all events. This is to handle the situation where DescribeStackStatus\n    // already returned an error, but the monitor hasn't seen all the events yet and we'd end\n    // up not printing the failure reason to users.\n    await this.finalPollToEnd(oldMonitorId);\n\n    await this.ioHost.notify(debug(this.action, `Completed ${this.stackName}`, 'CDK_TOOLKIT_I5503', {\n      deployment: oldMonitorId,\n      stack: this.stack,\n      stackName: this.stackName,\n      resourcesTotal: this.progressMonitor.total,\n    } as StackMonitoringControlEvent));\n  }\n\n  private scheduleNextTick() {\n    if (!this.monitorId) {\n      return;\n    }\n\n    this.tickTimer = setTimeout(() => void this.tick(), this.pollingInterval);\n  }\n\n  private async tick() {\n    if (!this.monitorId) {\n      return;\n    }\n\n    try {\n      this.readPromise = this.readNewEvents(this.monitorId);\n      await this.readPromise;\n      this.readPromise = undefined;\n\n      // We might have been stop()ped while the network call was in progress.\n      if (!this.monitorId) {\n        return;\n      }\n    } catch (e) {\n      await this.ioHost.notify(error(\n        this.action,\n        util.format('Error occurred while monitoring stack: %s', e),\n        'CDK_TOOLKIT_E5500',\n        { error: e },\n      ));\n    }\n    this.scheduleNextTick();\n  }\n\n  private findMetadataFor(logicalId: string | undefined): ResourceMetadata | undefined {\n    const metadata = this.stack.manifest?.metadata;\n    if (!logicalId || !metadata) {\n      return undefined;\n    }\n    for (const path of Object.keys(metadata)) {\n      const entry = metadata[path]\n        .filter((e) => e.type === ArtifactMetadataEntryType.LOGICAL_ID)\n        .find((e) => e.data === logicalId);\n      if (entry) {\n        return {\n          entry,\n          constructPath: this.simplifyConstructPath(path),\n        };\n      }\n    }\n    return undefined;\n  }\n\n  /**\n   * Reads all new events from the stack history\n   *\n   * The events are returned in reverse chronological order; we continue to the next page if we\n   * see a next page and the last event in the page is new to us (and within the time window).\n   * haven't seen the final event\n   */\n  private async readNewEvents(monitorId: string): Promise<void> {\n    const pollEvents = await this.poller.poll();\n\n    for (const resourceEvent of pollEvents) {\n      this.progressMonitor.process(resourceEvent.event);\n\n      const activity: StackActivity = {\n        deployment: monitorId,\n        event: resourceEvent.event,\n        metadata: this.findMetadataFor(resourceEvent.event.LogicalResourceId),\n        progress: this.progressMonitor.progress,\n      };\n\n      this.checkForErrors(activity);\n      await this.ioHost.notify(info(this.action, this.formatActivity(activity, true), 'CDK_TOOLKIT_I5502', activity));\n    }\n  }\n\n  /**\n   * Perform a final poll to the end and flush out all events to the printer\n   *\n   * Finish any poll currently in progress, then do a final one until we've\n   * reached the last page.\n   */\n  private async finalPollToEnd(monitorId: string) {\n    // If we were doing a poll, finish that first. It was started before\n    // the moment we were sure we weren't going to get any new events anymore\n    // so we need to do a new one anyway. Need to wait for this one though\n    // because our state is single-threaded.\n    if (this.readPromise) {\n      await this.readPromise;\n    }\n\n    await this.readNewEvents(monitorId);\n  }\n\n  /**\n   * Formats a stack activity into a basic string\n   */\n  private formatActivity(activity: StackActivity, progress: boolean): string {\n    const event = activity.event;\n    const metadata = activity.metadata;\n\n    const resourceName = metadata ? metadata.constructPath : event.LogicalResourceId || '';\n    const logicalId = resourceName !== event.LogicalResourceId ? `(${event.LogicalResourceId}) ` : '';\n\n    return util.format(\n      '%s | %s%s | %s | %s | %s %s%s%s',\n      event.StackName,\n      progress !== false ? `${activity.progress.formatted} | ` : '',\n      new Date(event.Timestamp!).toLocaleTimeString(),\n      event.ResourceStatus || '',\n      event.ResourceType,\n      resourceName,\n      logicalId,\n      event.ResourceStatusReason ? event.ResourceStatusReason : '',\n      metadata?.entry.trace ? `\\n\\t${metadata.entry.trace.join('\\n\\t\\\\_ ')}` : '',\n    );\n  }\n\n  private checkForErrors(activity: StackActivity) {\n    if (stackEventHasErrorMessage(activity.event.ResourceStatus ?? '')) {\n      const isCancelled = (activity.event.ResourceStatusReason ?? '').indexOf('cancelled') > -1;\n\n      // Cancelled is not an interesting failure reason, nor is the stack message (stack\n      // message will just say something like \"stack failed to update\")\n      if (!isCancelled && activity.event.StackName !== activity.event.LogicalResourceId) {\n        this.errors.push(activity.event.ResourceStatusReason ?? '');\n      }\n    }\n  }\n\n  private simplifyConstructPath(path: string) {\n    path = path.replace(/\\/Resource$/, '');\n    path = path.replace(/^\\//, ''); // remove \"/\" prefix\n\n    // remove \"<stack-name>/\" prefix\n    if (path.startsWith(this.stackName + '/')) {\n      path = path.slice(this.stackName.length + 1);\n    }\n    return path;\n  }\n}\n"]}