aws-cdk 2.1000.2 → 2.1001.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 (126) hide show
  1. package/README.md +1 -1
  2. package/THIRD_PARTY_LICENSES +95 -2981
  3. package/build-info.json +2 -2
  4. package/db.json.gz +0 -0
  5. package/lib/api/aws-auth/account-cache.d.ts +0 -3
  6. package/lib/api/aws-auth/account-cache.js +1 -4
  7. package/lib/api/aws-auth/awscli-compatible.js +1 -1
  8. package/lib/api/aws-auth/credential-plugins.js +4 -4
  9. package/lib/api/aws-auth/sdk-logger.js +2 -2
  10. package/lib/api/aws-auth/sdk-provider.js +3 -3
  11. package/lib/api/aws-auth/sdk.d.ts +6 -0
  12. package/lib/api/aws-auth/sdk.js +11 -3
  13. package/lib/api/aws-auth/tracing.js +1 -1
  14. package/lib/api/bootstrap/bootstrap-environment.js +3 -3
  15. package/lib/api/bootstrap/deploy-bootstrap.js +1 -1
  16. package/lib/api/cxapp/cloud-assembly.js +3 -2
  17. package/lib/api/cxapp/cloud-executable.js +1 -1
  18. package/lib/api/cxapp/environments.js +1 -1
  19. package/lib/api/cxapp/exec.js +1 -1
  20. package/lib/api/deployments/asset-publishing.js +1 -1
  21. package/lib/api/deployments/assets.js +1 -1
  22. package/lib/api/deployments/cloudformation.d.ts +2 -2
  23. package/lib/api/deployments/cloudformation.js +7 -7
  24. package/lib/api/deployments/deploy-stack.d.ts +1 -1
  25. package/lib/api/deployments/deploy-stack.js +8 -8
  26. package/lib/api/deployments/deployments.d.ts +3 -3
  27. package/lib/api/deployments/deployments.js +12 -14
  28. package/lib/api/deployments/hotswap-deployments.js +3 -3
  29. package/lib/api/deployments/nested-stack-helpers.js +3 -3
  30. package/lib/api/environment/environment-access.js +3 -3
  31. package/lib/api/environment/environment-resources.js +3 -3
  32. package/lib/api/evaluate-cloudformation-template.js +2 -2
  33. package/lib/api/garbage-collection/garbage-collector.js +1 -1
  34. package/lib/api/hotswap/appsync-mapping-templates.js +2 -2
  35. package/lib/api/hotswap/common.d.ts +1 -1
  36. package/lib/api/hotswap/common.js +2 -2
  37. package/lib/api/hotswap/lambda-functions.js +5 -3
  38. package/lib/api/logs/find-cloudwatch-logs.js +3 -3
  39. package/lib/api/logs/index.d.ts +2 -0
  40. package/lib/api/logs/index.js +19 -0
  41. package/lib/api/resource-import/importer.d.ts +2 -2
  42. package/lib/api/resource-import/importer.js +1 -1
  43. package/lib/api/resource-import/migrator.js +2 -2
  44. package/lib/api/settings.js +1 -1
  45. package/lib/api/{util → stack-events}/display.js +1 -1
  46. package/lib/api/stack-events/index.d.ts +3 -0
  47. package/lib/api/stack-events/index.js +20 -0
  48. package/lib/api/{util/cloudformation → stack-events}/stack-activity-monitor.d.ts +2 -2
  49. package/lib/api/{util/cloudformation → stack-events}/stack-activity-monitor.js +4 -4
  50. package/lib/api/{util/cloudformation → stack-events}/stack-event-poller.d.ts +1 -1
  51. package/lib/api/{util/cloudformation → stack-events}/stack-event-poller.js +3 -3
  52. package/lib/api/{util/cloudformation → stack-events}/stack-status.js +1 -1
  53. package/lib/api/toolkit-info.js +3 -2
  54. package/lib/api/util/rwlock.js +1 -1
  55. package/lib/api/util/template-body-parameter.js +2 -2
  56. package/lib/api/work-graph/index.d.ts +3 -0
  57. package/lib/api/work-graph/index.js +20 -0
  58. package/lib/{util → api/work-graph}/work-graph-builder.d.ts +4 -1
  59. package/lib/api/work-graph/work-graph-builder.js +169 -0
  60. package/lib/api/work-graph/work-graph-types.js +13 -0
  61. package/lib/{util → api/work-graph}/work-graph.d.ts +9 -2
  62. package/lib/api/work-graph/work-graph.js +349 -0
  63. package/lib/cli/cdk-toolkit.d.ts +3 -3
  64. package/lib/cli/cdk-toolkit.js +19 -17
  65. package/lib/cli/cli-config.js +5 -5
  66. package/lib/cli/cli.js +2 -2
  67. package/lib/cli/parse-command-line-arguments.js +2 -2
  68. package/lib/cli/platform-warnings.js +1 -1
  69. package/lib/cli/user-input.d.ts +3 -10
  70. package/lib/cli/user-input.js +1 -1
  71. package/lib/commands/context.js +3 -3
  72. package/lib/commands/docs.js +1 -1
  73. package/lib/commands/doctor.js +1 -1
  74. package/lib/context-providers/ami.js +1 -1
  75. package/lib/context-providers/availability-zones.js +1 -1
  76. package/lib/context-providers/cc-api-provider.d.ts +34 -0
  77. package/lib/context-providers/cc-api-provider.js +116 -0
  78. package/lib/context-providers/endpoint-service-availability-zones.js +1 -1
  79. package/lib/context-providers/hosted-zones.js +1 -1
  80. package/lib/context-providers/index.js +5 -3
  81. package/lib/context-providers/keys.js +1 -1
  82. package/lib/context-providers/load-balancers.js +1 -1
  83. package/lib/context-providers/security-groups.js +1 -1
  84. package/lib/context-providers/ssm-parameters.js +1 -1
  85. package/lib/context-providers/vpcs.js +1 -1
  86. package/lib/diff.js +1 -1
  87. package/lib/index.js +83175 -90981
  88. package/lib/init-hooks.js +3 -4
  89. package/lib/init-templates/.init-version.json +1 -1
  90. package/lib/init-templates/.recommended-feature-flags.json +2 -1
  91. package/lib/init.js +3 -3
  92. package/lib/legacy-exports-source.d.ts +1 -1
  93. package/lib/legacy-exports-source.js +2 -2
  94. package/lib/notices.d.ts +48 -5
  95. package/lib/notices.js +131 -86
  96. package/lib/tree.d.ts +3 -3
  97. package/lib/tree.js +4 -4
  98. package/lib/util/archive.js +3 -3
  99. package/lib/util/cloudformation.js +10 -0
  100. package/lib/util/directories.js +3 -2
  101. package/lib/util/format-error.js +22 -0
  102. package/lib/util/index.d.ts +11 -1
  103. package/lib/util/index.js +12 -2
  104. package/lib/util/json.d.ts +48 -0
  105. package/lib/util/json.js +68 -0
  106. package/lib/util/objects.js +1 -1
  107. package/lib/util/parallel.js +7 -3
  108. package/lib/{serialize.js → util/serialize.js} +3 -3
  109. package/lib/util/version-range.js +1 -1
  110. package/lib/util/yaml-cfn.js +4 -2
  111. package/package.json +18 -16
  112. package/lib/util/error.js +0 -22
  113. package/lib/util/validate-notification-arn.js +0 -10
  114. package/lib/util/work-graph-builder.js +0 -167
  115. package/lib/util/work-graph-types.js +0 -14
  116. package/lib/util/work-graph.js +0 -344
  117. /package/lib/api/{util → stack-events}/display.d.ts +0 -0
  118. /package/lib/api/{util/cloudformation → stack-events}/stack-status.d.ts +0 -0
  119. /package/lib/{util → api/work-graph}/work-graph-types.d.ts +0 -0
  120. /package/lib/{util → cli}/tables.d.ts +0 -0
  121. /package/lib/{util → cli}/tables.js +0 -0
  122. /package/lib/util/{validate-notification-arn.d.ts → cloudformation.d.ts} +0 -0
  123. /package/lib/util/{error.d.ts → format-error.d.ts} +0 -0
  124. /package/lib/{serialize.d.ts → util/serialize.d.ts} +0 -0
  125. /package/lib/{api/util → util}/string-manipulation.d.ts +0 -0
  126. /package/lib/{api/util → util}/string-manipulation.js +0 -0
