aws-cdk 2.1000.3 → 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.
- package/README.md +15 -0
- package/THIRD_PARTY_LICENSES +246 -40
- package/build-info.json +2 -2
- package/lib/api/aws-auth/sdk.d.ts +6 -0
- package/lib/api/aws-auth/sdk.js +9 -1
- package/lib/api/bootstrap/bootstrap-template.yaml +3 -1
- package/lib/api/deployments/deploy-stack.d.ts +0 -20
- package/lib/api/deployments/deploy-stack.js +25 -20
- package/lib/api/deployments/deployments.d.ts +0 -27
- package/lib/api/deployments/deployments.js +13 -13
- package/lib/api/resource-import/importer.d.ts +0 -8
- package/lib/api/resource-import/importer.js +1 -1
- package/lib/api/resource-import/migrator.js +1 -2
- package/lib/api/stack-events/stack-activity-monitor.d.ts +87 -165
- package/lib/api/stack-events/stack-activity-monitor.js +61 -445
- package/lib/api/stack-events/stack-event-poller.d.ts +6 -0
- package/lib/api/stack-events/stack-event-poller.js +1 -1
- package/lib/api/stack-events/stack-progress-monitor.d.ts +61 -0
- package/lib/api/stack-events/stack-progress-monitor.js +94 -0
- package/lib/api/work-graph/work-graph-builder.js +4 -4
- package/lib/cli/activity-printer/base.d.ts +51 -0
- package/lib/cli/activity-printer/base.js +115 -0
- package/lib/cli/activity-printer/current.d.ts +25 -0
- package/lib/cli/activity-printer/current.js +122 -0
- package/lib/cli/activity-printer/history.d.ts +31 -0
- package/lib/cli/activity-printer/history.js +109 -0
- package/lib/cli/activity-printer/index.d.ts +3 -0
- package/lib/cli/activity-printer/index.js +20 -0
- package/lib/cli/cdk-toolkit.d.ts +1 -1
- package/lib/cli/cdk-toolkit.js +10 -9
- package/lib/cli/cli-config.js +5 -4
- package/lib/cli/cli.js +3 -1
- package/lib/cli/convert-to-user-input.js +18 -16
- package/lib/cli/parse-command-line-arguments.js +7 -1
- package/lib/cli/user-input.d.ts +8 -0
- package/lib/cli/user-input.js +1 -1
- package/lib/commands/deploy.d.ts +13 -0
- package/lib/commands/deploy.js +18 -0
- package/lib/context-providers/cc-api-provider.d.ts +34 -0
- package/lib/context-providers/cc-api-provider.js +116 -0
- package/lib/context-providers/index.js +3 -1
- package/lib/index.js +24292 -21248
- package/lib/init.d.ts +5 -1
- package/lib/init.js +11 -8
- package/lib/legacy-exports-source.d.ts +1 -1
- package/lib/legacy-exports-source.js +2 -2
- package/lib/notices.d.ts +48 -5
- package/lib/notices.js +127 -82
- package/lib/toolkit/cli-io-host.d.ts +28 -0
- package/lib/toolkit/cli-io-host.js +74 -2
- package/lib/toolkit/error.d.ts +1 -44
- package/lib/toolkit/error.js +16 -76
- package/lib/tree.d.ts +3 -3
- package/lib/tree.js +4 -4
- package/lib/util/cloudformation.d.ts +12 -0
- package/lib/util/cloudformation.js +27 -1
- package/lib/util/json.d.ts +48 -0
- package/lib/util/json.js +68 -0
- package/lib/util/string-manipulation.d.ts +5 -1
- package/lib/util/string-manipulation.js +11 -5
- package/package.json +25 -23
- /package/lib/{api/stack-events → cli/activity-printer}/display.d.ts +0 -0
- /package/lib/{api/stack-events → cli/activity-printer}/display.js +0 -0
|
@@ -26,7 +26,13 @@ export interface StackEventPollerProps {
|
|
|
26
26
|
readonly stackStatuses?: string[];
|
|
27
27
|
}
|
|
28
28
|
export interface ResourceEvent {
|
|
29
|
+
/**
|
|
30
|
+
* The Stack Event as received from CloudFormation
|
|
31
|
+
*/
|
|
29
32
|
readonly event: StackEvent;
|
|
33
|
+
/**
|
|
34
|
+
* IDs of parent stacks of the resource, in case of resources in nested stacks
|
|
35
|
+
*/
|
|
30
36
|
readonly parentStackLogicalIds: string[];
|
|
31
37
|
/**
|
|
32
38
|
* Whether this event regards the root stack
|
|
@@ -126,4 +126,4 @@ function isStackBeginOperationState(state) {
|
|
|
126
126
|
function isStackTerminalState(state) {
|
|
127
127
|
return !(state !== null && state !== void 0 ? state : '').endsWith('_IN_PROGRESS');
|
|
128
128
|
}
|
|
129
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-event-poller.js","sourceRoot":"","sources":["stack-event-poller.ts"],"names":[],"mappings":"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","sourcesContent":["import type { StackEvent } from '@aws-sdk/client-cloudformation';\nimport { formatErrorMessage } from '../../util/format-error';\nimport type { ICloudFormationClient } from '../aws-auth';\n\nexport interface StackEventPollerProps {\n  /**\n   * The stack to poll\n   */\n  readonly stackName: string;\n\n  /**\n   * IDs of parent stacks of this resource, in case of resources in nested stacks\n   */\n  readonly parentStackLogicalIds?: string[];\n\n  /**\n   * Timestamp for the oldest event we're interested in\n   *\n   * @default - Read all events\n   */\n  readonly startTime?: number;\n\n  /**\n   * Stop reading when we see the stack entering this status\n   *\n   * Should be something like `CREATE_IN_PROGRESS`, `UPDATE_IN_PROGRESS`,\n   * `DELETE_IN_PROGRESS, `ROLLBACK_IN_PROGRESS`.\n   *\n   * @default - Read all events\n   */\n  readonly stackStatuses?: string[];\n}\n\nexport interface ResourceEvent {\n  readonly event: StackEvent;\n  readonly parentStackLogicalIds: string[];\n\n  /**\n   * Whether this event regards the root stack\n   *\n   * @default false\n   */\n  readonly isStackEvent?: boolean;\n}\n\nexport class StackEventPoller {\n  public readonly events: ResourceEvent[] = [];\n  public complete: boolean = false;\n\n  private readonly eventIds = new Set<string>();\n  private readonly nestedStackPollers: Record<string, StackEventPoller> = {};\n\n  constructor(\n    private readonly cfn: ICloudFormationClient,\n    private readonly props: StackEventPollerProps,\n  ) {\n  }\n\n  /**\n   * From all accumulated events, return only the errors\n   */\n  public get resourceErrors(): ResourceEvent[] {\n    return this.events.filter((e) => e.event.ResourceStatus?.endsWith('_FAILED') && !e.isStackEvent);\n  }\n\n  /**\n   * Poll for new stack events\n   *\n   * Will not return events older than events indicated by the constructor filters.\n   *\n   * Recurses into nested stacks, and returns events old-to-new.\n   */\n  public async poll(): Promise<ResourceEvent[]> {\n    const events: ResourceEvent[] = await this.doPoll();\n\n    // Also poll all nested stacks we're currently tracking\n    for (const [logicalId, poller] of Object.entries(this.nestedStackPollers)) {\n      events.push(...(await poller.poll()));\n      if (poller.complete) {\n        delete this.nestedStackPollers[logicalId];\n      }\n    }\n\n    // Return what we have so far\n    events.sort((a, b) => a.event.Timestamp!.valueOf() - b.event.Timestamp!.valueOf());\n    this.events.push(...events);\n    return events;\n  }\n\n  private async doPoll(): Promise<ResourceEvent[]> {\n    const events: ResourceEvent[] = [];\n    try {\n      let nextToken: string | undefined;\n      let finished = false;\n\n      while (!finished) {\n        const page = await this.cfn.describeStackEvents({ StackName: this.props.stackName, NextToken: nextToken });\n        for (const event of page?.StackEvents ?? []) {\n          // Event from before we were interested in 'em\n          if (this.props.startTime !== undefined && event.Timestamp!.valueOf() < this.props.startTime) {\n            return events;\n          }\n\n          // Already seen this one\n          if (this.eventIds.has(event.EventId!)) {\n            return events;\n          }\n          this.eventIds.add(event.EventId!);\n\n          // The events for the stack itself are also included next to events about resources; we can test for them in this way.\n          const isParentStackEvent = event.PhysicalResourceId === event.StackId;\n\n          if (isParentStackEvent && this.props.stackStatuses?.includes(event.ResourceStatus ?? '')) {\n            return events;\n          }\n\n          // Fresh event\n          const resEvent: ResourceEvent = {\n            event: event,\n            parentStackLogicalIds: this.props.parentStackLogicalIds ?? [],\n            isStackEvent: isParentStackEvent,\n          };\n          events.push(resEvent);\n\n          if (\n            !isParentStackEvent &&\n              event.ResourceType === 'AWS::CloudFormation::Stack' &&\n              isStackBeginOperationState(event.ResourceStatus)\n          ) {\n            // If the event is not for `this` stack and has a physical resource Id, recursively call for events in the nested stack\n            this.trackNestedStack(event, [...(this.props.parentStackLogicalIds ?? []), event.LogicalResourceId ?? '']);\n          }\n\n          if (isParentStackEvent && isStackTerminalState(event.ResourceStatus)) {\n            this.complete = true;\n          }\n        }\n\n        nextToken = page?.NextToken;\n        if (nextToken === undefined) {\n          finished = true;\n        }\n      }\n    } catch (e: any) {\n      if (!(e.name === 'ValidationError' && formatErrorMessage(e) === `Stack [${this.props.stackName}] does not exist`)) {\n        throw e;\n      }\n    }\n\n    return events;\n  }\n\n  /**\n   * On the CREATE_IN_PROGRESS, UPDATE_IN_PROGRESS, DELETE_IN_PROGRESS event of a nested stack, poll the nested stack updates\n   */\n  private trackNestedStack(event: StackEvent, parentStackLogicalIds: string[]) {\n    const logicalId = event.LogicalResourceId;\n    const physicalResourceId = event.PhysicalResourceId;\n\n    // The CREATE_IN_PROGRESS event for a Nested Stack is emitted twice; first without a PhysicalResourceId\n    // and then with. Ignore this event if we don't have that property yet.\n    //\n    // (At this point, I also don't trust that logicalId is always going to be there so validate that as well)\n    if (!logicalId || !physicalResourceId) {\n      return;\n    }\n\n    if (!this.nestedStackPollers[logicalId]) {\n      this.nestedStackPollers[logicalId] = new StackEventPoller(this.cfn, {\n        stackName: physicalResourceId,\n        parentStackLogicalIds: parentStackLogicalIds,\n        startTime: event.Timestamp!.valueOf(),\n      });\n    }\n  }\n}\n\nfunction isStackBeginOperationState(state: string | undefined) {\n  return [\n    'CREATE_IN_PROGRESS',\n    'UPDATE_IN_PROGRESS',\n    'DELETE_IN_PROGRESS',\n    'UPDATE_ROLLBACK_IN_PROGRESS',\n    'ROLLBACK_IN_PROGRESS',\n  ].includes(state ?? '');\n}\n\nfunction isStackTerminalState(state: string | undefined) {\n  return !(state ?? '').endsWith('_IN_PROGRESS');\n}\n"]}
|
|
129
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-event-poller.js","sourceRoot":"","sources":["stack-event-poller.ts"],"names":[],"mappings":"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","sourcesContent":["import type { StackEvent } from '@aws-sdk/client-cloudformation';\nimport { formatErrorMessage } from '../../util/format-error';\nimport type { ICloudFormationClient } from '../aws-auth';\n\nexport interface StackEventPollerProps {\n  /**\n   * The stack to poll\n   */\n  readonly stackName: string;\n\n  /**\n   * IDs of parent stacks of this resource, in case of resources in nested stacks\n   */\n  readonly parentStackLogicalIds?: string[];\n\n  /**\n   * Timestamp for the oldest event we're interested in\n   *\n   * @default - Read all events\n   */\n  readonly startTime?: number;\n\n  /**\n   * Stop reading when we see the stack entering this status\n   *\n   * Should be something like `CREATE_IN_PROGRESS`, `UPDATE_IN_PROGRESS`,\n   * `DELETE_IN_PROGRESS, `ROLLBACK_IN_PROGRESS`.\n   *\n   * @default - Read all events\n   */\n  readonly stackStatuses?: string[];\n}\n\nexport interface ResourceEvent {\n  /**\n   * The Stack Event as received from CloudFormation\n   */\n  readonly event: StackEvent;\n\n  /**\n   * IDs of parent stacks of the resource, in case of resources in nested stacks\n   */\n  readonly parentStackLogicalIds: string[];\n\n  /**\n   * Whether this event regards the root stack\n   *\n   * @default false\n   */\n  readonly isStackEvent?: boolean;\n}\n\nexport class StackEventPoller {\n  public readonly events: ResourceEvent[] = [];\n  public complete: boolean = false;\n\n  private readonly eventIds = new Set<string>();\n  private readonly nestedStackPollers: Record<string, StackEventPoller> = {};\n\n  constructor(\n    private readonly cfn: ICloudFormationClient,\n    private readonly props: StackEventPollerProps,\n  ) {\n  }\n\n  /**\n   * From all accumulated events, return only the errors\n   */\n  public get resourceErrors(): ResourceEvent[] {\n    return this.events.filter((e) => e.event.ResourceStatus?.endsWith('_FAILED') && !e.isStackEvent);\n  }\n\n  /**\n   * Poll for new stack events\n   *\n   * Will not return events older than events indicated by the constructor filters.\n   *\n   * Recurses into nested stacks, and returns events old-to-new.\n   */\n  public async poll(): Promise<ResourceEvent[]> {\n    const events: ResourceEvent[] = await this.doPoll();\n\n    // Also poll all nested stacks we're currently tracking\n    for (const [logicalId, poller] of Object.entries(this.nestedStackPollers)) {\n      events.push(...(await poller.poll()));\n      if (poller.complete) {\n        delete this.nestedStackPollers[logicalId];\n      }\n    }\n\n    // Return what we have so far\n    events.sort((a, b) => a.event.Timestamp!.valueOf() - b.event.Timestamp!.valueOf());\n    this.events.push(...events);\n    return events;\n  }\n\n  private async doPoll(): Promise<ResourceEvent[]> {\n    const events: ResourceEvent[] = [];\n    try {\n      let nextToken: string | undefined;\n      let finished = false;\n\n      while (!finished) {\n        const page = await this.cfn.describeStackEvents({ StackName: this.props.stackName, NextToken: nextToken });\n        for (const event of page?.StackEvents ?? []) {\n          // Event from before we were interested in 'em\n          if (this.props.startTime !== undefined && event.Timestamp!.valueOf() < this.props.startTime) {\n            return events;\n          }\n\n          // Already seen this one\n          if (this.eventIds.has(event.EventId!)) {\n            return events;\n          }\n          this.eventIds.add(event.EventId!);\n\n          // The events for the stack itself are also included next to events about resources; we can test for them in this way.\n          const isParentStackEvent = event.PhysicalResourceId === event.StackId;\n\n          if (isParentStackEvent && this.props.stackStatuses?.includes(event.ResourceStatus ?? '')) {\n            return events;\n          }\n\n          // Fresh event\n          const resEvent: ResourceEvent = {\n            event: event,\n            parentStackLogicalIds: this.props.parentStackLogicalIds ?? [],\n            isStackEvent: isParentStackEvent,\n          };\n          events.push(resEvent);\n\n          if (\n            !isParentStackEvent &&\n              event.ResourceType === 'AWS::CloudFormation::Stack' &&\n              isStackBeginOperationState(event.ResourceStatus)\n          ) {\n            // If the event is not for `this` stack and has a physical resource Id, recursively call for events in the nested stack\n            this.trackNestedStack(event, [...(this.props.parentStackLogicalIds ?? []), event.LogicalResourceId ?? '']);\n          }\n\n          if (isParentStackEvent && isStackTerminalState(event.ResourceStatus)) {\n            this.complete = true;\n          }\n        }\n\n        nextToken = page?.NextToken;\n        if (nextToken === undefined) {\n          finished = true;\n        }\n      }\n    } catch (e: any) {\n      if (!(e.name === 'ValidationError' && formatErrorMessage(e) === `Stack [${this.props.stackName}] does not exist`)) {\n        throw e;\n      }\n    }\n\n    return events;\n  }\n\n  /**\n   * On the CREATE_IN_PROGRESS, UPDATE_IN_PROGRESS, DELETE_IN_PROGRESS event of a nested stack, poll the nested stack updates\n   */\n  private trackNestedStack(event: StackEvent, parentStackLogicalIds: string[]) {\n    const logicalId = event.LogicalResourceId;\n    const physicalResourceId = event.PhysicalResourceId;\n\n    // The CREATE_IN_PROGRESS event for a Nested Stack is emitted twice; first without a PhysicalResourceId\n    // and then with. Ignore this event if we don't have that property yet.\n    //\n    // (At this point, I also don't trust that logicalId is always going to be there so validate that as well)\n    if (!logicalId || !physicalResourceId) {\n      return;\n    }\n\n    if (!this.nestedStackPollers[logicalId]) {\n      this.nestedStackPollers[logicalId] = new StackEventPoller(this.cfn, {\n        stackName: physicalResourceId,\n        parentStackLogicalIds: parentStackLogicalIds,\n        startTime: event.Timestamp!.valueOf(),\n      });\n    }\n  }\n}\n\nfunction isStackBeginOperationState(state: string | undefined) {\n  return [\n    'CREATE_IN_PROGRESS',\n    'UPDATE_IN_PROGRESS',\n    'DELETE_IN_PROGRESS',\n    'UPDATE_ROLLBACK_IN_PROGRESS',\n    'ROLLBACK_IN_PROGRESS',\n  ].includes(state ?? '');\n}\n\nfunction isStackTerminalState(state: string | undefined) {\n  return !(state ?? '').endsWith('_IN_PROGRESS');\n}\n"]}
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import { StackEvent } from '@aws-sdk/client-cloudformation';
|
|
2
|
+
export interface StackProgress {
|
|
3
|
+
/**
|
|
4
|
+
* The total number of progress monitored resources.
|
|
5
|
+
*/
|
|
6
|
+
readonly total?: number;
|
|
7
|
+
/**
|
|
8
|
+
* The number of completed resources.
|
|
9
|
+
*/
|
|
10
|
+
readonly completed: number;
|
|
11
|
+
/**
|
|
12
|
+
* The current progress as a [34/42] string, or just [34] if the total is unknown.
|
|
13
|
+
*/
|
|
14
|
+
readonly formatted: string;
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* Monitors stack progress.s
|
|
18
|
+
*/
|
|
19
|
+
export declare class StackProgressMonitor {
|
|
20
|
+
/**
|
|
21
|
+
* Previous completion state observed by logical ID
|
|
22
|
+
*
|
|
23
|
+
* We use this to detect that if we see a DELETE_COMPLETE after a
|
|
24
|
+
* CREATE_COMPLETE, it's actually a rollback and we should DECREASE
|
|
25
|
+
* resourcesDone instead of increase it
|
|
26
|
+
*/
|
|
27
|
+
private resourcesPrevCompleteState;
|
|
28
|
+
/**
|
|
29
|
+
* Count of resources that have reported a _COMPLETE status
|
|
30
|
+
*/
|
|
31
|
+
private resourcesDone;
|
|
32
|
+
/**
|
|
33
|
+
* How many digits we need to represent the total count (for lining up the status reporting)
|
|
34
|
+
*/
|
|
35
|
+
private readonly resourceDigits;
|
|
36
|
+
/**
|
|
37
|
+
* Number of expected resources in the monitor.
|
|
38
|
+
*/
|
|
39
|
+
private readonly resourcesTotal?;
|
|
40
|
+
constructor(resourcesTotal?: number);
|
|
41
|
+
/**
|
|
42
|
+
* Report the stack progress
|
|
43
|
+
*/
|
|
44
|
+
get progress(): StackProgress;
|
|
45
|
+
/**
|
|
46
|
+
* The total number of progress monitored resources.
|
|
47
|
+
*/
|
|
48
|
+
get total(): number | undefined;
|
|
49
|
+
/**
|
|
50
|
+
* The number of completed resources.
|
|
51
|
+
*/
|
|
52
|
+
get completed(): number;
|
|
53
|
+
/**
|
|
54
|
+
* Report the current progress as a [34/42] string, or just [34] if the total is unknown
|
|
55
|
+
*/
|
|
56
|
+
get formatted(): string;
|
|
57
|
+
/**
|
|
58
|
+
* Process as stack event and update the progress state.
|
|
59
|
+
*/
|
|
60
|
+
process(event: StackEvent): void;
|
|
61
|
+
}
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.StackProgressMonitor = void 0;
|
|
4
|
+
const util = require("util");
|
|
5
|
+
const string_manipulation_1 = require("../../util/string-manipulation");
|
|
6
|
+
/**
|
|
7
|
+
* Monitors stack progress.s
|
|
8
|
+
*/
|
|
9
|
+
class StackProgressMonitor {
|
|
10
|
+
constructor(resourcesTotal) {
|
|
11
|
+
/**
|
|
12
|
+
* Previous completion state observed by logical ID
|
|
13
|
+
*
|
|
14
|
+
* We use this to detect that if we see a DELETE_COMPLETE after a
|
|
15
|
+
* CREATE_COMPLETE, it's actually a rollback and we should DECREASE
|
|
16
|
+
* resourcesDone instead of increase it
|
|
17
|
+
*/
|
|
18
|
+
this.resourcesPrevCompleteState = {};
|
|
19
|
+
/**
|
|
20
|
+
* Count of resources that have reported a _COMPLETE status
|
|
21
|
+
*/
|
|
22
|
+
this.resourcesDone = 0;
|
|
23
|
+
/**
|
|
24
|
+
* How many digits we need to represent the total count (for lining up the status reporting)
|
|
25
|
+
*/
|
|
26
|
+
this.resourceDigits = 0;
|
|
27
|
+
// +1 because the stack also emits a "COMPLETE" event at the end, and that wasn't
|
|
28
|
+
// counted yet. This makes it line up with the amount of events we expect.
|
|
29
|
+
this.resourcesTotal = resourcesTotal ? resourcesTotal + 1 : undefined;
|
|
30
|
+
// How many digits does this number take to represent?
|
|
31
|
+
this.resourceDigits = this.resourcesTotal ? Math.ceil(Math.log10(this.resourcesTotal)) : 0;
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Report the stack progress
|
|
35
|
+
*/
|
|
36
|
+
get progress() {
|
|
37
|
+
return {
|
|
38
|
+
total: this.total,
|
|
39
|
+
completed: this.completed,
|
|
40
|
+
formatted: this.formatted,
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* The total number of progress monitored resources.
|
|
45
|
+
*/
|
|
46
|
+
get total() {
|
|
47
|
+
return this.resourcesTotal;
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* The number of completed resources.
|
|
51
|
+
*/
|
|
52
|
+
get completed() {
|
|
53
|
+
return this.resourcesDone;
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Report the current progress as a [34/42] string, or just [34] if the total is unknown
|
|
57
|
+
*/
|
|
58
|
+
get formatted() {
|
|
59
|
+
if (this.resourcesTotal == null) {
|
|
60
|
+
// Don't have total, show simple count and hope the human knows
|
|
61
|
+
return (0, string_manipulation_1.padLeft)(3, util.format('%s', this.resourcesDone)); // max 500 resources
|
|
62
|
+
}
|
|
63
|
+
return util.format('%s/%s', (0, string_manipulation_1.padLeft)(this.resourceDigits, this.resourcesDone.toString()), (0, string_manipulation_1.padLeft)(this.resourceDigits, this.resourcesTotal.toString()));
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Process as stack event and update the progress state.
|
|
67
|
+
*/
|
|
68
|
+
process(event) {
|
|
69
|
+
const status = event.ResourceStatus;
|
|
70
|
+
if (!status || !event.LogicalResourceId) {
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
if (status.endsWith('_COMPLETE_CLEANUP_IN_PROGRESS')) {
|
|
74
|
+
this.resourcesDone++;
|
|
75
|
+
}
|
|
76
|
+
if (status.endsWith('_COMPLETE')) {
|
|
77
|
+
const prevState = this.resourcesPrevCompleteState[event.LogicalResourceId];
|
|
78
|
+
if (!prevState) {
|
|
79
|
+
this.resourcesDone++;
|
|
80
|
+
}
|
|
81
|
+
else {
|
|
82
|
+
// If we completed this before and we're completing it AGAIN, means we're rolling back.
|
|
83
|
+
// Protect against silly underflow.
|
|
84
|
+
this.resourcesDone--;
|
|
85
|
+
if (this.resourcesDone < 0) {
|
|
86
|
+
this.resourcesDone = 0;
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
this.resourcesPrevCompleteState[event.LogicalResourceId] = status;
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
exports.StackProgressMonitor = StackProgressMonitor;
|
|
94
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhY2stcHJvZ3Jlc3MtbW9uaXRvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInN0YWNrLXByb2dyZXNzLW1vbml0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQ0EsNkJBQTZCO0FBRTdCLHdFQUF5RDtBQW1CekQ7O0dBRUc7QUFDSCxNQUFhLG9CQUFvQjtJQXlCL0IsWUFBWSxjQUF1QjtRQXhCbkM7Ozs7OztXQU1HO1FBQ0ssK0JBQTBCLEdBQTJCLEVBQUUsQ0FBQztRQUVoRTs7V0FFRztRQUNLLGtCQUFhLEdBQVcsQ0FBQyxDQUFDO1FBRWxDOztXQUVHO1FBQ2MsbUJBQWMsR0FBVyxDQUFDLENBQUM7UUFRMUMsaUZBQWlGO1FBQ2pGLDBFQUEwRTtRQUMxRSxJQUFJLENBQUMsY0FBYyxHQUFHLGNBQWMsQ0FBQyxDQUFDLENBQUMsY0FBYyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDO1FBRXRFLHNEQUFzRDtRQUN0RCxJQUFJLENBQUMsY0FBYyxHQUFHLElBQUksQ0FBQyxjQUFjLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsY0FBYyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQzdGLENBQUM7SUFFRDs7T0FFRztJQUNILElBQVcsUUFBUTtRQUNqQixPQUFPO1lBQ0wsS0FBSyxFQUFFLElBQUksQ0FBQyxLQUFLO1lBQ2pCLFNBQVMsRUFBRSxJQUFJLENBQUMsU0FBUztZQUN6QixTQUFTLEVBQUUsSUFBSSxDQUFDLFNBQVM7U0FDMUIsQ0FBQztJQUNKLENBQUM7SUFFRDs7T0FFRztJQUNILElBQVcsS0FBSztRQUNkLE9BQU8sSUFBSSxDQUFDLGNBQWMsQ0FBQztJQUM3QixDQUFDO0lBRUQ7O09BRUc7SUFDSCxJQUFXLFNBQVM7UUFDbEIsT0FBTyxJQUFJLENBQUMsYUFBYSxDQUFDO0lBQzVCLENBQUM7SUFFRDs7T0FFRztJQUNILElBQVcsU0FBUztRQUNsQixJQUFJLElBQUksQ0FBQyxjQUFjLElBQUksSUFBSSxFQUFFLENBQUM7WUFDaEMsK0RBQStEO1lBQy9ELE9BQU8sSUFBQSw2QkFBTyxFQUFDLENBQUMsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksRUFBRSxJQUFJLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQyxDQUFDLG9CQUFvQjtRQUNoRixDQUFDO1FBRUQsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUNoQixPQUFPLEVBQ1AsSUFBQSw2QkFBTyxFQUFDLElBQUksQ0FBQyxjQUFjLEVBQUUsSUFBSSxDQUFDLGFBQWEsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxFQUMzRCxJQUFBLDZCQUFPLEVBQUMsSUFBSSxDQUFDLGNBQWMsRUFBRSxJQUFJLENBQUMsY0FBYyxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQzdELENBQUM7SUFDSixDQUFDO0lBRUQ7O09BRUc7SUFDSSxPQUFPLENBQUMsS0FBaUI7UUFDOUIsTUFBTSxNQUFNLEdBQUcsS0FBSyxDQUFDLGNBQWMsQ0FBQztRQUNwQyxJQUFJLENBQUMsTUFBTSxJQUFJLENBQUMsS0FBSyxDQUFDLGlCQUFpQixFQUFFLENBQUM7WUFDeEMsT0FBTztRQUNULENBQUM7UUFFRCxJQUFJLE1BQU0sQ0FBQyxRQUFRLENBQUMsK0JBQStCLENBQUMsRUFBRSxDQUFDO1lBQ3JELElBQUksQ0FBQyxhQUFhLEVBQUUsQ0FBQztRQUN2QixDQUFDO1FBRUQsSUFBSSxNQUFNLENBQUMsUUFBUSxDQUFDLFdBQVcsQ0FBQyxFQUFFLENBQUM7WUFDakMsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLDBCQUEwQixDQUFDLEtBQUssQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO1lBQzNFLElBQUksQ0FBQyxTQUFTLEVBQUUsQ0FBQztnQkFDZixJQUFJLENBQUMsYUFBYSxFQUFFLENBQUM7WUFDdkIsQ0FBQztpQkFBTSxDQUFDO2dCQUNOLHVGQUF1RjtnQkFDdkYsbUNBQW1DO2dCQUNuQyxJQUFJLENBQUMsYUFBYSxFQUFFLENBQUM7Z0JBQ3JCLElBQUksSUFBSSxDQUFDLGFBQWEsR0FBRyxDQUFDLEVBQUUsQ0FBQztvQkFDM0IsSUFBSSxDQUFDLGFBQWEsR0FBRyxDQUFDLENBQUM7Z0JBQ3pCLENBQUM7WUFDSCxDQUFDO1lBQ0QsSUFBSSxDQUFDLDBCQUEwQixDQUFDLEtBQUssQ0FBQyxpQkFBaUIsQ0FBQyxHQUFHLE1BQU0sQ0FBQztRQUNwRSxDQUFDO0lBQ0gsQ0FBQztDQUNGO0FBdkdELG9EQXVHQyIsInNvdXJjZXNDb250ZW50IjpbIlxuaW1wb3J0ICogYXMgdXRpbCBmcm9tICd1dGlsJztcbmltcG9ydCB7IFN0YWNrRXZlbnQgfSBmcm9tICdAYXdzLXNkay9jbGllbnQtY2xvdWRmb3JtYXRpb24nO1xuaW1wb3J0IHsgcGFkTGVmdCB9IGZyb20gJy4uLy4uL3V0aWwvc3RyaW5nLW1hbmlwdWxhdGlvbic7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3RhY2tQcm9ncmVzcyB7XG4gIC8qKlxuICAgKiBUaGUgdG90YWwgbnVtYmVyIG9mIHByb2dyZXNzIG1vbml0b3JlZCByZXNvdXJjZXMuXG4gICAqL1xuICByZWFkb25seSB0b3RhbD86IG51bWJlcjtcblxuICAvKipcbiAgICogVGhlIG51bWJlciBvZiBjb21wbGV0ZWQgcmVzb3VyY2VzLlxuICAgKi9cbiAgcmVhZG9ubHkgY29tcGxldGVkOiBudW1iZXI7XG5cbiAgLyoqXG4gICAqIFRoZSBjdXJyZW50IHByb2dyZXNzIGFzIGEgWzM0LzQyXSBzdHJpbmcsIG9yIGp1c3QgWzM0XSBpZiB0aGUgdG90YWwgaXMgdW5rbm93bi5cbiAgICovXG4gIHJlYWRvbmx5IGZvcm1hdHRlZDogc3RyaW5nO1xufVxuXG4vKipcbiAqIE1vbml0b3JzIHN0YWNrIHByb2dyZXNzLnNcbiAqL1xuZXhwb3J0IGNsYXNzIFN0YWNrUHJvZ3Jlc3NNb25pdG9yIHtcbiAgLyoqXG4gICAqIFByZXZpb3VzIGNvbXBsZXRpb24gc3RhdGUgb2JzZXJ2ZWQgYnkgbG9naWNhbCBJRFxuICAgKlxuICAgKiBXZSB1c2UgdGhpcyB0byBkZXRlY3QgdGhhdCBpZiB3ZSBzZWUgYSBERUxFVEVfQ09NUExFVEUgYWZ0ZXIgYVxuICAgKiBDUkVBVEVfQ09NUExFVEUsIGl0J3MgYWN0dWFsbHkgYSByb2xsYmFjayBhbmQgd2Ugc2hvdWxkIERFQ1JFQVNFXG4gICAqIHJlc291cmNlc0RvbmUgaW5zdGVhZCBvZiBpbmNyZWFzZSBpdFxuICAgKi9cbiAgcHJpdmF0ZSByZXNvdXJjZXNQcmV2Q29tcGxldGVTdGF0ZTogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHt9O1xuXG4gIC8qKlxuICAgKiBDb3VudCBvZiByZXNvdXJjZXMgdGhhdCBoYXZlIHJlcG9ydGVkIGEgX0NPTVBMRVRFIHN0YXR1c1xuICAgKi9cbiAgcHJpdmF0ZSByZXNvdXJjZXNEb25lOiBudW1iZXIgPSAwO1xuXG4gIC8qKlxuICAgKiBIb3cgbWFueSBkaWdpdHMgd2UgbmVlZCB0byByZXByZXNlbnQgdGhlIHRvdGFsIGNvdW50IChmb3IgbGluaW5nIHVwIHRoZSBzdGF0dXMgcmVwb3J0aW5nKVxuICAgKi9cbiAgcHJpdmF0ZSByZWFkb25seSByZXNvdXJjZURpZ2l0czogbnVtYmVyID0gMDtcblxuICAvKipcbiAgICogTnVtYmVyIG9mIGV4cGVjdGVkIHJlc291cmNlcyBpbiB0aGUgbW9uaXRvci5cbiAgICovXG4gIHByaXZhdGUgcmVhZG9ubHkgcmVzb3VyY2VzVG90YWw/OiBudW1iZXI7XG5cbiAgY29uc3RydWN0b3IocmVzb3VyY2VzVG90YWw/OiBudW1iZXIpIHtcbiAgICAvLyArMSBiZWNhdXNlIHRoZSBzdGFjayBhbHNvIGVtaXRzIGEgXCJDT01QTEVURVwiIGV2ZW50IGF0IHRoZSBlbmQsIGFuZCB0aGF0IHdhc24ndFxuICAgIC8vIGNvdW50ZWQgeWV0LiBUaGlzIG1ha2VzIGl0IGxpbmUgdXAgd2l0aCB0aGUgYW1vdW50IG9mIGV2ZW50cyB3ZSBleHBlY3QuXG4gICAgdGhpcy5yZXNvdXJjZXNUb3RhbCA9IHJlc291cmNlc1RvdGFsID8gcmVzb3VyY2VzVG90YWwgKyAxIDogdW5kZWZpbmVkO1xuXG4gICAgLy8gSG93IG1hbnkgZGlnaXRzIGRvZXMgdGhpcyBudW1iZXIgdGFrZSB0byByZXByZXNlbnQ/XG4gICAgdGhpcy5yZXNvdXJjZURpZ2l0cyA9IHRoaXMucmVzb3VyY2VzVG90YWwgPyBNYXRoLmNlaWwoTWF0aC5sb2cxMCh0aGlzLnJlc291cmNlc1RvdGFsKSkgOiAwO1xuICB9XG5cbiAgLyoqXG4gICAqIFJlcG9ydCB0aGUgc3RhY2sgcHJvZ3Jlc3NcbiAgICovXG4gIHB1YmxpYyBnZXQgcHJvZ3Jlc3MoKTogU3RhY2tQcm9ncmVzcyB7XG4gICAgcmV0dXJuIHtcbiAgICAgIHRvdGFsOiB0aGlzLnRvdGFsLFxuICAgICAgY29tcGxldGVkOiB0aGlzLmNvbXBsZXRlZCxcbiAgICAgIGZvcm1hdHRlZDogdGhpcy5mb3JtYXR0ZWQsXG4gICAgfTtcbiAgfVxuXG4gIC8qKlxuICAgKiBUaGUgdG90YWwgbnVtYmVyIG9mIHByb2dyZXNzIG1vbml0b3JlZCByZXNvdXJjZXMuXG4gICAqL1xuICBwdWJsaWMgZ2V0IHRvdGFsKCk6IG51bWJlciB8IHVuZGVmaW5lZCB7XG4gICAgcmV0dXJuIHRoaXMucmVzb3VyY2VzVG90YWw7XG4gIH1cblxuICAvKipcbiAgICogVGhlIG51bWJlciBvZiBjb21wbGV0ZWQgcmVzb3VyY2VzLlxuICAgKi9cbiAgcHVibGljIGdldCBjb21wbGV0ZWQoKTogbnVtYmVyIHtcbiAgICByZXR1cm4gdGhpcy5yZXNvdXJjZXNEb25lO1xuICB9XG5cbiAgLyoqXG4gICAqIFJlcG9ydCB0aGUgY3VycmVudCBwcm9ncmVzcyBhcyBhIFszNC80Ml0gc3RyaW5nLCBvciBqdXN0IFszNF0gaWYgdGhlIHRvdGFsIGlzIHVua25vd25cbiAgICovXG4gIHB1YmxpYyBnZXQgZm9ybWF0dGVkKCk6IHN0cmluZyB7XG4gICAgaWYgKHRoaXMucmVzb3VyY2VzVG90YWwgPT0gbnVsbCkge1xuICAgICAgLy8gRG9uJ3QgaGF2ZSB0b3RhbCwgc2hvdyBzaW1wbGUgY291bnQgYW5kIGhvcGUgdGhlIGh1bWFuIGtub3dzXG4gICAgICByZXR1cm4gcGFkTGVmdCgzLCB1dGlsLmZvcm1hdCgnJXMnLCB0aGlzLnJlc291cmNlc0RvbmUpKTsgLy8gbWF4IDUwMCByZXNvdXJjZXNcbiAgICB9XG5cbiAgICByZXR1cm4gdXRpbC5mb3JtYXQoXG4gICAgICAnJXMvJXMnLFxuICAgICAgcGFkTGVmdCh0aGlzLnJlc291cmNlRGlnaXRzLCB0aGlzLnJlc291cmNlc0RvbmUudG9TdHJpbmcoKSksXG4gICAgICBwYWRMZWZ0KHRoaXMucmVzb3VyY2VEaWdpdHMsIHRoaXMucmVzb3VyY2VzVG90YWwudG9TdHJpbmcoKSksXG4gICAgKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBQcm9jZXNzIGFzIHN0YWNrIGV2ZW50IGFuZCB1cGRhdGUgdGhlIHByb2dyZXNzIHN0YXRlLlxuICAgKi9cbiAgcHVibGljIHByb2Nlc3MoZXZlbnQ6IFN0YWNrRXZlbnQpOiB2b2lkIHtcbiAgICBjb25zdCBzdGF0dXMgPSBldmVudC5SZXNvdXJjZVN0YXR1cztcbiAgICBpZiAoIXN0YXR1cyB8fCAhZXZlbnQuTG9naWNhbFJlc291cmNlSWQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBpZiAoc3RhdHVzLmVuZHNXaXRoKCdfQ09NUExFVEVfQ0xFQU5VUF9JTl9QUk9HUkVTUycpKSB7XG4gICAgICB0aGlzLnJlc291cmNlc0RvbmUrKztcbiAgICB9XG5cbiAgICBpZiAoc3RhdHVzLmVuZHNXaXRoKCdfQ09NUExFVEUnKSkge1xuICAgICAgY29uc3QgcHJldlN0YXRlID0gdGhpcy5yZXNvdXJjZXNQcmV2Q29tcGxldGVTdGF0ZVtldmVudC5Mb2dpY2FsUmVzb3VyY2VJZF07XG4gICAgICBpZiAoIXByZXZTdGF0ZSkge1xuICAgICAgICB0aGlzLnJlc291cmNlc0RvbmUrKztcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIElmIHdlIGNvbXBsZXRlZCB0aGlzIGJlZm9yZSBhbmQgd2UncmUgY29tcGxldGluZyBpdCBBR0FJTiwgbWVhbnMgd2UncmUgcm9sbGluZyBiYWNrLlxuICAgICAgICAvLyBQcm90ZWN0IGFnYWluc3Qgc2lsbHkgdW5kZXJmbG93LlxuICAgICAgICB0aGlzLnJlc291cmNlc0RvbmUtLTtcbiAgICAgICAgaWYgKHRoaXMucmVzb3VyY2VzRG9uZSA8IDApIHtcbiAgICAgICAgICB0aGlzLnJlc291cmNlc0RvbmUgPSAwO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgICB0aGlzLnJlc291cmNlc1ByZXZDb21wbGV0ZVN0YXRlW2V2ZW50LkxvZ2ljYWxSZXNvdXJjZUlkXSA9IHN0YXR1cztcbiAgICB9XG4gIH1cbn1cbiJdfQ==
|
|
@@ -155,8 +155,8 @@ WorkGraphBuilder.PRIORITIES = {
|
|
|
155
155
|
};
|
|
156
156
|
function stacksFromAssets(artifacts) {
|
|
157
157
|
const ret = new Map();
|
|
158
|
-
for (const stack of artifacts.filter(cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact)) {
|
|
159
|
-
const assetArtifacts = stack.dependencies.filter(cxapi.AssetManifestArtifact.isAssetManifestArtifact);
|
|
158
|
+
for (const stack of artifacts.filter(x => cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(x))) {
|
|
159
|
+
const assetArtifacts = stack.dependencies.filter((x) => cxapi.AssetManifestArtifact.isAssetManifestArtifact(x));
|
|
160
160
|
for (const art of assetArtifacts) {
|
|
161
161
|
ret.set(art, stack);
|
|
162
162
|
}
|
|
@@ -164,6 +164,6 @@ function stacksFromAssets(artifacts) {
|
|
|
164
164
|
return ret;
|
|
165
165
|
}
|
|
166
166
|
function onlyStacks(artifacts) {
|
|
167
|
-
return artifacts.filter(cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact);
|
|
167
|
+
return artifacts.filter(x => cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(x));
|
|
168
168
|
}
|
|
169
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"work-graph-builder.js","sourceRoot":"","sources":["work-graph-builder.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport { AssetManifest, type IManifestEntry } from 'cdk-assets';\nimport { WorkGraph } from './work-graph';\nimport { DeploymentState, AssetBuildNode, WorkNode } from './work-graph-types';\nimport { IoMessaging } from '../../toolkit/cli-io-host';\nimport { ToolkitError } from '../../toolkit/error';\nimport { contentHashAny } from '../../util/content-hash';\n\nexport class WorkGraphBuilder {\n  /**\n   * Default priorities for nodes\n   *\n   * Assets builds have higher priority than the other two operations, to make good on our promise that\n   * '--prebuild-assets' will actually do assets before stacks (if it can). Unfortunately it is the\n   * default :(\n   *\n   * But between stack dependencies and publish dependencies, stack dependencies go first\n   */\n  public static PRIORITIES: Record<WorkNode['type'], number> = {\n    'asset-build': 10,\n    'asset-publish': 0,\n    'stack': 5,\n  };\n  private readonly graph: WorkGraph;\n  private readonly ioHost: IoMessaging['ioHost'];\n  private readonly action: IoMessaging['action'];\n\n  constructor(\n    { ioHost, action }: IoMessaging,\n    private readonly prebuildAssets: boolean,\n    private readonly idPrefix = '',\n  ) {\n    this.graph = new WorkGraph({}, { ioHost, action });\n    this.ioHost = ioHost;\n    this.action = action;\n  }\n\n  private addStack(artifact: cxapi.CloudFormationStackArtifact) {\n    this.graph.addNodes({\n      type: 'stack',\n      id: `${this.idPrefix}${artifact.id}`,\n      dependencies: new Set(this.stackArtifactIds(onlyStacks(artifact.dependencies))),\n      stack: artifact,\n      deploymentState: DeploymentState.PENDING,\n      priority: WorkGraphBuilder.PRIORITIES.stack,\n    });\n  }\n\n  /**\n   * Oof, see this parameter list\n   */\n  // eslint-disable-next-line max-len\n  private addAsset(parentStack: cxapi.CloudFormationStackArtifact, assetManifestArtifact: cxapi.AssetManifestArtifact, assetManifest: AssetManifest, asset: IManifestEntry) {\n    // Just the artifact identifier\n    const assetId = asset.id.assetId;\n\n    const buildId = `build-${assetId}-${contentHashAny([assetId, asset.genericSource]).substring(0, 10)}`;\n    const publishId = `publish-${assetId}-${contentHashAny([assetId, asset.genericDestination]).substring(0, 10)}`;\n\n    // Build node only gets added once because they are all the same\n    if (!this.graph.tryGetNode(buildId)) {\n      const node: AssetBuildNode = {\n        type: 'asset-build',\n        id: buildId,\n        note: assetId,\n        dependencies: new Set([\n          ...this.stackArtifactIds(assetManifestArtifact.dependencies),\n          // If we disable prebuild, then assets inherit (stack) dependencies from their parent stack\n          ...!this.prebuildAssets ? this.stackArtifactIds(onlyStacks(parentStack.dependencies)) : [],\n        ]),\n        parentStack: parentStack,\n        assetManifestArtifact,\n        assetManifest,\n        asset,\n        deploymentState: DeploymentState.PENDING,\n        priority: WorkGraphBuilder.PRIORITIES['asset-build'],\n      };\n      this.graph.addNodes(node);\n    }\n\n    const publishNode = this.graph.tryGetNode(publishId);\n    if (!publishNode) {\n      this.graph.addNodes({\n        type: 'asset-publish',\n        id: publishId,\n        note: `${asset.id}`,\n        dependencies: new Set([\n          buildId,\n        ]),\n        parentStack,\n        assetManifestArtifact,\n        assetManifest,\n        asset,\n        deploymentState: DeploymentState.PENDING,\n        priority: WorkGraphBuilder.PRIORITIES['asset-publish'],\n      });\n    }\n\n    for (const inheritedDep of this.stackArtifactIds(onlyStacks(parentStack.dependencies))) {\n      // The asset publish step also depends on the stacks that the parent depends on.\n      // This is purely cosmetic: if we don't do this, the progress printing of asset publishing\n      // is going to interfere with the progress bar of the stack deployment. We could remove this\n      // for overall faster deployments if we ever have a better method of progress displaying.\n      // Note: this may introduce a cycle if one of the parent's dependencies is another stack that\n      // depends on this asset. To workaround this we remove these cycles once all nodes have\n      // been added to the graph.\n      this.graph.addDependency(publishId, inheritedDep);\n    }\n\n    // This will work whether the stack node has been added yet or not\n    this.graph.addDependency(`${this.idPrefix}${parentStack.id}`, publishId);\n  }\n\n  public build(artifacts: cxapi.CloudArtifact[]): WorkGraph {\n    const parentStacks = stacksFromAssets(artifacts);\n\n    for (const artifact of artifacts) {\n      if (cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(artifact)) {\n        this.addStack(artifact);\n      } else if (cxapi.AssetManifestArtifact.isAssetManifestArtifact(artifact)) {\n        const manifest = AssetManifest.fromFile(artifact.file);\n\n        for (const entry of manifest.entries) {\n          const parentStack = parentStacks.get(artifact);\n          if (parentStack === undefined) {\n            throw new ToolkitError('Found an asset manifest that is not associated with a stack');\n          }\n          this.addAsset(parentStack, artifact, manifest, entry);\n        }\n      } else if (cxapi.NestedCloudAssemblyArtifact.isNestedCloudAssemblyArtifact(artifact)) {\n        const assembly = new cxapi.CloudAssembly(artifact.fullPath, { topoSort: false });\n        const nestedGraph = new WorkGraphBuilder(\n          { ioHost: this.ioHost, action: this.action },\n          this.prebuildAssets,\n          `${this.idPrefix}${artifact.id}.`,\n        ).build(assembly.artifacts);\n        this.graph.absorb(nestedGraph);\n      } else {\n        // Ignore whatever else\n      }\n    }\n\n    this.graph.removeUnavailableDependencies();\n\n    // Remove any potentially introduced cycles between asset publishing and the stacks that depend on them.\n    this.removeStackPublishCycles();\n\n    return this.graph;\n  }\n\n  private stackArtifactIds(deps: cxapi.CloudArtifact[]): string[] {\n    return deps.flatMap((d) => cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(d) ? [this.stackArtifactId(d)] : []);\n  }\n\n  private stackArtifactId(artifact: cxapi.CloudArtifact): string {\n    if (!cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(artifact)) {\n      throw new ToolkitError(`Can only call this on CloudFormationStackArtifact, got: ${artifact.constructor.name}`);\n    }\n    return `${this.idPrefix}${artifact.id}`;\n  }\n\n  /**\n   * We may have accidentally introduced cycles in an attempt to make the messages printed to the\n   * console not interfere with each other too much. Remove them again.\n   */\n  private removeStackPublishCycles() {\n    const publishSteps = this.graph.nodesOfType('asset-publish');\n    for (const publishStep of publishSteps) {\n      for (const dep of publishStep.dependencies) {\n        if (this.graph.reachable(dep, publishStep.id)) {\n          publishStep.dependencies.delete(dep);\n        }\n      }\n    }\n  }\n}\n\nfunction stacksFromAssets(artifacts: cxapi.CloudArtifact[]) {\n  const ret = new Map<cxapi.AssetManifestArtifact, cxapi.CloudFormationStackArtifact>();\n  for (const stack of artifacts.filter(cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact)) {\n    const assetArtifacts = stack.dependencies.filter(cxapi.AssetManifestArtifact.isAssetManifestArtifact);\n    for (const art of assetArtifacts) {\n      ret.set(art, stack);\n    }\n  }\n\n  return ret;\n}\n\nfunction onlyStacks(artifacts: cxapi.CloudArtifact[]) {\n  return artifacts.filter(cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact);\n}\n"]}
|
|
169
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"work-graph-builder.js","sourceRoot":"","sources":["work-graph-builder.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport { AssetManifest, type IManifestEntry } from 'cdk-assets';\nimport { WorkGraph } from './work-graph';\nimport { DeploymentState, AssetBuildNode, WorkNode } from './work-graph-types';\nimport { IoMessaging } from '../../toolkit/cli-io-host';\nimport { ToolkitError } from '../../toolkit/error';\nimport { contentHashAny } from '../../util/content-hash';\n\nexport class WorkGraphBuilder {\n  /**\n   * Default priorities for nodes\n   *\n   * Assets builds have higher priority than the other two operations, to make good on our promise that\n   * '--prebuild-assets' will actually do assets before stacks (if it can). Unfortunately it is the\n   * default :(\n   *\n   * But between stack dependencies and publish dependencies, stack dependencies go first\n   */\n  public static PRIORITIES: Record<WorkNode['type'], number> = {\n    'asset-build': 10,\n    'asset-publish': 0,\n    'stack': 5,\n  };\n  private readonly graph: WorkGraph;\n  private readonly ioHost: IoMessaging['ioHost'];\n  private readonly action: IoMessaging['action'];\n\n  constructor(\n    { ioHost, action }: IoMessaging,\n    private readonly prebuildAssets: boolean,\n    private readonly idPrefix = '',\n  ) {\n    this.graph = new WorkGraph({}, { ioHost, action });\n    this.ioHost = ioHost;\n    this.action = action;\n  }\n\n  private addStack(artifact: cxapi.CloudFormationStackArtifact) {\n    this.graph.addNodes({\n      type: 'stack',\n      id: `${this.idPrefix}${artifact.id}`,\n      dependencies: new Set(this.stackArtifactIds(onlyStacks(artifact.dependencies))),\n      stack: artifact,\n      deploymentState: DeploymentState.PENDING,\n      priority: WorkGraphBuilder.PRIORITIES.stack,\n    });\n  }\n\n  /**\n   * Oof, see this parameter list\n   */\n  // eslint-disable-next-line max-len\n  private addAsset(parentStack: cxapi.CloudFormationStackArtifact, assetManifestArtifact: cxapi.AssetManifestArtifact, assetManifest: AssetManifest, asset: IManifestEntry) {\n    // Just the artifact identifier\n    const assetId = asset.id.assetId;\n\n    const buildId = `build-${assetId}-${contentHashAny([assetId, asset.genericSource]).substring(0, 10)}`;\n    const publishId = `publish-${assetId}-${contentHashAny([assetId, asset.genericDestination]).substring(0, 10)}`;\n\n    // Build node only gets added once because they are all the same\n    if (!this.graph.tryGetNode(buildId)) {\n      const node: AssetBuildNode = {\n        type: 'asset-build',\n        id: buildId,\n        note: assetId,\n        dependencies: new Set([\n          ...this.stackArtifactIds(assetManifestArtifact.dependencies),\n          // If we disable prebuild, then assets inherit (stack) dependencies from their parent stack\n          ...!this.prebuildAssets ? this.stackArtifactIds(onlyStacks(parentStack.dependencies)) : [],\n        ]),\n        parentStack: parentStack,\n        assetManifestArtifact,\n        assetManifest,\n        asset,\n        deploymentState: DeploymentState.PENDING,\n        priority: WorkGraphBuilder.PRIORITIES['asset-build'],\n      };\n      this.graph.addNodes(node);\n    }\n\n    const publishNode = this.graph.tryGetNode(publishId);\n    if (!publishNode) {\n      this.graph.addNodes({\n        type: 'asset-publish',\n        id: publishId,\n        note: `${asset.id}`,\n        dependencies: new Set([\n          buildId,\n        ]),\n        parentStack,\n        assetManifestArtifact,\n        assetManifest,\n        asset,\n        deploymentState: DeploymentState.PENDING,\n        priority: WorkGraphBuilder.PRIORITIES['asset-publish'],\n      });\n    }\n\n    for (const inheritedDep of this.stackArtifactIds(onlyStacks(parentStack.dependencies))) {\n      // The asset publish step also depends on the stacks that the parent depends on.\n      // This is purely cosmetic: if we don't do this, the progress printing of asset publishing\n      // is going to interfere with the progress bar of the stack deployment. We could remove this\n      // for overall faster deployments if we ever have a better method of progress displaying.\n      // Note: this may introduce a cycle if one of the parent's dependencies is another stack that\n      // depends on this asset. To workaround this we remove these cycles once all nodes have\n      // been added to the graph.\n      this.graph.addDependency(publishId, inheritedDep);\n    }\n\n    // This will work whether the stack node has been added yet or not\n    this.graph.addDependency(`${this.idPrefix}${parentStack.id}`, publishId);\n  }\n\n  public build(artifacts: cxapi.CloudArtifact[]): WorkGraph {\n    const parentStacks = stacksFromAssets(artifacts);\n\n    for (const artifact of artifacts) {\n      if (cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(artifact)) {\n        this.addStack(artifact);\n      } else if (cxapi.AssetManifestArtifact.isAssetManifestArtifact(artifact)) {\n        const manifest = AssetManifest.fromFile(artifact.file);\n\n        for (const entry of manifest.entries) {\n          const parentStack = parentStacks.get(artifact);\n          if (parentStack === undefined) {\n            throw new ToolkitError('Found an asset manifest that is not associated with a stack');\n          }\n          this.addAsset(parentStack, artifact, manifest, entry);\n        }\n      } else if (cxapi.NestedCloudAssemblyArtifact.isNestedCloudAssemblyArtifact(artifact)) {\n        const assembly = new cxapi.CloudAssembly(artifact.fullPath, { topoSort: false });\n        const nestedGraph = new WorkGraphBuilder(\n          { ioHost: this.ioHost, action: this.action },\n          this.prebuildAssets,\n          `${this.idPrefix}${artifact.id}.`,\n        ).build(assembly.artifacts);\n        this.graph.absorb(nestedGraph);\n      } else {\n        // Ignore whatever else\n      }\n    }\n\n    this.graph.removeUnavailableDependencies();\n\n    // Remove any potentially introduced cycles between asset publishing and the stacks that depend on them.\n    this.removeStackPublishCycles();\n\n    return this.graph;\n  }\n\n  private stackArtifactIds(deps: cxapi.CloudArtifact[]): string[] {\n    return deps.flatMap((d) => cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(d) ? [this.stackArtifactId(d)] : []);\n  }\n\n  private stackArtifactId(artifact: cxapi.CloudArtifact): string {\n    if (!cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(artifact)) {\n      throw new ToolkitError(`Can only call this on CloudFormationStackArtifact, got: ${artifact.constructor.name}`);\n    }\n    return `${this.idPrefix}${artifact.id}`;\n  }\n\n  /**\n   * We may have accidentally introduced cycles in an attempt to make the messages printed to the\n   * console not interfere with each other too much. Remove them again.\n   */\n  private removeStackPublishCycles() {\n    const publishSteps = this.graph.nodesOfType('asset-publish');\n    for (const publishStep of publishSteps) {\n      for (const dep of publishStep.dependencies) {\n        if (this.graph.reachable(dep, publishStep.id)) {\n          publishStep.dependencies.delete(dep);\n        }\n      }\n    }\n  }\n}\n\nfunction stacksFromAssets(artifacts: cxapi.CloudArtifact[]) {\n  const ret = new Map<cxapi.AssetManifestArtifact, cxapi.CloudFormationStackArtifact>();\n  for (const stack of artifacts.filter(x => cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(x))) {\n    const assetArtifacts = stack.dependencies.filter((x) => cxapi.AssetManifestArtifact.isAssetManifestArtifact(x));\n    for (const art of assetArtifacts) {\n      ret.set(art, stack);\n    }\n  }\n\n  return ret;\n}\n\nfunction onlyStacks(artifacts: cxapi.CloudArtifact[]) {\n  return artifacts.filter(x => cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(x));\n}\n"]}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { CloudFormationStackArtifact } from '@aws-cdk/cx-api';
|
|
2
|
+
import type { StackActivity } from '../../api/stack-events';
|
|
3
|
+
import { StackProgress } from '../../api/stack-events/stack-progress-monitor';
|
|
4
|
+
import { IoMessage } from '../../toolkit/cli-io-host';
|
|
5
|
+
export interface IActivityPrinter {
|
|
6
|
+
notify<T>(msg: IoMessage<T>): void;
|
|
7
|
+
}
|
|
8
|
+
export interface ActivityPrinterProps {
|
|
9
|
+
/**
|
|
10
|
+
* Stream to write to
|
|
11
|
+
*/
|
|
12
|
+
readonly stream: NodeJS.WriteStream;
|
|
13
|
+
}
|
|
14
|
+
export declare abstract class ActivityPrinterBase implements IActivityPrinter {
|
|
15
|
+
protected readonly props: ActivityPrinterProps;
|
|
16
|
+
protected static readonly TIMESTAMP_WIDTH = 12;
|
|
17
|
+
protected static readonly STATUS_WIDTH = 20;
|
|
18
|
+
/**
|
|
19
|
+
* Stream to write to
|
|
20
|
+
*/
|
|
21
|
+
protected readonly stream: NodeJS.WriteStream;
|
|
22
|
+
/**
|
|
23
|
+
* The with of the "resource type" column.
|
|
24
|
+
*/
|
|
25
|
+
protected resourceTypeColumnWidth: number;
|
|
26
|
+
/**
|
|
27
|
+
* A list of resource IDs which are currently being processed
|
|
28
|
+
*/
|
|
29
|
+
protected resourcesInProgress: Record<string, StackActivity>;
|
|
30
|
+
protected stackProgress?: StackProgress;
|
|
31
|
+
protected rollingBack: boolean;
|
|
32
|
+
protected readonly failures: StackActivity[];
|
|
33
|
+
protected hookFailureMap: Map<string, Map<string, string>>;
|
|
34
|
+
constructor(props: ActivityPrinterProps);
|
|
35
|
+
protected abstract print(): void;
|
|
36
|
+
/**
|
|
37
|
+
* Receive a stack activity message
|
|
38
|
+
*/
|
|
39
|
+
notify(msg: IoMessage<any>): void;
|
|
40
|
+
start({ stack }: {
|
|
41
|
+
stack: CloudFormationStackArtifact;
|
|
42
|
+
}): void;
|
|
43
|
+
activity(activity: StackActivity): void;
|
|
44
|
+
stop(): void;
|
|
45
|
+
protected addActivity(activity: StackActivity): void;
|
|
46
|
+
protected failureReason(activity: StackActivity): string;
|
|
47
|
+
/**
|
|
48
|
+
* Is the activity a meta activity for the stack itself.
|
|
49
|
+
*/
|
|
50
|
+
protected isActivityForTheStack(activity: StackActivity): boolean;
|
|
51
|
+
}
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ActivityPrinterBase = void 0;
|
|
4
|
+
const util_1 = require("../../util");
|
|
5
|
+
class ActivityPrinterBase {
|
|
6
|
+
constructor(props) {
|
|
7
|
+
this.props = props;
|
|
8
|
+
/**
|
|
9
|
+
* The with of the "resource type" column.
|
|
10
|
+
*/
|
|
11
|
+
this.resourceTypeColumnWidth = (0, util_1.maxResourceTypeLength)({});
|
|
12
|
+
/**
|
|
13
|
+
* A list of resource IDs which are currently being processed
|
|
14
|
+
*/
|
|
15
|
+
this.resourcesInProgress = {};
|
|
16
|
+
this.rollingBack = false;
|
|
17
|
+
this.failures = new Array();
|
|
18
|
+
this.hookFailureMap = new Map();
|
|
19
|
+
this.stream = props.stream;
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Receive a stack activity message
|
|
23
|
+
*/
|
|
24
|
+
notify(msg) {
|
|
25
|
+
switch (msg.code) {
|
|
26
|
+
case 'CDK_TOOLKIT_I5501':
|
|
27
|
+
this.start(msg.data);
|
|
28
|
+
break;
|
|
29
|
+
case 'CDK_TOOLKIT_I5502':
|
|
30
|
+
this.activity(msg.data);
|
|
31
|
+
break;
|
|
32
|
+
case 'CDK_TOOLKIT_I5503':
|
|
33
|
+
this.stop();
|
|
34
|
+
break;
|
|
35
|
+
default:
|
|
36
|
+
// ignore all other messages
|
|
37
|
+
break;
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
start({ stack }) {
|
|
41
|
+
this.resourceTypeColumnWidth = (0, util_1.maxResourceTypeLength)(stack.template);
|
|
42
|
+
}
|
|
43
|
+
activity(activity) {
|
|
44
|
+
// process the activity and then call print
|
|
45
|
+
this.addActivity(activity);
|
|
46
|
+
this.print();
|
|
47
|
+
}
|
|
48
|
+
stop() {
|
|
49
|
+
// final print after the stack is done
|
|
50
|
+
this.print();
|
|
51
|
+
}
|
|
52
|
+
addActivity(activity) {
|
|
53
|
+
var _a, _b, _c, _d, _e;
|
|
54
|
+
const status = activity.event.ResourceStatus;
|
|
55
|
+
const hookStatus = activity.event.HookStatus;
|
|
56
|
+
const hookType = activity.event.HookType;
|
|
57
|
+
if (!status || !activity.event.LogicalResourceId) {
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
this.stackProgress = activity.progress;
|
|
61
|
+
if (status === 'ROLLBACK_IN_PROGRESS' || status === 'UPDATE_ROLLBACK_IN_PROGRESS') {
|
|
62
|
+
// Only triggered on the stack once we've started doing a rollback
|
|
63
|
+
this.rollingBack = true;
|
|
64
|
+
}
|
|
65
|
+
if (status.endsWith('_IN_PROGRESS')) {
|
|
66
|
+
this.resourcesInProgress[activity.event.LogicalResourceId] = activity;
|
|
67
|
+
}
|
|
68
|
+
if ((0, util_1.stackEventHasErrorMessage)(status)) {
|
|
69
|
+
const isCancelled = ((_a = activity.event.ResourceStatusReason) !== null && _a !== void 0 ? _a : '').indexOf('cancelled') > -1;
|
|
70
|
+
// Cancelled is not an interesting failure reason
|
|
71
|
+
if (!isCancelled) {
|
|
72
|
+
this.failures.push(activity);
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
if (status.endsWith('_COMPLETE') || status.endsWith('_FAILED')) {
|
|
76
|
+
delete this.resourcesInProgress[activity.event.LogicalResourceId];
|
|
77
|
+
}
|
|
78
|
+
if (hookStatus !== undefined &&
|
|
79
|
+
hookStatus.endsWith('_COMPLETE_FAILED') &&
|
|
80
|
+
activity.event.LogicalResourceId !== undefined &&
|
|
81
|
+
hookType !== undefined) {
|
|
82
|
+
if (this.hookFailureMap.has(activity.event.LogicalResourceId)) {
|
|
83
|
+
(_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 : '');
|
|
84
|
+
}
|
|
85
|
+
else {
|
|
86
|
+
this.hookFailureMap.set(activity.event.LogicalResourceId, new Map());
|
|
87
|
+
(_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 : '');
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
failureReason(activity) {
|
|
92
|
+
var _a, _b;
|
|
93
|
+
const resourceStatusReason = (_a = activity.event.ResourceStatusReason) !== null && _a !== void 0 ? _a : '';
|
|
94
|
+
const logicalResourceId = (_b = activity.event.LogicalResourceId) !== null && _b !== void 0 ? _b : '';
|
|
95
|
+
const hookFailureReasonMap = this.hookFailureMap.get(logicalResourceId);
|
|
96
|
+
if (hookFailureReasonMap !== undefined) {
|
|
97
|
+
for (const hookType of hookFailureReasonMap.keys()) {
|
|
98
|
+
if (resourceStatusReason.includes(hookType)) {
|
|
99
|
+
return resourceStatusReason + ' : ' + hookFailureReasonMap.get(hookType);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
return resourceStatusReason;
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* Is the activity a meta activity for the stack itself.
|
|
107
|
+
*/
|
|
108
|
+
isActivityForTheStack(activity) {
|
|
109
|
+
return activity.event.PhysicalResourceId === activity.event.StackId;
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
exports.ActivityPrinterBase = ActivityPrinterBase;
|
|
113
|
+
ActivityPrinterBase.TIMESTAMP_WIDTH = 12;
|
|
114
|
+
ActivityPrinterBase.STATUS_WIDTH = 20;
|
|
115
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base.js","sourceRoot":"","sources":["base.ts"],"names":[],"mappings":"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","sourcesContent":["import { CloudFormationStackArtifact } from '@aws-cdk/cx-api';\nimport type { StackActivity } from '../../api/stack-events';\nimport { StackProgress } from '../../api/stack-events/stack-progress-monitor';\nimport { IoMessage } from '../../toolkit/cli-io-host';\nimport { maxResourceTypeLength, stackEventHasErrorMessage } from '../../util';\n\nexport interface IActivityPrinter {\n  notify<T>(msg: IoMessage<T>): void;\n}\n\nexport interface ActivityPrinterProps {\n  /**\n   * Stream to write to\n   */\n  readonly stream: NodeJS.WriteStream;\n}\n\nexport abstract class ActivityPrinterBase implements IActivityPrinter {\n  protected static readonly TIMESTAMP_WIDTH = 12;\n  protected static readonly STATUS_WIDTH = 20;\n\n  /**\n   * Stream to write to\n   */\n  protected readonly stream: NodeJS.WriteStream;\n\n  /**\n   * The with of the \"resource type\" column.\n   */\n  protected resourceTypeColumnWidth: number = maxResourceTypeLength({});\n\n  /**\n   * A list of resource IDs which are currently being processed\n   */\n  protected resourcesInProgress: Record<string, StackActivity> = {};\n\n  protected stackProgress?: StackProgress;\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: ActivityPrinterProps) {\n    this.stream = props.stream;\n  }\n\n  protected abstract print(): void;\n\n  /**\n   * Receive a stack activity message\n   */\n  public notify(msg: IoMessage<any>): void {\n    switch (msg.code) {\n      case 'CDK_TOOLKIT_I5501':\n        this.start(msg.data);\n        break;\n      case 'CDK_TOOLKIT_I5502':\n        this.activity(msg.data as StackActivity);\n        break;\n      case 'CDK_TOOLKIT_I5503':\n        this.stop();\n        break;\n      default:\n        // ignore all other messages\n        break;\n    }\n  }\n\n  public start({ stack }: { stack: CloudFormationStackArtifact}) {\n    this.resourceTypeColumnWidth = maxResourceTypeLength(stack.template);\n  }\n\n  public activity(activity: StackActivity) {\n    // process the activity and then call print\n    this.addActivity(activity);\n    this.print();\n  }\n\n  public stop() {\n    // final print after the stack is done\n    this.print();\n  }\n\n  protected 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    this.stackProgress = activity.progress;\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 (stackEventHasErrorMessage(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 (\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  protected 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  /**\n   * Is the activity a meta activity for the stack itself.\n   */\n  protected isActivityForTheStack(activity: StackActivity) {\n    return activity.event.PhysicalResourceId === activity.event.StackId;\n  }\n}\n"]}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { ActivityPrinterBase, ActivityPrinterProps } from './base';
|
|
2
|
+
/**
|
|
3
|
+
* Activity Printer which shows the resources currently being updated
|
|
4
|
+
*
|
|
5
|
+
* It will continuously re-update the terminal and show only the resources
|
|
6
|
+
* that are currently being updated, in addition to a progress bar which
|
|
7
|
+
* shows how far along the deployment is.
|
|
8
|
+
*
|
|
9
|
+
* Resources that have failed will always be shown, and will be recapitulated
|
|
10
|
+
* along with their stack trace when the monitoring ends.
|
|
11
|
+
*
|
|
12
|
+
* Resources that failed deployment because they have been cancelled are
|
|
13
|
+
* not included.
|
|
14
|
+
*/
|
|
15
|
+
export declare class CurrentActivityPrinter extends ActivityPrinterBase {
|
|
16
|
+
/**
|
|
17
|
+
* Continuously write to the same output block.
|
|
18
|
+
*/
|
|
19
|
+
private block;
|
|
20
|
+
constructor(props: ActivityPrinterProps);
|
|
21
|
+
protected print(): void;
|
|
22
|
+
stop(): void;
|
|
23
|
+
private progressBar;
|
|
24
|
+
private failureReasonOnNextLine;
|
|
25
|
+
}
|