@@ -1,167 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.WorkGraphBuilder = void 0;
4
- const cxapi = require("@aws-cdk/cx-api");
5
- const cdk_assets_1 = require("cdk-assets");
6
- const content_hash_1 = require("./content-hash");
7
- const work_graph_1 = require("./work-graph");
8
- const work_graph_types_1 = require("./work-graph-types");
9
- const error_1 = require("../toolkit/error");
10
- class WorkGraphBuilder {
11
- constructor(prebuildAssets, idPrefix = '') {
12
- this.prebuildAssets = prebuildAssets;
13
- this.idPrefix = idPrefix;
14
- this.graph = new work_graph_1.WorkGraph();
15
- }
16
- addStack(artifact) {
17
- this.graph.addNodes({
18
- type: 'stack',
19
- id: `${this.idPrefix}${artifact.id}`,
20
- dependencies: new Set(this.stackArtifactIds(onlyStacks(artifact.dependencies))),
21
- stack: artifact,
22
- deploymentState: work_graph_types_1.DeploymentState.PENDING,
23
- priority: WorkGraphBuilder.PRIORITIES.stack,
24
- });
25
- }
26
- /**
27
- * Oof, see this parameter list
28
- */
29
- // eslint-disable-next-line max-len
30
- addAsset(parentStack, assetManifestArtifact, assetManifest, asset) {
31
- // Just the artifact identifier
32
- const assetId = asset.id.assetId;
33
- const buildId = `build-${assetId}-${(0, content_hash_1.contentHashAny)([assetId, asset.genericSource]).substring(0, 10)}`;
34
- const publishId = `publish-${assetId}-${(0, content_hash_1.contentHashAny)([assetId, asset.genericDestination]).substring(0, 10)}`;
35
- // Build node only gets added once because they are all the same
36
- if (!this.graph.tryGetNode(buildId)) {
37
- const node = {
38
- type: 'asset-build',
39
- id: buildId,
40
- note: assetId,
41
- dependencies: new Set([
42
- ...this.stackArtifactIds(assetManifestArtifact.dependencies),
43
- // If we disable prebuild, then assets inherit (stack) dependencies from their parent stack
44
- ...!this.prebuildAssets ? this.stackArtifactIds(onlyStacks(parentStack.dependencies)) : [],
45
- ]),
46
- parentStack: parentStack,
47
- assetManifestArtifact,
48
- assetManifest,
49
- asset,
50
- deploymentState: work_graph_types_1.DeploymentState.PENDING,
51
- priority: WorkGraphBuilder.PRIORITIES['asset-build'],
52
- };
53
- this.graph.addNodes(node);
54
- }
55
- const publishNode = this.graph.tryGetNode(publishId);
56
- if (!publishNode) {
57
- this.graph.addNodes({
58
- type: 'asset-publish',
59
- id: publishId,
60
- note: `${asset.id}`,
61
- dependencies: new Set([
62
- buildId,
63
- ]),
64
- parentStack,
65
- assetManifestArtifact,
66
- assetManifest,
67
- asset,
68
- deploymentState: work_graph_types_1.DeploymentState.PENDING,
69
- priority: WorkGraphBuilder.PRIORITIES['asset-publish'],
70
- });
71
- }
72
- for (const inheritedDep of this.stackArtifactIds(onlyStacks(parentStack.dependencies))) {
73
- // The asset publish step also depends on the stacks that the parent depends on.
74
- // This is purely cosmetic: if we don't do this, the progress printing of asset publishing
75
- // is going to interfere with the progress bar of the stack deployment. We could remove this
76
- // for overall faster deployments if we ever have a better method of progress displaying.
77
- // Note: this may introduce a cycle if one of the parent's dependencies is another stack that
78
- // depends on this asset. To workaround this we remove these cycles once all nodes have
79
- // been added to the graph.
80
- this.graph.addDependency(publishId, inheritedDep);
81
- }
82
- // This will work whether the stack node has been added yet or not
83
- this.graph.addDependency(`${this.idPrefix}${parentStack.id}`, publishId);
84
- }
85
- build(artifacts) {
86
- const parentStacks = stacksFromAssets(artifacts);
87
- for (const artifact of artifacts) {
88
- if (cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(artifact)) {
89
- this.addStack(artifact);
90
- }
91
- else if (cxapi.AssetManifestArtifact.isAssetManifestArtifact(artifact)) {
92
- const manifest = cdk_assets_1.AssetManifest.fromFile(artifact.file);
93
- for (const entry of manifest.entries) {
94
- const parentStack = parentStacks.get(artifact);
95
- if (parentStack === undefined) {
96
- throw new error_1.ToolkitError('Found an asset manifest that is not associated with a stack');
97
- }
98
- this.addAsset(parentStack, artifact, manifest, entry);
99
- }
100
- }
101
- else if (cxapi.NestedCloudAssemblyArtifact.isNestedCloudAssemblyArtifact(artifact)) {
102
- const assembly = new cxapi.CloudAssembly(artifact.fullPath, { topoSort: false });
103
- const nestedGraph = new WorkGraphBuilder(this.prebuildAssets, `${this.idPrefix}${artifact.id}.`).build(assembly.artifacts);
104
- this.graph.absorb(nestedGraph);
105
- }
106
- else {
107
- // Ignore whatever else
108
- }
109
- }
110
- this.graph.removeUnavailableDependencies();
111
- // Remove any potentially introduced cycles between asset publishing and the stacks that depend on them.
112
- this.removeStackPublishCycles();
113
- return this.graph;
114
- }
115
- stackArtifactIds(deps) {
116
- return deps.flatMap((d) => cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(d) ? [this.stackArtifactId(d)] : []);
117
- }
118
- stackArtifactId(artifact) {
119
- if (!cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(artifact)) {
120
- throw new error_1.ToolkitError(`Can only call this on CloudFormationStackArtifact, got: ${artifact.constructor.name}`);
121
- }
122
- return `${this.idPrefix}${artifact.id}`;
123
- }
124
- /**
125
- * We may have accidentally introduced cycles in an attempt to make the messages printed to the
126
- * console not interfere with each other too much. Remove them again.
127
- */
128
- removeStackPublishCycles() {
129
- const publishSteps = this.graph.nodesOfType('asset-publish');
130
- for (const publishStep of publishSteps) {
131
- for (const dep of publishStep.dependencies) {
132
- if (this.graph.reachable(dep, publishStep.id)) {
133
- publishStep.dependencies.delete(dep);
134
- }
135
- }
136
- }
137
- }
138
- }
139
- exports.WorkGraphBuilder = WorkGraphBuilder;
140
- /**
141
- * Default priorities for nodes
142
- *
143
- * Assets builds have higher priority than the other two operations, to make good on our promise that
144
- * '--prebuild-assets' will actually do assets before stacks (if it can). Unfortunately it is the
145
- * default :(
146
- *
147
- * But between stack dependencies and publish dependencies, stack dependencies go first
148
- */
149
- WorkGraphBuilder.PRIORITIES = {
150
- 'asset-build': 10,
151
- 'asset-publish': 0,
152
- 'stack': 5,
153
- };
154
- function stacksFromAssets(artifacts) {
155
- const ret = new Map();
156
- for (const stack of artifacts.filter(cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact)) {
157
- const assetArtifacts = stack.dependencies.filter(cxapi.AssetManifestArtifact.isAssetManifestArtifact);
158
- for (const art of assetArtifacts) {
159
- ret.set(art, stack);
160
- }
161
- }
162
- return ret;
163
- }
164
- function onlyStacks(artifacts) {
165
- return artifacts.filter(cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact);
166
- }
167
- //# 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 { contentHashAny } from './content-hash';\nimport { WorkGraph } from './work-graph';\nimport { DeploymentState, AssetBuildNode, WorkNode } from './work-graph-types';\nimport { ToolkitError } from '../toolkit/error';\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 = new WorkGraph();\n\n  constructor(private readonly prebuildAssets: boolean, private readonly idPrefix = '') { }\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(this.prebuildAssets, `${this.idPrefix}${artifact.id}.`).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"]}
@@ -1,14 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.DeploymentState = void 0;
4
- var DeploymentState;
5
- (function (DeploymentState) {
6
- DeploymentState["PENDING"] = "pending";
7
- DeploymentState["QUEUED"] = "queued";
8
- DeploymentState["DEPLOYING"] = "deploying";
9
- DeploymentState["COMPLETED"] = "completed";
10
- DeploymentState["FAILED"] = "failed";
11
- DeploymentState["SKIPPED"] = "skipped";
12
- })(DeploymentState || (exports.DeploymentState = DeploymentState = {}));
13
- ;
14
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid29yay1ncmFwaC10eXBlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIndvcmstZ3JhcGgtdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBR0EsSUFBWSxlQU9YO0FBUEQsV0FBWSxlQUFlO0lBQ3pCLHNDQUFtQixDQUFBO0lBQ25CLG9DQUFpQixDQUFBO0lBQ2pCLDBDQUF1QixDQUFBO0lBQ3ZCLDBDQUF1QixDQUFBO0lBQ3ZCLG9DQUFpQixDQUFBO0lBQ2pCLHNDQUFtQixDQUFBO0FBQ3JCLENBQUMsRUFQVyxlQUFlLCtCQUFmLGVBQWUsUUFPMUI7QUFBQSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY3hhcGkgZnJvbSAnQGF3cy1jZGsvY3gtYXBpJztcbmltcG9ydCB7IEFzc2V0TWFuaWZlc3QsIHR5cGUgSU1hbmlmZXN0RW50cnkgfSBmcm9tICdjZGstYXNzZXRzJztcblxuZXhwb3J0IGVudW0gRGVwbG95bWVudFN0YXRlIHtcbiAgUEVORElORyA9ICdwZW5kaW5nJyxcbiAgUVVFVUVEID0gJ3F1ZXVlZCcsXG4gIERFUExPWUlORyA9ICdkZXBsb3lpbmcnLFxuICBDT01QTEVURUQgPSAnY29tcGxldGVkJyxcbiAgRkFJTEVEID0gJ2ZhaWxlZCcsXG4gIFNLSVBQRUQgPSAnc2tpcHBlZCcsXG59O1xuXG5leHBvcnQgdHlwZSBXb3JrTm9kZSA9IFN0YWNrTm9kZSB8IEFzc2V0QnVpbGROb2RlIHwgQXNzZXRQdWJsaXNoTm9kZTtcblxuZXhwb3J0IGludGVyZmFjZSBXb3JrTm9kZUNvbW1vbiB7XG4gIHJlYWRvbmx5IGlkOiBzdHJpbmc7XG4gIHJlYWRvbmx5IGRlcGVuZGVuY2llczogU2V0PHN0cmluZz47XG4gIGRlcGxveW1lbnRTdGF0ZTogRGVwbG95bWVudFN0YXRlO1xuICAvKiogU29tZSByZWFkYWJsZSBpbmZvcm1hdGlvbiB0byBhdHRhY2ggdG8gdGhlIGlkLCB3aGljaCBtYXkgYmUgdW5yZWFkYWJsZSAqL1xuICByZWFkb25seSBub3RlPzogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFN0YWNrTm9kZSBleHRlbmRzIFdvcmtOb2RlQ29tbW9uIHtcbiAgcmVhZG9ubHkgdHlwZTogJ3N0YWNrJztcbiAgcmVhZG9ubHkgc3RhY2s6IGN4YXBpLkNsb3VkRm9ybWF0aW9uU3RhY2tBcnRpZmFjdDtcbiAgLyoqIFNvcnQgYnkgcHJpb3JpdHkgd2hlbiBwaWNraW5nIHVwIHdvcmssIGhpZ2hlciBpcyBlYXJsaWVyICovXG4gIHJlYWRvbmx5IHByaW9yaXR5PzogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEFzc2V0QnVpbGROb2RlIGV4dGVuZHMgV29ya05vZGVDb21tb24ge1xuICByZWFkb25seSB0eXBlOiAnYXNzZXQtYnVpbGQnO1xuICAvKiogVGhlIGFzc2V0IG1hbmlmZXN0IHRoaXMgYXNzZXQgcmVzaWRlcyBpbiAoYXJ0aWZhY3QpICovXG4gIHJlYWRvbmx5IGFzc2V0TWFuaWZlc3RBcnRpZmFjdDogY3hhcGkuQXNzZXRNYW5pZmVzdEFydGlmYWN0O1xuICAvKiogVGhlIGFzc2V0IG1hbmlmZXN0IHRoaXMgYXNzZXQgcmVzaWRlcyBpbiAqL1xuICByZWFkb25seSBhc3NldE1hbmlmZXN0OiBBc3NldE1hbmlmZXN0O1xuICAvKiogVGhlIHN0YWNrIHRoaXMgYXNzZXQgd2FzIGRlZmluZWQgaW4gKHVzZWQgZm9yIGVudmlyb25tZW50IHNldHRpbmdzKSAqL1xuICByZWFkb25seSBwYXJlbnRTdGFjazogY3hhcGkuQ2xvdWRGb3JtYXRpb25TdGFja0FydGlmYWN0O1xuICAvKiogVGhlIGFzc2V0IHRoYXQgbmVlZHMgdG8gYmUgYnVpbHQgKi9cbiAgcmVhZG9ubHkgYXNzZXQ6IElNYW5pZmVzdEVudHJ5O1xuICAvKiogU29ydCBieSBwcmlvcml0eSB3aGVuIHBpY2tpbmcgdXAgd29yaywgaGlnaGVyIGlzIGVhcmxpZXIgKi9cbiAgcmVhZG9ubHkgcHJpb3JpdHk/OiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQXNzZXRQdWJsaXNoTm9kZSBleHRlbmRzIFdvcmtOb2RlQ29tbW9uIHtcbiAgcmVhZG9ubHkgdHlwZTogJ2Fzc2V0LXB1Ymxpc2gnO1xuICAvKiogVGhlIGFzc2V0IG1hbmlmZXN0IHRoaXMgYXNzZXQgcmVzaWRlcyBpbiAoYXJ0aWZhY3QpICovXG4gIHJlYWRvbmx5IGFzc2V0TWFuaWZlc3RBcnRpZmFjdDogY3hhcGkuQXNzZXRNYW5pZmVzdEFydGlmYWN0O1xuICAvKiogVGhlIGFzc2V0IG1hbmlmZXN0IHRoaXMgYXNzZXQgcmVzaWRlcyBpbiAqL1xuICByZWFkb25seSBhc3NldE1hbmlmZXN0OiBBc3NldE1hbmlmZXN0O1xuICAvKiogVGhlIHN0YWNrIHRoaXMgYXNzZXQgd2FzIGRlZmluZWQgaW4gKHVzZWQgZm9yIGVudmlyb25tZW50IHNldHRpbmdzKSAqL1xuICByZWFkb25seSBwYXJlbnRTdGFjazogY3hhcGkuQ2xvdWRGb3JtYXRpb25TdGFja0FydGlmYWN0O1xuICAvKiogVGhlIGFzc2V0IHRoYXQgbmVlZHMgdG8gYmUgcHVibGlzaGVkICovXG4gIHJlYWRvbmx5IGFzc2V0OiBJTWFuaWZlc3RFbnRyeTtcbiAgLyoqIFNvcnQgYnkgcHJpb3JpdHkgd2hlbiBwaWNraW5nIHVwIHdvcmssIGhpZ2hlciBpcyBlYXJsaWVyICovXG4gIHJlYWRvbmx5IHByaW9yaXR5PzogbnVtYmVyO1xufVxuIl19
@@ -1,344 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.WorkGraph = void 0;
4
- const parallel_1 = require("./parallel");
5
- const work_graph_types_1 = require("./work-graph-types");
6
- const logging_1 = require("../logging");
7
- const error_1 = require("../toolkit/error");
8
- class WorkGraph {
9
- constructor(nodes = {}) {
10
- this.readyPool = [];
11
- this.lazyDependencies = new Map();
12
- this.nodes = { ...nodes };
13
- }
14
- addNodes(...nodes) {
15
- for (const node of nodes) {
16
- if (this.nodes[node.id]) {
17
- throw new error_1.ToolkitError(`Duplicate use of node id: ${node.id}`);
18
- }
19
- const ld = this.lazyDependencies.get(node.id);
20
- if (ld) {
21
- for (const x of ld) {
22
- node.dependencies.add(x);
23
- }
24
- this.lazyDependencies.delete(node.id);
25
- }
26
- this.nodes[node.id] = node;
27
- }
28
- }
29
- removeNode(nodeId) {
30
- const id = typeof nodeId === 'string' ? nodeId : nodeId.id;
31
- const removedNode = this.nodes[id];
32
- this.lazyDependencies.delete(id);
33
- delete this.nodes[id];
34
- if (removedNode) {
35
- for (const node of Object.values(this.nodes)) {
36
- node.dependencies.delete(removedNode.id);
37
- }
38
- }
39
- }
40
- /**
41
- * Return all nodes of a given type
42
- */
43
- nodesOfType(type) {
44
- return Object.values(this.nodes).filter(n => n.type === type);
45
- }
46
- /**
47
- * Return all nodes that depend on a given node
48
- */
49
- dependees(nodeId) {
50
- const id = typeof nodeId === 'string' ? nodeId : nodeId.id;
51
- return Object.values(this.nodes).filter(n => n.dependencies.has(id));
52
- }
53
- /**
54
- * Add a dependency, that may come before or after the nodes involved
55
- */
56
- addDependency(fromId, toId) {
57
- const node = this.nodes[fromId];
58
- if (node) {
59
- node.dependencies.add(toId);
60
- return;
61
- }
62
- let lazyDeps = this.lazyDependencies.get(fromId);
63
- if (!lazyDeps) {
64
- lazyDeps = [];
65
- this.lazyDependencies.set(fromId, lazyDeps);
66
- }
67
- lazyDeps.push(toId);
68
- }
69
- tryGetNode(id) {
70
- return this.nodes[id];
71
- }
72
- node(id) {
73
- const ret = this.nodes[id];
74
- if (!ret) {
75
- throw new error_1.ToolkitError(`No node with id ${id} among ${Object.keys(this.nodes)}`);
76
- }
77
- return ret;
78
- }
79
- absorb(graph) {
80
- this.addNodes(...Object.values(graph.nodes));
81
- }
82
- hasFailed() {
83
- return Object.values(this.nodes).some((n) => n.deploymentState === work_graph_types_1.DeploymentState.FAILED);
84
- }
85
- doParallel(concurrency, actions) {
86
- return this.forAllArtifacts(concurrency, async (x) => {
87
- switch (x.type) {
88
- case 'stack':
89
- await actions.deployStack(x);
90
- break;
91
- case 'asset-build':
92
- await actions.buildAsset(x);
93
- break;
94
- case 'asset-publish':
95
- await actions.publishAsset(x);
96
- break;
97
- }
98
- });
99
- }
100
- /**
101
- * Return the set of unblocked nodes
102
- */
103
- ready() {
104
- this.updateReadyPool();
105
- return this.readyPool;
106
- }
107
- forAllArtifacts(n, fn) {
108
- const graph = this;
109
- // If 'n' is a number, we limit all concurrency equally (effectively we will be using totalMax)
110
- // If 'n' is a record, we limit each job independently (effectively we will be using max)
111
- const max = typeof n === 'number' ?
112
- {
113
- 'asset-build': n,
114
- 'asset-publish': n,
115
- 'stack': n,
116
- } : n;
117
- const totalMax = typeof n === 'number' ? n : sum(Object.values(n));
118
- return new Promise((ok, fail) => {
119
- let active = {
120
- 'asset-build': 0,
121
- 'asset-publish': 0,
122
- 'stack': 0,
123
- };
124
- function totalActive() {
125
- return sum(Object.values(active));
126
- }
127
- start();
128
- function start() {
129
- graph.updateReadyPool();
130
- for (let i = 0; i < graph.readyPool.length;) {
131
- const node = graph.readyPool[i];
132
- if (active[node.type] < max[node.type] && totalActive() < totalMax) {
133
- graph.readyPool.splice(i, 1);
134
- startOne(node);
135
- }
136
- else {
137
- i += 1;
138
- }
139
- }
140
- if (totalActive() === 0) {
141
- if (graph.done()) {
142
- ok();
143
- }
144
- // wait for other active deploys to finish before failing
145
- if (graph.hasFailed()) {
146
- fail(graph.error);
147
- }
148
- }
149
- }
150
- function startOne(x) {
151
- x.deploymentState = work_graph_types_1.DeploymentState.DEPLOYING;
152
- active[x.type]++;
153
- void fn(x)
154
- .finally(() => {
155
- active[x.type]--;
156
- })
157
- .then(() => {
158
- graph.deployed(x);
159
- start();
160
- }).catch((err) => {
161
- // By recording the failure immediately as the queued task exits, we prevent the next
162
- // queued task from starting.
163
- graph.failed(x, err);
164
- start();
165
- });
166
- }
167
- });
168
- }
169
- done() {
170
- return Object.values(this.nodes).every((n) => work_graph_types_1.DeploymentState.COMPLETED === n.deploymentState);
171
- }
172
- deployed(node) {
173
- node.deploymentState = work_graph_types_1.DeploymentState.COMPLETED;
174
- }
175
- failed(node, error) {
176
- this.error = error;
177
- node.deploymentState = work_graph_types_1.DeploymentState.FAILED;
178
- this.skipRest();
179
- this.readyPool.splice(0);
180
- }
181
- toString() {
182
- return [
183
- 'digraph D {',
184
- ...Object.entries(this.nodes).flatMap(([id, node]) => renderNode(id, node)),
185
- '}',
186
- ].join('\n');
187
- function renderNode(id, node) {
188
- const ret = [];
189
- if (node.deploymentState === work_graph_types_1.DeploymentState.COMPLETED) {
190
- ret.push(` ${gv(id, { style: 'filled', fillcolor: 'yellow', comment: node.note })};`);
191
- }
192
- else {
193
- ret.push(` ${gv(id, { comment: node.note })};`);
194
- }
195
- for (const dep of node.dependencies) {
196
- ret.push(` ${gv(id)} -> ${gv(dep)};`);
197
- }
198
- return ret;
199
- }
200
- }
201
- /**
202
- * Ensure all dependencies actually exist. This protects against scenarios such as the following:
203
- * StackA depends on StackB, but StackB is not selected to deploy. The dependency is redundant
204
- * and will be dropped.
205
- * This assumes the manifest comes uncorrupted so we will not fail if a dependency is not found.
206
- */
207
- removeUnavailableDependencies() {
208
- for (const node of Object.values(this.nodes)) {
209
- const removeDeps = Array.from(node.dependencies).filter((dep) => this.nodes[dep] === undefined);
210
- removeDeps.forEach((d) => {
211
- node.dependencies.delete(d);
212
- });
213
- }
214
- }
215
- /**
216
- * Remove all asset publishing steps for assets that are already published, and then build
217
- * that aren't used anymore.
218
- *
219
- * Do this in parallel, because there may be a lot of assets in an application (seen in practice: >100 assets)
220
- */
221
- async removeUnnecessaryAssets(isUnnecessary) {
222
- (0, logging_1.debug)('Checking for previously published assets');
223
- const publishes = this.nodesOfType('asset-publish');
224
- const classifiedNodes = await (0, parallel_1.parallelPromises)(8, publishes.map((assetNode) => async () => [assetNode, await isUnnecessary(assetNode)]));
225
- const alreadyPublished = classifiedNodes.filter(([_, unnecessary]) => unnecessary).map(([assetNode, _]) => assetNode);
226
- for (const assetNode of alreadyPublished) {
227
- this.removeNode(assetNode);
228
- }
229
- (0, logging_1.debug)(`${publishes.length} total assets, ${publishes.length - alreadyPublished.length} still need to be published`);
230
- // Now also remove any asset build steps that don't have any dependencies on them anymore
231
- const unusedBuilds = this.nodesOfType('asset-build').filter(build => this.dependees(build).length === 0);
232
- for (const unusedBuild of unusedBuilds) {
233
- this.removeNode(unusedBuild);
234
- }
235
- }
236
- updateReadyPool() {
237
- var _a;
238
- const activeCount = Object.values(this.nodes).filter((x) => x.deploymentState === work_graph_types_1.DeploymentState.DEPLOYING).length;
239
- const pendingCount = Object.values(this.nodes).filter((x) => x.deploymentState === work_graph_types_1.DeploymentState.PENDING).length;
240
- const newlyReady = Object.values(this.nodes).filter((x) => x.deploymentState === work_graph_types_1.DeploymentState.PENDING &&
241
- Array.from(x.dependencies).every((id) => this.node(id).deploymentState === work_graph_types_1.DeploymentState.COMPLETED));
242
- // Add newly available nodes to the ready pool
243
- for (const node of newlyReady) {
244
- node.deploymentState = work_graph_types_1.DeploymentState.QUEUED;
245
- this.readyPool.push(node);
246
- }
247
- // Remove nodes from the ready pool that have already started deploying
248
- retainOnly(this.readyPool, (node) => node.deploymentState === work_graph_types_1.DeploymentState.QUEUED);
249
- // Sort by reverse priority
250
- this.readyPool.sort((a, b) => { var _a, _b; return ((_a = b.priority) !== null && _a !== void 0 ? _a : 0) - ((_b = a.priority) !== null && _b !== void 0 ? _b : 0); });
251
- if (this.readyPool.length === 0 && activeCount === 0 && pendingCount > 0) {
252
- const cycle = (_a = this.findCycle()) !== null && _a !== void 0 ? _a : ['No cycle found!'];
253
- (0, logging_1.trace)(`Cycle ${cycle.join(' -> ')} in graph ${this}`);
254
- throw new error_1.ToolkitError(`Unable to make progress anymore, dependency cycle between remaining artifacts: ${cycle.join(' -> ')} (run with -vv for full graph)`);
255
- }
256
- }
257
- skipRest() {
258
- for (const node of Object.values(this.nodes)) {
259
- if ([work_graph_types_1.DeploymentState.QUEUED, work_graph_types_1.DeploymentState.PENDING].includes(node.deploymentState)) {
260
- node.deploymentState = work_graph_types_1.DeploymentState.SKIPPED;
261
- }
262
- }
263
- }
264
- /**
265
- * Find cycles in a graph
266
- *
267
- * Not the fastest, but effective and should be rare
268
- */
269
- findCycle() {
270
- const seen = new Set();
271
- const self = this;
272
- for (const nodeId of Object.keys(this.nodes)) {
273
- const cycle = recurse(nodeId, [nodeId]);
274
- if (cycle) {
275
- return cycle;
276
- }
277
- }
278
- return undefined;
279
- function recurse(nodeId, path) {
280
- var _a;
281
- if (seen.has(nodeId)) {
282
- return undefined;
283
- }
284
- try {
285
- for (const dep of (_a = self.nodes[nodeId].dependencies) !== null && _a !== void 0 ? _a : []) {
286
- const index = path.indexOf(dep);
287
- if (index > -1) {
288
- return [...path.slice(index), dep];
289
- }
290
- const cycle = recurse(dep, [...path, dep]);
291
- if (cycle) {
292
- return cycle;
293
- }
294
- }
295
- return undefined;
296
- }
297
- finally {
298
- seen.add(nodeId);
299
- }
300
- }
301
- }
302
- /**
303
- * Whether the `end` node is reachable from the `start` node, following the dependency arrows
304
- */
305
- reachable(start, end) {
306
- const seen = new Set();
307
- const self = this;
308
- return recurse(start);
309
- function recurse(current) {
310
- if (seen.has(current)) {
311
- return false;
312
- }
313
- seen.add(current);
314
- if (current === end) {
315
- return true;
316
- }
317
- for (const dep of self.nodes[current].dependencies) {
318
- if (recurse(dep)) {
319
- return true;
320
- }
321
- }
322
- return false;
323
- }
324
- }
325
- }
326
- exports.WorkGraph = WorkGraph;
327
- function sum(xs) {
328
- let ret = 0;
329
- for (const x of xs) {
330
- ret += x;
331
- }
332
- return ret;
333
- }
334
- function retainOnly(xs, pred) {
335
- xs.splice(0, xs.length, ...xs.filter(pred));
336
- }
337
- function gv(id, attrs) {
338
- const attrString = Object.entries(attrs !== null && attrs !== void 0 ? attrs : {}).flatMap(([k, v]) => v !== undefined ? [`${k}="${v}"`] : []).join(',');
339
- return attrString ? `"${simplifyId(id)}" [${attrString}]` : `"${simplifyId(id)}"`;
340
- }
341
- function simplifyId(id) {
342
- return id.replace(/([0-9a-f]{6})[0-9a-f]{6,}/g, '$1');
343
- }
344
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"work-graph.js","sourceRoot":"","sources":["work-graph.ts"],"names":[],"mappings":"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","sourcesContent":["import { parallelPromises } from './parallel';\nimport { WorkNode, DeploymentState, StackNode, AssetBuildNode, AssetPublishNode } from './work-graph-types';\nimport { debug, trace } from '../logging';\nimport { ToolkitError } from '../toolkit/error';\n\nexport type Concurrency = number | Record<WorkNode['type'], number>;\n\nexport class WorkGraph {\n  public readonly nodes: Record<string, WorkNode>;\n  private readonly readyPool: Array<WorkNode> = [];\n  private readonly lazyDependencies = new Map<string, string[]>();\n  public error?: Error;\n\n  public constructor(nodes: Record<string, WorkNode> = {}) {\n    this.nodes = { ...nodes };\n  }\n\n  public addNodes(...nodes: WorkNode[]) {\n    for (const node of nodes) {\n      if (this.nodes[node.id]) {\n        throw new ToolkitError(`Duplicate use of node id: ${node.id}`);\n      }\n\n      const ld = this.lazyDependencies.get(node.id);\n      if (ld) {\n        for (const x of ld) {\n          node.dependencies.add(x);\n        }\n        this.lazyDependencies.delete(node.id);\n      }\n\n      this.nodes[node.id] = node;\n    }\n  }\n\n  public removeNode(nodeId: string | WorkNode) {\n    const id = typeof nodeId === 'string' ? nodeId : nodeId.id;\n    const removedNode = this.nodes[id];\n\n    this.lazyDependencies.delete(id);\n    delete this.nodes[id];\n\n    if (removedNode) {\n      for (const node of Object.values(this.nodes)) {\n        node.dependencies.delete(removedNode.id);\n      }\n    }\n  }\n\n  /**\n   * Return all nodes of a given type\n   */\n  public nodesOfType<T extends WorkNode['type']>(type: T): Extract<WorkNode, { type: T }>[] {\n    return Object.values(this.nodes).filter(n => n.type === type) as any;\n  }\n\n  /**\n   * Return all nodes that depend on a given node\n   */\n  public dependees(nodeId: string | WorkNode) {\n    const id = typeof nodeId === 'string' ? nodeId : nodeId.id;\n    return Object.values(this.nodes).filter(n => n.dependencies.has(id));\n  }\n\n  /**\n   * Add a dependency, that may come before or after the nodes involved\n   */\n  public addDependency(fromId: string, toId: string) {\n    const node = this.nodes[fromId];\n    if (node) {\n      node.dependencies.add(toId);\n      return;\n    }\n    let lazyDeps = this.lazyDependencies.get(fromId);\n    if (!lazyDeps) {\n      lazyDeps = [];\n      this.lazyDependencies.set(fromId, lazyDeps);\n    }\n    lazyDeps.push(toId);\n  }\n\n  public tryGetNode(id: string): WorkNode | undefined {\n    return this.nodes[id];\n  }\n\n  public node(id: string) {\n    const ret = this.nodes[id];\n    if (!ret) {\n      throw new ToolkitError(`No node with id ${id} among ${Object.keys(this.nodes)}`);\n    }\n    return ret;\n  }\n\n  public absorb(graph: WorkGraph) {\n    this.addNodes(...Object.values(graph.nodes));\n  }\n\n  private hasFailed(): boolean {\n    return Object.values(this.nodes).some((n) => n.deploymentState === DeploymentState.FAILED);\n  }\n\n  public doParallel(concurrency: Concurrency, actions: WorkGraphActions) {\n    return this.forAllArtifacts(concurrency, async (x: WorkNode) => {\n      switch (x.type) {\n        case 'stack':\n          await actions.deployStack(x);\n          break;\n        case 'asset-build':\n          await actions.buildAsset(x);\n          break;\n        case 'asset-publish':\n          await actions.publishAsset(x);\n          break;\n      }\n    });\n  }\n\n  /**\n   * Return the set of unblocked nodes\n   */\n  public ready(): ReadonlyArray<WorkNode> {\n    this.updateReadyPool();\n    return this.readyPool;\n  }\n\n  private forAllArtifacts(n: Concurrency, fn: (x: WorkNode) => Promise<void>): Promise<void> {\n    const graph = this;\n\n    // If 'n' is a number, we limit all concurrency equally (effectively we will be using totalMax)\n    // If 'n' is a record, we limit each job independently (effectively we will be using max)\n    const max: Record<WorkNode['type'], number> = typeof n === 'number' ?\n      {\n        'asset-build': n,\n        'asset-publish': n,\n        'stack': n,\n      } : n;\n    const totalMax = typeof n === 'number' ? n : sum(Object.values(n));\n\n    return new Promise((ok, fail) => {\n      let active: Record<WorkNode['type'], number> = {\n        'asset-build': 0,\n        'asset-publish': 0,\n        'stack': 0,\n      };\n      function totalActive() {\n        return sum(Object.values(active));\n      }\n\n      start();\n\n      function start() {\n        graph.updateReadyPool();\n\n        for (let i = 0; i < graph.readyPool.length; ) {\n          const node = graph.readyPool[i];\n\n          if (active[node.type] < max[node.type] && totalActive() < totalMax) {\n            graph.readyPool.splice(i, 1);\n            startOne(node);\n          } else {\n            i += 1;\n          }\n        }\n\n        if (totalActive() === 0) {\n          if (graph.done()) {\n            ok();\n          }\n          // wait for other active deploys to finish before failing\n          if (graph.hasFailed()) {\n            fail(graph.error);\n          }\n        }\n      }\n\n      function startOne(x: WorkNode) {\n        x.deploymentState = DeploymentState.DEPLOYING;\n        active[x.type]++;\n        void fn(x)\n          .finally(() => {\n            active[x.type]--;\n          })\n          .then(() => {\n            graph.deployed(x);\n            start();\n          }).catch((err) => {\n            // By recording the failure immediately as the queued task exits, we prevent the next\n            // queued task from starting.\n            graph.failed(x, err);\n            start();\n          });\n      }\n    });\n  }\n\n  private done(): boolean {\n    return Object.values(this.nodes).every((n) => DeploymentState.COMPLETED === n.deploymentState);\n  }\n\n  private deployed(node: WorkNode) {\n    node.deploymentState = DeploymentState.COMPLETED;\n  }\n\n  private failed(node: WorkNode, error?: Error) {\n    this.error = error;\n    node.deploymentState = DeploymentState.FAILED;\n    this.skipRest();\n    this.readyPool.splice(0);\n  }\n\n  public toString() {\n    return [\n      'digraph D {',\n      ...Object.entries(this.nodes).flatMap(([id, node]) => renderNode(id, node)),\n      '}',\n    ].join('\\n');\n\n    function renderNode(id: string, node: WorkNode): string[] {\n      const ret = [];\n      if (node.deploymentState === DeploymentState.COMPLETED) {\n        ret.push(`  ${gv(id, { style: 'filled', fillcolor: 'yellow', comment: node.note })};`);\n      } else {\n        ret.push(`  ${gv(id, { comment: node.note })};`);\n      }\n      for (const dep of node.dependencies) {\n        ret.push(`  ${gv(id)} -> ${gv(dep)};`);\n      }\n      return ret;\n    }\n\n  }\n\n  /**\n   * Ensure all dependencies actually exist. This protects against scenarios such as the following:\n   * StackA depends on StackB, but StackB is not selected to deploy. The dependency is redundant\n   * and will be dropped.\n   * This assumes the manifest comes uncorrupted so we will not fail if a dependency is not found.\n   */\n  public removeUnavailableDependencies() {\n    for (const node of Object.values(this.nodes)) {\n      const removeDeps = Array.from(node.dependencies).filter((dep) => this.nodes[dep] === undefined);\n\n      removeDeps.forEach((d) => {\n        node.dependencies.delete(d);\n      });\n    }\n  }\n\n  /**\n   * Remove all asset publishing steps for assets that are already published, and then build\n   * that aren't used anymore.\n   *\n   * Do this in parallel, because there may be a lot of assets in an application (seen in practice: >100 assets)\n   */\n  public async removeUnnecessaryAssets(isUnnecessary: (x: AssetPublishNode) => Promise<boolean>) {\n    debug('Checking for previously published assets');\n\n    const publishes = this.nodesOfType('asset-publish');\n\n    const classifiedNodes = await parallelPromises(\n      8,\n      publishes.map((assetNode) => async() => [assetNode, await isUnnecessary(assetNode)] as const));\n\n    const alreadyPublished = classifiedNodes.filter(([_, unnecessary]) => unnecessary).map(([assetNode, _]) => assetNode);\n    for (const assetNode of alreadyPublished) {\n      this.removeNode(assetNode);\n    }\n\n    debug(`${publishes.length} total assets, ${publishes.length - alreadyPublished.length} still need to be published`);\n\n    // Now also remove any asset build steps that don't have any dependencies on them anymore\n    const unusedBuilds = this.nodesOfType('asset-build').filter(build => this.dependees(build).length === 0);\n    for (const unusedBuild of unusedBuilds) {\n      this.removeNode(unusedBuild);\n    }\n  }\n\n  private updateReadyPool() {\n    const activeCount = Object.values(this.nodes).filter((x) => x.deploymentState === DeploymentState.DEPLOYING).length;\n    const pendingCount = Object.values(this.nodes).filter((x) => x.deploymentState === DeploymentState.PENDING).length;\n\n    const newlyReady = Object.values(this.nodes).filter((x) =>\n      x.deploymentState === DeploymentState.PENDING &&\n      Array.from(x.dependencies).every((id) => this.node(id).deploymentState === DeploymentState.COMPLETED));\n\n    // Add newly available nodes to the ready pool\n    for (const node of newlyReady) {\n      node.deploymentState = DeploymentState.QUEUED;\n      this.readyPool.push(node);\n    }\n\n    // Remove nodes from the ready pool that have already started deploying\n    retainOnly(this.readyPool, (node) => node.deploymentState === DeploymentState.QUEUED);\n\n    // Sort by reverse priority\n    this.readyPool.sort((a, b) => (b.priority ?? 0) - (a.priority ?? 0));\n\n    if (this.readyPool.length === 0 && activeCount === 0 && pendingCount > 0) {\n      const cycle = this.findCycle() ?? ['No cycle found!'];\n      trace(`Cycle ${cycle.join(' -> ')} in graph ${this}`);\n      throw new ToolkitError(`Unable to make progress anymore, dependency cycle between remaining artifacts: ${cycle.join(' -> ')} (run with -vv for full graph)`);\n    }\n  }\n\n  private skipRest() {\n    for (const node of Object.values(this.nodes)) {\n      if ([DeploymentState.QUEUED, DeploymentState.PENDING].includes(node.deploymentState)) {\n        node.deploymentState = DeploymentState.SKIPPED;\n      }\n    }\n  }\n\n  /**\n   * Find cycles in a graph\n   *\n   * Not the fastest, but effective and should be rare\n   */\n  public findCycle(): string[] | undefined {\n    const seen = new Set<string>();\n    const self = this;\n    for (const nodeId of Object.keys(this.nodes)) {\n      const cycle = recurse(nodeId, [nodeId]);\n      if (cycle) { return cycle; }\n    }\n    return undefined;\n\n    function recurse(nodeId: string, path: string[]): string[] | undefined {\n      if (seen.has(nodeId)) {\n        return undefined;\n      }\n      try {\n        for (const dep of self.nodes[nodeId].dependencies ?? []) {\n          const index = path.indexOf(dep);\n          if (index > -1) {\n            return [...path.slice(index), dep];\n          }\n\n          const cycle = recurse(dep, [...path, dep]);\n          if (cycle) { return cycle; }\n        }\n\n        return undefined;\n      } finally {\n        seen.add(nodeId);\n      }\n    }\n  }\n\n  /**\n   * Whether the `end` node is reachable from the `start` node, following the dependency arrows\n   */\n  public reachable(start: string, end: string): boolean {\n    const seen = new Set<string>();\n    const self = this;\n    return recurse(start);\n\n    function recurse(current: string) {\n      if (seen.has(current)) {\n        return false;\n      }\n      seen.add(current);\n\n      if (current === end) {\n        return true;\n      }\n      for (const dep of self.nodes[current].dependencies) {\n        if (recurse(dep)) {\n          return true;\n        }\n      }\n      return false;\n    }\n  }\n}\n\nexport interface WorkGraphActions {\n  deployStack: (stackNode: StackNode) => Promise<void>;\n  buildAsset: (assetNode: AssetBuildNode) => Promise<void>;\n  publishAsset: (assetNode: AssetPublishNode) => Promise<void>;\n}\n\nfunction sum(xs: number[]) {\n  let ret = 0;\n  for (const x of xs) {\n    ret += x;\n  }\n  return ret;\n}\n\nfunction retainOnly<A>(xs: A[], pred: (x: A) => boolean) {\n  xs.splice(0, xs.length, ...xs.filter(pred));\n}\n\nfunction gv(id: string, attrs?: Record<string, string | undefined>) {\n  const attrString = Object.entries(attrs ?? {}).flatMap(([k, v]) => v !== undefined ? [`${k}=\"${v}\"`] : []).join(',');\n\n  return attrString ? `\"${simplifyId(id)}\" [${attrString}]` : `\"${simplifyId(id)}\"`;\n}\n\nfunction simplifyId(id: string) {\n  return id.replace(/([0-9a-f]{6})[0-9a-f]{6,}/g, '$1');\n}\n"]}
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes