@aws-cdk/toolkit-lib 1.18.1 → 1.19.1
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/build-info.json +2 -2
- package/lib/actions/destroy/index.d.ts +6 -0
- package/lib/actions/destroy/index.js +1 -1
- package/lib/api/cloud-assembly/private/prepare-source.d.ts +4 -0
- package/lib/api/cloud-assembly/private/prepare-source.js +22 -13
- package/lib/api/diff/diff-formatter.js +2 -2
- package/lib/api/hotswap/hotswap-deployments.js +15 -3
- package/lib/api/io/private/span.d.ts +44 -14
- package/lib/api/io/private/span.js +23 -1
- package/lib/api/tree.js +2 -2
- package/lib/api/work-graph/build-destroy-work-graph.d.ts +11 -0
- package/lib/api/work-graph/build-destroy-work-graph.js +35 -0
- package/lib/api/work-graph/index.d.ts +1 -0
- package/lib/api/work-graph/index.js +2 -1
- package/lib/api/work-graph/work-graph-builder.js +1 -1
- package/lib/api/work-graph/work-graph.d.ts +7 -0
- package/lib/api/work-graph/work-graph.js +15 -1
- package/lib/payloads/hotswap.d.ts +7 -0
- package/lib/payloads/hotswap.js +1 -1
- package/lib/toolkit/toolkit.js +30 -8
- package/package.json +2 -2
|
@@ -58,6 +58,11 @@ async function tryHotswapDeployment(sdkProvider, ioHelper, assetParams, cloudFor
|
|
|
58
58
|
});
|
|
59
59
|
const result = await hotswapDeployment(sdkProvider, hotswapSpan, assetParams, stackArtifact, hotswapMode, hotswapPropertyOverrides);
|
|
60
60
|
await hotswapSpan.end(result);
|
|
61
|
+
// This is a hotswap error that was caught during the hotswapDeployment function
|
|
62
|
+
// It is thrown after the hotswap span ends so it can be reported to telemetry
|
|
63
|
+
if (result.error) {
|
|
64
|
+
throw result.error;
|
|
65
|
+
}
|
|
61
66
|
if (result?.hotswapped === true) {
|
|
62
67
|
return {
|
|
63
68
|
type: 'did-deploy-stack',
|
|
@@ -112,13 +117,20 @@ async function hotswapDeployment(sdkProvider, ioSpan, assetParams, stack, hotswa
|
|
|
112
117
|
}
|
|
113
118
|
}
|
|
114
119
|
// apply the short-circuitable changes
|
|
115
|
-
|
|
120
|
+
let error;
|
|
121
|
+
try {
|
|
122
|
+
await applyAllHotswapOperations(sdk, ioSpan, hotswappable);
|
|
123
|
+
}
|
|
124
|
+
catch (e) {
|
|
125
|
+
error = e;
|
|
126
|
+
}
|
|
116
127
|
return {
|
|
117
128
|
stack,
|
|
118
129
|
mode: hotswapMode,
|
|
119
|
-
hotswapped:
|
|
130
|
+
hotswapped: !error,
|
|
120
131
|
hotswappableChanges,
|
|
121
132
|
nonHotswappableChanges,
|
|
133
|
+
error,
|
|
122
134
|
};
|
|
123
135
|
}
|
|
124
136
|
/**
|
|
@@ -440,4 +452,4 @@ function nonHotswappableResourceMessage(subject, reason) {
|
|
|
440
452
|
}
|
|
441
453
|
return (0, util_1.format)('resource: %s, type: %s, reason: %s', chalk.bold(subject.logicalId), chalk.bold(subject.resourceType), chalk.red(reason));
|
|
442
454
|
}
|
|
443
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-deployments.js","sourceRoot":"","sources":["hotswap-deployments.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'util';\nimport type * as cxapi from '@aws-cdk/cloud-assembly-api';\nimport * as cfn_diff from '@aws-cdk/cloudformation-diff';\nimport type { WaiterResult } from '@smithy/util-waiter';\nimport * as chalk from 'chalk';\nimport type { AffectedResource, HotswapResult, ResourceSubject, ResourceChange, NonHotswappableChange } from '../../payloads';\nimport { NonHotswappableReason } from '../../payloads';\nimport { formatErrorMessage } from '../../util';\nimport type { SDK, SdkProvider } from '../aws-auth/private';\nimport type { CloudFormationStack, NestedStackTemplates } from '../cloudformation';\nimport { loadCurrentTemplateWithNestedStacks, EvaluateCloudFormationTemplate } from '../cloudformation';\nimport { isHotswappableAppSyncChange } from './appsync-mapping-templates';\nimport { isHotswappableBedrockAgentCoreRuntimeChange } from './bedrock-agentcore-runtimes';\nimport { isHotswappableCodeBuildProjectChange } from './code-build-projects';\nimport type {\n  HotswapChange,\n  HotswapOperation,\n  RejectedChange,\n  HotswapPropertyOverrides,\n} from './common';\nimport {\n  ICON,\n  nonHotswappableResource,\n} from './common';\nimport { isHotswappableEcsServiceChange } from './ecs-services';\nimport { isHotswappableLambdaFunctionChange } from './lambda-functions';\nimport {\n  skipChangeForS3DeployCustomResourcePolicy,\n  isHotswappableS3BucketDeploymentChange,\n} from './s3-bucket-deployments';\nimport { isHotswappableStateMachineChange } from './stepfunctions-state-machines';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\nimport type { SuccessfulDeployStackResult } from '../deployments';\nimport { IO, SPAN } from '../io/private';\nimport type { IMessageSpan, IoHelper } from '../io/private';\nimport { Mode } from '../plugin';\n\n// Must use a require() otherwise esbuild complains about calling a namespace\n// eslint-disable-next-line @typescript-eslint/no-require-imports,@typescript-eslint/consistent-type-imports\nconst pLimit: typeof import('p-limit') = require('p-limit');\n\ntype HotswapDetector = (\n  logicalId: string,\n  change: ResourceChange,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n) => Promise<HotswapChange[]>;\n\ntype HotswapMode = 'hotswap-only' | 'fall-back';\n\nconst RESOURCE_DETECTORS: { [key: string]: HotswapDetector } = {\n  // Lambda\n  'AWS::Lambda::Function': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Version': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Alias': isHotswappableLambdaFunctionChange,\n\n  // AppSync\n  'AWS::AppSync::Resolver': isHotswappableAppSyncChange,\n  'AWS::AppSync::FunctionConfiguration': isHotswappableAppSyncChange,\n  'AWS::AppSync::GraphQLSchema': isHotswappableAppSyncChange,\n  'AWS::AppSync::ApiKey': isHotswappableAppSyncChange,\n\n  'AWS::BedrockAgentCore::Runtime': isHotswappableBedrockAgentCoreRuntimeChange,\n  'AWS::ECS::TaskDefinition': isHotswappableEcsServiceChange,\n  'AWS::CodeBuild::Project': isHotswappableCodeBuildProjectChange,\n  'AWS::StepFunctions::StateMachine': isHotswappableStateMachineChange,\n  'Custom::CDKBucketDeployment': isHotswappableS3BucketDeploymentChange,\n  'AWS::IAM::Policy': async (\n    logicalId: string,\n    change: ResourceChange,\n    evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  ): Promise<HotswapChange[]> => {\n    // If the policy is for a S3BucketDeploymentChange, we can ignore the change\n    if (await skipChangeForS3DeployCustomResourcePolicy(logicalId, change, evaluateCfnTemplate)) {\n      return [];\n    }\n\n    return [nonHotswappableResource(change)];\n  },\n\n  'AWS::CDK::Metadata': async () => [],\n};\n\n/**\n * Perform a hotswap deployment, short-circuiting CloudFormation if possible.\n * If it's not possible to short-circuit the deployment\n * (because the CDK Stack contains changes that cannot be deployed without CloudFormation),\n * returns `undefined`.\n */\nexport async function tryHotswapDeployment(\n  sdkProvider: SdkProvider,\n  ioHelper: IoHelper,\n  assetParams: { [key: string]: string },\n  cloudFormationStack: CloudFormationStack,\n  stackArtifact: cxapi.CloudFormationStackArtifact,\n  hotswapMode: HotswapMode,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<SuccessfulDeployStackResult | undefined> {\n  const hotswapSpan = await ioHelper.span(SPAN.HOTSWAP).begin({\n    stack: stackArtifact,\n    mode: hotswapMode,\n  });\n\n  const result = await hotswapDeployment(\n    sdkProvider,\n    hotswapSpan,\n    assetParams,\n    stackArtifact,\n    hotswapMode,\n    hotswapPropertyOverrides,\n  );\n\n  await hotswapSpan.end(result);\n\n  if (result?.hotswapped === true) {\n    return {\n      type: 'did-deploy-stack',\n      noOp: result.hotswappableChanges.length === 0,\n      stackArn: cloudFormationStack.stackId,\n      outputs: cloudFormationStack.outputs,\n    };\n  }\n\n  return undefined;\n}\n\n/**\n * Perform a hotswap deployment, short-circuiting CloudFormation if possible.\n * Returns information about the attempted hotswap deployment\n */\nasync function hotswapDeployment(\n  sdkProvider: SdkProvider,\n  ioSpan: IMessageSpan<any>,\n  assetParams: { [key: string]: string },\n  stack: cxapi.CloudFormationStackArtifact,\n  hotswapMode: HotswapMode,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<Omit<HotswapResult, 'duration'>> {\n  // resolve the environment, so we can substitute things like AWS::Region in CFN expressions\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stack.environment);\n  // create a new SDK using the CLI credentials, because the default one will not work for new-style synthesis -\n  // it assumes the bootstrap deploy Role, which doesn't have permissions to update Lambda functions\n  const sdk = (await sdkProvider.forEnvironment(resolvedEnv, Mode.ForWriting)).sdk;\n\n  const currentTemplate = await loadCurrentTemplateWithNestedStacks(stack, sdk);\n\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    stackArtifact: stack,\n    parameters: assetParams,\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    sdk,\n    nestedStacks: currentTemplate.nestedStacks,\n  });\n\n  const stackChanges = cfn_diff.fullDiff(currentTemplate.deployedRootTemplate, stack.template);\n  const { hotswappable, nonHotswappable } = await classifyResourceChanges(\n    stackChanges,\n    evaluateCfnTemplate,\n    sdk,\n    currentTemplate.nestedStacks, hotswapPropertyOverrides,\n  );\n\n  await logRejectedChanges(ioSpan, nonHotswappable, hotswapMode);\n\n  const hotswappableChanges = hotswappable.map(o => o.change);\n  const nonHotswappableChanges = nonHotswappable.map(n => n.change);\n\n  await ioSpan.notify(IO.CDK_TOOLKIT_I5401.msg('Hotswap plan created', {\n    stack,\n    mode: hotswapMode,\n    hotswappableChanges,\n    nonHotswappableChanges,\n  }));\n\n  // preserve classic hotswap behavior\n  if (hotswapMode === 'fall-back') {\n    if (nonHotswappableChanges.length > 0) {\n      return {\n        stack,\n        mode: hotswapMode,\n        hotswapped: false,\n        hotswappableChanges,\n        nonHotswappableChanges,\n      };\n    }\n  }\n\n  // apply the short-circuitable changes\n  await applyAllHotswapOperations(sdk, ioSpan, hotswappable);\n\n  return {\n    stack,\n    mode: hotswapMode,\n    hotswapped: true,\n    hotswappableChanges,\n    nonHotswappableChanges,\n  };\n}\n\ninterface ClassifiedChanges {\n  hotswappable: HotswapOperation[];\n  nonHotswappable: RejectedChange[];\n}\n\n/**\n * Classifies all changes to all resources as either hotswappable or not.\n * Metadata changes are excluded from the list of (non)hotswappable resources.\n */\nasync function classifyResourceChanges(\n  stackChanges: cfn_diff.TemplateDiff,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: SDK,\n  nestedStackNames: { [nestedStackName: string]: NestedStackTemplates },\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<ClassifiedChanges> {\n  const resourceDifferences = getStackResourceDifferences(stackChanges);\n\n  const promises: Array<() => Promise<HotswapChange[]>> = [];\n  const hotswappableResources = new Array<HotswapOperation>();\n  const nonHotswappableResources = new Array<RejectedChange>();\n  for (const logicalId of Object.keys(stackChanges.outputs.changes)) {\n    nonHotswappableResources.push({\n      hotswappable: false,\n      change: {\n        reason: NonHotswappableReason.OUTPUT,\n        description: 'output was changed',\n        subject: {\n          type: 'Output',\n          logicalId,\n          metadata: evaluateCfnTemplate.metadataFor(logicalId),\n        },\n      },\n    });\n  }\n  // gather the results of the detector functions\n  for (const [logicalId, change] of Object.entries(resourceDifferences)) {\n    if (change.newValue?.Type === 'AWS::CloudFormation::Stack' && change.oldValue?.Type === 'AWS::CloudFormation::Stack') {\n      const nestedHotswappableResources = await findNestedHotswappableChanges(\n        logicalId,\n        change,\n        nestedStackNames,\n        evaluateCfnTemplate,\n        sdk,\n        hotswapPropertyOverrides,\n      );\n      hotswappableResources.push(...nestedHotswappableResources.hotswappable);\n      nonHotswappableResources.push(...nestedHotswappableResources.nonHotswappable);\n\n      continue;\n    }\n\n    const hotswappableChangeCandidate = isCandidateForHotswapping(logicalId, change, evaluateCfnTemplate);\n    // we don't need to run this through the detector functions, we can already judge this\n    if ('hotswappable' in hotswappableChangeCandidate) {\n      if (!hotswappableChangeCandidate.hotswappable) {\n        nonHotswappableResources.push(hotswappableChangeCandidate);\n      }\n\n      continue;\n    }\n\n    const resourceType: string = hotswappableChangeCandidate.newValue.Type;\n    if (resourceType in RESOURCE_DETECTORS) {\n      // run detector functions lazily to prevent unhandled promise rejections\n      promises.push(() =>\n        RESOURCE_DETECTORS[resourceType](logicalId, hotswappableChangeCandidate, evaluateCfnTemplate, hotswapPropertyOverrides),\n      );\n    } else {\n      nonHotswappableResources.push(nonHotswappableResource(hotswappableChangeCandidate));\n    }\n  }\n\n  // resolve all detector results\n  const changesDetectionResults: Array<HotswapChange[]> = [];\n  for (const detectorResultPromises of promises) {\n    // Constant set of promises per resource\n    // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n    const hotswapDetectionResults = await Promise.all(await detectorResultPromises());\n    changesDetectionResults.push(hotswapDetectionResults);\n  }\n\n  for (const resourceDetectionResults of changesDetectionResults) {\n    for (const propertyResult of resourceDetectionResults) {\n      propertyResult.hotswappable\n        ? hotswappableResources.push(propertyResult)\n        : nonHotswappableResources.push(propertyResult);\n    }\n  }\n\n  return {\n    hotswappable: hotswappableResources,\n    nonHotswappable: nonHotswappableResources,\n  };\n}\n\n/**\n * Returns all changes to resources in the given Stack.\n *\n * @param stackChanges - the collection of all changes to a given Stack\n */\nfunction getStackResourceDifferences(stackChanges: cfn_diff.TemplateDiff): {\n  [logicalId: string]: cfn_diff.ResourceDifference;\n} {\n  // we need to collapse logical ID rename changes into one change,\n  // as they are represented in stackChanges as a pair of two changes: one addition and one removal\n  const allResourceChanges: { [logId: string]: cfn_diff.ResourceDifference } = stackChanges.resources.changes;\n  const allRemovalChanges = filterDict(allResourceChanges, (resChange) => resChange.isRemoval);\n  const allNonRemovalChanges = filterDict(allResourceChanges, (resChange) => !resChange.isRemoval);\n  for (const [logId, nonRemovalChange] of Object.entries(allNonRemovalChanges)) {\n    if (nonRemovalChange.isAddition) {\n      const addChange = nonRemovalChange;\n      // search for an identical removal change\n      const identicalRemovalChange = Object.entries(allRemovalChanges).find(([_, remChange]) => {\n        return changesAreForSameResource(remChange, addChange);\n      });\n      // if we found one, then this means this is a rename change\n      if (identicalRemovalChange) {\n        const [removedLogId, removedResourceChange] = identicalRemovalChange;\n        allNonRemovalChanges[logId] = makeRenameDifference(removedResourceChange, addChange);\n        // delete the removal change that forms the rename pair\n        delete allRemovalChanges[removedLogId];\n      }\n    }\n  }\n  // the final result are all of the remaining removal changes,\n  // plus all of the non-removal changes\n  // (we saved the rename changes in that object already)\n  return {\n    ...allRemovalChanges,\n    ...allNonRemovalChanges,\n  };\n}\n\n/** Filters an object with string keys based on whether the callback returns 'true' for the given value in the object. */\nfunction filterDict<T>(dict: { [key: string]: T }, func: (t: T) => boolean): { [key: string]: T } {\n  return Object.entries(dict).reduce(\n    (acc, [key, t]) => {\n      if (func(t)) {\n        acc[key] = t;\n      }\n      return acc;\n    },\n    {} as { [key: string]: T },\n  );\n}\n\n/** Finds any hotswappable changes in all nested stacks. */\nasync function findNestedHotswappableChanges(\n  logicalId: string,\n  change: cfn_diff.ResourceDifference,\n  nestedStackTemplates: { [nestedStackName: string]: NestedStackTemplates },\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: SDK,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<ClassifiedChanges> {\n  const nestedStack = nestedStackTemplates[logicalId];\n  if (!nestedStack.physicalName) {\n    return {\n      hotswappable: [],\n      nonHotswappable: [\n        {\n          hotswappable: false,\n          change: {\n            reason: NonHotswappableReason.NESTED_STACK_CREATION,\n            description: 'newly created nested stacks cannot be hotswapped',\n            subject: {\n              type: 'Resource',\n              logicalId,\n              resourceType: 'AWS::CloudFormation::Stack',\n              metadata: evaluateCfnTemplate.metadataFor(logicalId),\n            },\n          },\n        },\n      ],\n    };\n  }\n\n  const evaluateNestedCfnTemplate = await evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(\n    nestedStack.physicalName,\n    nestedStack.generatedTemplate,\n    change.newValue?.Properties?.Parameters,\n  );\n\n  const nestedDiff = cfn_diff.fullDiff(\n    nestedStackTemplates[logicalId].deployedTemplate,\n    nestedStackTemplates[logicalId].generatedTemplate,\n  );\n\n  return classifyResourceChanges(\n    nestedDiff,\n    evaluateNestedCfnTemplate,\n    sdk,\n    nestedStackTemplates[logicalId].nestedStackTemplates,\n    hotswapPropertyOverrides,\n  );\n}\n\n/** Returns 'true' if a pair of changes is for the same resource. */\nfunction changesAreForSameResource(\n  oldChange: cfn_diff.ResourceDifference,\n  newChange: cfn_diff.ResourceDifference,\n): boolean {\n  return (\n    oldChange.oldResourceType === newChange.newResourceType &&\n    // this isn't great, but I don't want to bring in something like underscore just for this comparison\n    JSON.stringify(oldChange.oldProperties) === JSON.stringify(newChange.newProperties)\n  );\n}\n\nfunction makeRenameDifference(\n  remChange: cfn_diff.ResourceDifference,\n  addChange: cfn_diff.ResourceDifference,\n): cfn_diff.ResourceDifference {\n  return new cfn_diff.ResourceDifference(\n    // we have to fill in the old value, because otherwise this will be classified as a non-hotswappable change\n    remChange.oldValue,\n    addChange.newValue,\n    {\n      resourceType: {\n        oldType: remChange.oldResourceType,\n        newType: addChange.newResourceType,\n      },\n      propertyDiffs: (addChange as any).propertyDiffs,\n      otherDiffs: (addChange as any).otherDiffs,\n    },\n  );\n}\n\n/**\n * Returns a `HotswappableChangeCandidate` if the change is hotswappable\n * Returns an empty `HotswappableChange` if the change is to CDK::Metadata\n * Returns a `NonHotswappableChange` if the change is not hotswappable\n */\nfunction isCandidateForHotswapping(\n  logicalId: string,\n  change: cfn_diff.ResourceDifference,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): RejectedChange | ResourceChange {\n  // a resource has been removed OR a resource has been added; we can't short-circuit that change\n  if (!change.oldValue) {\n    return {\n      hotswappable: false,\n      change: {\n        reason: NonHotswappableReason.RESOURCE_CREATION,\n        description: `resource '${logicalId}' was created by this deployment`,\n        subject: {\n          type: 'Resource',\n          logicalId,\n          resourceType: change.newValue!.Type,\n          metadata: evaluateCfnTemplate.metadataFor(logicalId),\n        },\n      },\n    };\n  } else if (!change.newValue) {\n    return {\n      hotswappable: false,\n      logicalId,\n      change: {\n        reason: NonHotswappableReason.RESOURCE_DELETION,\n        description: `resource '${logicalId}' was destroyed by this deployment`,\n        subject: {\n          type: 'Resource',\n          logicalId,\n          resourceType: change.oldValue.Type,\n          metadata: evaluateCfnTemplate.metadataFor(logicalId),\n        },\n      },\n    };\n  }\n\n  // a resource has had its type changed\n  if (change.newValue.Type !== change.oldValue.Type) {\n    return {\n      hotswappable: false,\n      change: {\n        reason: NonHotswappableReason.RESOURCE_TYPE_CHANGED,\n        description: `resource '${logicalId}' had its type changed from '${change.oldValue?.Type}' to '${change.newValue?.Type}'`,\n        subject: {\n          type: 'Resource',\n          logicalId,\n          resourceType: change.newValue.Type,\n          metadata: evaluateCfnTemplate.metadataFor(logicalId),\n        },\n      },\n    };\n  }\n\n  return {\n    logicalId,\n    oldValue: change.oldValue,\n    newValue: change.newValue,\n    propertyUpdates: change.propertyUpdates,\n    metadata: evaluateCfnTemplate.metadataFor(logicalId),\n  };\n}\n\nasync function applyAllHotswapOperations(sdk: SDK, ioSpan: IMessageSpan<any>, hotswappableChanges: HotswapOperation[]): Promise<void[]> {\n  if (hotswappableChanges.length === 0) {\n    return Promise.resolve([]);\n  }\n\n  await ioSpan.defaults.info(`\\n${ICON} hotswapping resources:`);\n  const limit = pLimit(10);\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  return Promise.all(hotswappableChanges.map(hotswapOperation => limit(() => {\n    return applyHotswapOperation(sdk, ioSpan, hotswapOperation);\n  })));\n}\n\nasync function applyHotswapOperation(sdk: SDK, ioSpan: IMessageSpan<any>, hotswapOperation: HotswapOperation): Promise<void> {\n  // note the type of service that was successfully hotswapped in the User-Agent\n  const customUserAgent = `cdk-hotswap/success-${hotswapOperation.service}`;\n  sdk.appendCustomUserAgent(customUserAgent);\n  const resourceText = (r: AffectedResource) => r.description ?? `${r.resourceType} '${r.physicalName ?? r.logicalId}'`;\n\n  await ioSpan.notify(IO.CDK_TOOLKIT_I5402.msg(\n    hotswapOperation.change.resources.map(r => format(`   ${ICON} %s`, chalk.bold(resourceText(r)))).join('\\n'),\n    hotswapOperation.change,\n  ));\n\n  // if the SDK call fails, an error will be thrown by the SDK\n  // and will prevent the green 'hotswapped!' text from being displayed\n  try {\n    await hotswapOperation.apply(sdk);\n  } catch (e: any) {\n    if (e.name === 'TimeoutError' || e.name === 'AbortError') {\n      const result: WaiterResult = JSON.parse(formatErrorMessage(e));\n      const error = new ToolkitError(formatWaiterErrorResult(result));\n      error.name = e.name;\n      throw error;\n    }\n    throw e;\n  }\n\n  await ioSpan.notify(IO.CDK_TOOLKIT_I5403.msg(\n    hotswapOperation.change.resources.map(r => format(`   ${ICON} %s %s`, chalk.bold(resourceText(r)), chalk.green('hotswapped!'))).join('\\n'),\n    hotswapOperation.change,\n  ));\n\n  sdk.removeCustomUserAgent(customUserAgent);\n}\n\nfunction formatWaiterErrorResult(result: WaiterResult) {\n  const main = [\n    `Resource is not in the expected state due to waiter status: ${result.state}`,\n    result.reason ? `${result.reason}.` : '',\n  ].join('. ');\n\n  if (result.observedResponses != null) {\n    const observedResponses = Object\n      .entries(result.observedResponses)\n      .map(([msg, count]) => `  - ${msg} (${count})`)\n      .join('\\n');\n\n    return `${main} Observed responses:\\n${observedResponses}`;\n  }\n\n  return main;\n}\n\nasync function logRejectedChanges(\n  ioSpan: IMessageSpan<any>,\n  rejectedChanges: RejectedChange[],\n  hotswapMode: HotswapMode,\n): Promise<void> {\n  if (rejectedChanges.length === 0) {\n    return;\n  }\n  /**\n   * EKS Services can have a task definition that doesn't refer to the task definition being updated.\n   * We have to log this as a non-hotswappable change to the task definition, but when we do,\n   * we wind up hotswapping the task definition and logging it as a non-hotswappable change.\n   *\n   * This logic prevents us from logging that change as non-hotswappable when we hotswap it.\n   */\n  if (hotswapMode === 'hotswap-only') {\n    rejectedChanges = rejectedChanges.filter((change) => change.hotswapOnlyVisible === true);\n\n    if (rejectedChanges.length === 0) {\n      return;\n    }\n  }\n\n  const messages = ['']; // start with empty line\n\n  if (hotswapMode === 'hotswap-only') {\n    messages.push(format('%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found. To reconcile these using CloudFormation, specify --hotswap-fallback')));\n  } else {\n    messages.push(format('%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found:')));\n  }\n\n  for (const { change } of rejectedChanges) {\n    messages.push('    ' + nonHotswappableChangeMessage(change));\n  }\n  messages.push(''); // newline\n\n  await ioSpan.defaults.info(messages.join('\\n'));\n}\n\n/**\n * Formats a NonHotswappableChange\n */\nfunction nonHotswappableChangeMessage(change: NonHotswappableChange): string {\n  const subject = change.subject;\n  const reason = change.description ?? change.reason;\n\n  switch (subject.type) {\n    case 'Output':\n      return format(\n        'output: %s, reason: %s',\n        chalk.bold(subject.logicalId),\n        chalk.red(reason),\n      );\n    case 'Resource':\n      return nonHotswappableResourceMessage(subject, reason);\n  }\n}\n\n/**\n * Formats a non-hotswappable resource subject\n */\nfunction nonHotswappableResourceMessage(subject: ResourceSubject, reason: string): string {\n  if (subject.rejectedProperties?.length) {\n    return format(\n      'resource: %s, type: %s, rejected changes: %s, reason: %s',\n      chalk.bold(subject.logicalId),\n      chalk.bold(subject.resourceType),\n      chalk.bold(subject.rejectedProperties),\n      chalk.red(reason),\n    );\n  }\n\n  return format(\n    'resource: %s, type: %s, reason: %s',\n    chalk.bold(subject.logicalId),\n    chalk.bold(subject.resourceType),\n    chalk.red(reason),\n  );\n}\n"]}
|
|
455
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-deployments.js","sourceRoot":"","sources":["hotswap-deployments.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'util';\nimport type * as cxapi from '@aws-cdk/cloud-assembly-api';\nimport * as cfn_diff from '@aws-cdk/cloudformation-diff';\nimport type { WaiterResult } from '@smithy/util-waiter';\nimport * as chalk from 'chalk';\nimport type { AffectedResource, HotswapResult, ResourceSubject, ResourceChange, NonHotswappableChange } from '../../payloads';\nimport { NonHotswappableReason } from '../../payloads';\nimport { formatErrorMessage } from '../../util';\nimport type { SDK, SdkProvider } from '../aws-auth/private';\nimport type { CloudFormationStack, NestedStackTemplates } from '../cloudformation';\nimport { loadCurrentTemplateWithNestedStacks, EvaluateCloudFormationTemplate } from '../cloudformation';\nimport { isHotswappableAppSyncChange } from './appsync-mapping-templates';\nimport { isHotswappableBedrockAgentCoreRuntimeChange } from './bedrock-agentcore-runtimes';\nimport { isHotswappableCodeBuildProjectChange } from './code-build-projects';\nimport type {\n  HotswapChange,\n  HotswapOperation,\n  RejectedChange,\n  HotswapPropertyOverrides,\n} from './common';\nimport {\n  ICON,\n  nonHotswappableResource,\n} from './common';\nimport { isHotswappableEcsServiceChange } from './ecs-services';\nimport { isHotswappableLambdaFunctionChange } from './lambda-functions';\nimport {\n  skipChangeForS3DeployCustomResourcePolicy,\n  isHotswappableS3BucketDeploymentChange,\n} from './s3-bucket-deployments';\nimport { isHotswappableStateMachineChange } from './stepfunctions-state-machines';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\nimport type { SuccessfulDeployStackResult } from '../deployments';\nimport { IO, SPAN } from '../io/private';\nimport type { IMessageSpan, IoHelper } from '../io/private';\nimport { Mode } from '../plugin';\n\n// Must use a require() otherwise esbuild complains about calling a namespace\n// eslint-disable-next-line @typescript-eslint/no-require-imports,@typescript-eslint/consistent-type-imports\nconst pLimit: typeof import('p-limit') = require('p-limit');\n\ntype HotswapDetector = (\n  logicalId: string,\n  change: ResourceChange,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n) => Promise<HotswapChange[]>;\n\ntype HotswapMode = 'hotswap-only' | 'fall-back';\n\nconst RESOURCE_DETECTORS: { [key: string]: HotswapDetector } = {\n  // Lambda\n  'AWS::Lambda::Function': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Version': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Alias': isHotswappableLambdaFunctionChange,\n\n  // AppSync\n  'AWS::AppSync::Resolver': isHotswappableAppSyncChange,\n  'AWS::AppSync::FunctionConfiguration': isHotswappableAppSyncChange,\n  'AWS::AppSync::GraphQLSchema': isHotswappableAppSyncChange,\n  'AWS::AppSync::ApiKey': isHotswappableAppSyncChange,\n\n  'AWS::BedrockAgentCore::Runtime': isHotswappableBedrockAgentCoreRuntimeChange,\n  'AWS::ECS::TaskDefinition': isHotswappableEcsServiceChange,\n  'AWS::CodeBuild::Project': isHotswappableCodeBuildProjectChange,\n  'AWS::StepFunctions::StateMachine': isHotswappableStateMachineChange,\n  'Custom::CDKBucketDeployment': isHotswappableS3BucketDeploymentChange,\n  'AWS::IAM::Policy': async (\n    logicalId: string,\n    change: ResourceChange,\n    evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  ): Promise<HotswapChange[]> => {\n    // If the policy is for a S3BucketDeploymentChange, we can ignore the change\n    if (await skipChangeForS3DeployCustomResourcePolicy(logicalId, change, evaluateCfnTemplate)) {\n      return [];\n    }\n\n    return [nonHotswappableResource(change)];\n  },\n\n  'AWS::CDK::Metadata': async () => [],\n};\n\n/**\n * Perform a hotswap deployment, short-circuiting CloudFormation if possible.\n * If it's not possible to short-circuit the deployment\n * (because the CDK Stack contains changes that cannot be deployed without CloudFormation),\n * returns `undefined`.\n */\nexport async function tryHotswapDeployment(\n  sdkProvider: SdkProvider,\n  ioHelper: IoHelper,\n  assetParams: { [key: string]: string },\n  cloudFormationStack: CloudFormationStack,\n  stackArtifact: cxapi.CloudFormationStackArtifact,\n  hotswapMode: HotswapMode,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<SuccessfulDeployStackResult | undefined> {\n  const hotswapSpan = await ioHelper.span(SPAN.HOTSWAP).begin({\n    stack: stackArtifact,\n    mode: hotswapMode,\n  });\n\n  const result = await hotswapDeployment(\n    sdkProvider,\n    hotswapSpan,\n    assetParams,\n    stackArtifact,\n    hotswapMode,\n    hotswapPropertyOverrides,\n  );\n\n  await hotswapSpan.end(result);\n\n  // This is a hotswap error that was caught during the hotswapDeployment function\n  // It is thrown after the hotswap span ends so it can be reported to telemetry\n  if (result.error) {\n    throw result.error;\n  }\n\n  if (result?.hotswapped === true) {\n    return {\n      type: 'did-deploy-stack',\n      noOp: result.hotswappableChanges.length === 0,\n      stackArn: cloudFormationStack.stackId,\n      outputs: cloudFormationStack.outputs,\n    };\n  }\n\n  return undefined;\n}\n\n/**\n * Perform a hotswap deployment, short-circuiting CloudFormation if possible.\n * Returns information about the attempted hotswap deployment\n */\nasync function hotswapDeployment(\n  sdkProvider: SdkProvider,\n  ioSpan: IMessageSpan<any>,\n  assetParams: { [key: string]: string },\n  stack: cxapi.CloudFormationStackArtifact,\n  hotswapMode: HotswapMode,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<Omit<HotswapResult, 'duration'>> {\n  // resolve the environment, so we can substitute things like AWS::Region in CFN expressions\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stack.environment);\n  // create a new SDK using the CLI credentials, because the default one will not work for new-style synthesis -\n  // it assumes the bootstrap deploy Role, which doesn't have permissions to update Lambda functions\n  const sdk = (await sdkProvider.forEnvironment(resolvedEnv, Mode.ForWriting)).sdk;\n\n  const currentTemplate = await loadCurrentTemplateWithNestedStacks(stack, sdk);\n\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    stackArtifact: stack,\n    parameters: assetParams,\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    sdk,\n    nestedStacks: currentTemplate.nestedStacks,\n  });\n\n  const stackChanges = cfn_diff.fullDiff(currentTemplate.deployedRootTemplate, stack.template);\n  const { hotswappable, nonHotswappable } = await classifyResourceChanges(\n    stackChanges,\n    evaluateCfnTemplate,\n    sdk,\n    currentTemplate.nestedStacks, hotswapPropertyOverrides,\n  );\n\n  await logRejectedChanges(ioSpan, nonHotswappable, hotswapMode);\n\n  const hotswappableChanges = hotswappable.map(o => o.change);\n  const nonHotswappableChanges = nonHotswappable.map(n => n.change);\n\n  await ioSpan.notify(IO.CDK_TOOLKIT_I5401.msg('Hotswap plan created', {\n    stack,\n    mode: hotswapMode,\n    hotswappableChanges,\n    nonHotswappableChanges,\n  }));\n\n  // preserve classic hotswap behavior\n  if (hotswapMode === 'fall-back') {\n    if (nonHotswappableChanges.length > 0) {\n      return {\n        stack,\n        mode: hotswapMode,\n        hotswapped: false,\n        hotswappableChanges,\n        nonHotswappableChanges,\n      };\n    }\n  }\n\n  // apply the short-circuitable changes\n  let error: Error | undefined;\n  try {\n    await applyAllHotswapOperations(sdk, ioSpan, hotswappable);\n  } catch (e: any) {\n    error = e;\n  }\n\n  return {\n    stack,\n    mode: hotswapMode,\n    hotswapped: !error,\n    hotswappableChanges,\n    nonHotswappableChanges,\n    error,\n  };\n}\n\ninterface ClassifiedChanges {\n  hotswappable: HotswapOperation[];\n  nonHotswappable: RejectedChange[];\n}\n\n/**\n * Classifies all changes to all resources as either hotswappable or not.\n * Metadata changes are excluded from the list of (non)hotswappable resources.\n */\nasync function classifyResourceChanges(\n  stackChanges: cfn_diff.TemplateDiff,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: SDK,\n  nestedStackNames: { [nestedStackName: string]: NestedStackTemplates },\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<ClassifiedChanges> {\n  const resourceDifferences = getStackResourceDifferences(stackChanges);\n\n  const promises: Array<() => Promise<HotswapChange[]>> = [];\n  const hotswappableResources = new Array<HotswapOperation>();\n  const nonHotswappableResources = new Array<RejectedChange>();\n  for (const logicalId of Object.keys(stackChanges.outputs.changes)) {\n    nonHotswappableResources.push({\n      hotswappable: false,\n      change: {\n        reason: NonHotswappableReason.OUTPUT,\n        description: 'output was changed',\n        subject: {\n          type: 'Output',\n          logicalId,\n          metadata: evaluateCfnTemplate.metadataFor(logicalId),\n        },\n      },\n    });\n  }\n  // gather the results of the detector functions\n  for (const [logicalId, change] of Object.entries(resourceDifferences)) {\n    if (change.newValue?.Type === 'AWS::CloudFormation::Stack' && change.oldValue?.Type === 'AWS::CloudFormation::Stack') {\n      const nestedHotswappableResources = await findNestedHotswappableChanges(\n        logicalId,\n        change,\n        nestedStackNames,\n        evaluateCfnTemplate,\n        sdk,\n        hotswapPropertyOverrides,\n      );\n      hotswappableResources.push(...nestedHotswappableResources.hotswappable);\n      nonHotswappableResources.push(...nestedHotswappableResources.nonHotswappable);\n\n      continue;\n    }\n\n    const hotswappableChangeCandidate = isCandidateForHotswapping(logicalId, change, evaluateCfnTemplate);\n    // we don't need to run this through the detector functions, we can already judge this\n    if ('hotswappable' in hotswappableChangeCandidate) {\n      if (!hotswappableChangeCandidate.hotswappable) {\n        nonHotswappableResources.push(hotswappableChangeCandidate);\n      }\n\n      continue;\n    }\n\n    const resourceType: string = hotswappableChangeCandidate.newValue.Type;\n    if (resourceType in RESOURCE_DETECTORS) {\n      // run detector functions lazily to prevent unhandled promise rejections\n      promises.push(() =>\n        RESOURCE_DETECTORS[resourceType](logicalId, hotswappableChangeCandidate, evaluateCfnTemplate, hotswapPropertyOverrides),\n      );\n    } else {\n      nonHotswappableResources.push(nonHotswappableResource(hotswappableChangeCandidate));\n    }\n  }\n\n  // resolve all detector results\n  const changesDetectionResults: Array<HotswapChange[]> = [];\n  for (const detectorResultPromises of promises) {\n    // Constant set of promises per resource\n    // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n    const hotswapDetectionResults = await Promise.all(await detectorResultPromises());\n    changesDetectionResults.push(hotswapDetectionResults);\n  }\n\n  for (const resourceDetectionResults of changesDetectionResults) {\n    for (const propertyResult of resourceDetectionResults) {\n      propertyResult.hotswappable\n        ? hotswappableResources.push(propertyResult)\n        : nonHotswappableResources.push(propertyResult);\n    }\n  }\n\n  return {\n    hotswappable: hotswappableResources,\n    nonHotswappable: nonHotswappableResources,\n  };\n}\n\n/**\n * Returns all changes to resources in the given Stack.\n *\n * @param stackChanges - the collection of all changes to a given Stack\n */\nfunction getStackResourceDifferences(stackChanges: cfn_diff.TemplateDiff): {\n  [logicalId: string]: cfn_diff.ResourceDifference;\n} {\n  // we need to collapse logical ID rename changes into one change,\n  // as they are represented in stackChanges as a pair of two changes: one addition and one removal\n  const allResourceChanges: { [logId: string]: cfn_diff.ResourceDifference } = stackChanges.resources.changes;\n  const allRemovalChanges = filterDict(allResourceChanges, (resChange) => resChange.isRemoval);\n  const allNonRemovalChanges = filterDict(allResourceChanges, (resChange) => !resChange.isRemoval);\n  for (const [logId, nonRemovalChange] of Object.entries(allNonRemovalChanges)) {\n    if (nonRemovalChange.isAddition) {\n      const addChange = nonRemovalChange;\n      // search for an identical removal change\n      const identicalRemovalChange = Object.entries(allRemovalChanges).find(([_, remChange]) => {\n        return changesAreForSameResource(remChange, addChange);\n      });\n      // if we found one, then this means this is a rename change\n      if (identicalRemovalChange) {\n        const [removedLogId, removedResourceChange] = identicalRemovalChange;\n        allNonRemovalChanges[logId] = makeRenameDifference(removedResourceChange, addChange);\n        // delete the removal change that forms the rename pair\n        delete allRemovalChanges[removedLogId];\n      }\n    }\n  }\n  // the final result are all of the remaining removal changes,\n  // plus all of the non-removal changes\n  // (we saved the rename changes in that object already)\n  return {\n    ...allRemovalChanges,\n    ...allNonRemovalChanges,\n  };\n}\n\n/** Filters an object with string keys based on whether the callback returns 'true' for the given value in the object. */\nfunction filterDict<T>(dict: { [key: string]: T }, func: (t: T) => boolean): { [key: string]: T } {\n  return Object.entries(dict).reduce(\n    (acc, [key, t]) => {\n      if (func(t)) {\n        acc[key] = t;\n      }\n      return acc;\n    },\n    {} as { [key: string]: T },\n  );\n}\n\n/** Finds any hotswappable changes in all nested stacks. */\nasync function findNestedHotswappableChanges(\n  logicalId: string,\n  change: cfn_diff.ResourceDifference,\n  nestedStackTemplates: { [nestedStackName: string]: NestedStackTemplates },\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: SDK,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<ClassifiedChanges> {\n  const nestedStack = nestedStackTemplates[logicalId];\n  if (!nestedStack.physicalName) {\n    return {\n      hotswappable: [],\n      nonHotswappable: [\n        {\n          hotswappable: false,\n          change: {\n            reason: NonHotswappableReason.NESTED_STACK_CREATION,\n            description: 'newly created nested stacks cannot be hotswapped',\n            subject: {\n              type: 'Resource',\n              logicalId,\n              resourceType: 'AWS::CloudFormation::Stack',\n              metadata: evaluateCfnTemplate.metadataFor(logicalId),\n            },\n          },\n        },\n      ],\n    };\n  }\n\n  const evaluateNestedCfnTemplate = await evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(\n    nestedStack.physicalName,\n    nestedStack.generatedTemplate,\n    change.newValue?.Properties?.Parameters,\n  );\n\n  const nestedDiff = cfn_diff.fullDiff(\n    nestedStackTemplates[logicalId].deployedTemplate,\n    nestedStackTemplates[logicalId].generatedTemplate,\n  );\n\n  return classifyResourceChanges(\n    nestedDiff,\n    evaluateNestedCfnTemplate,\n    sdk,\n    nestedStackTemplates[logicalId].nestedStackTemplates,\n    hotswapPropertyOverrides,\n  );\n}\n\n/** Returns 'true' if a pair of changes is for the same resource. */\nfunction changesAreForSameResource(\n  oldChange: cfn_diff.ResourceDifference,\n  newChange: cfn_diff.ResourceDifference,\n): boolean {\n  return (\n    oldChange.oldResourceType === newChange.newResourceType &&\n    // this isn't great, but I don't want to bring in something like underscore just for this comparison\n    JSON.stringify(oldChange.oldProperties) === JSON.stringify(newChange.newProperties)\n  );\n}\n\nfunction makeRenameDifference(\n  remChange: cfn_diff.ResourceDifference,\n  addChange: cfn_diff.ResourceDifference,\n): cfn_diff.ResourceDifference {\n  return new cfn_diff.ResourceDifference(\n    // we have to fill in the old value, because otherwise this will be classified as a non-hotswappable change\n    remChange.oldValue,\n    addChange.newValue,\n    {\n      resourceType: {\n        oldType: remChange.oldResourceType,\n        newType: addChange.newResourceType,\n      },\n      propertyDiffs: (addChange as any).propertyDiffs,\n      otherDiffs: (addChange as any).otherDiffs,\n    },\n  );\n}\n\n/**\n * Returns a `HotswappableChangeCandidate` if the change is hotswappable\n * Returns an empty `HotswappableChange` if the change is to CDK::Metadata\n * Returns a `NonHotswappableChange` if the change is not hotswappable\n */\nfunction isCandidateForHotswapping(\n  logicalId: string,\n  change: cfn_diff.ResourceDifference,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): RejectedChange | ResourceChange {\n  // a resource has been removed OR a resource has been added; we can't short-circuit that change\n  if (!change.oldValue) {\n    return {\n      hotswappable: false,\n      change: {\n        reason: NonHotswappableReason.RESOURCE_CREATION,\n        description: `resource '${logicalId}' was created by this deployment`,\n        subject: {\n          type: 'Resource',\n          logicalId,\n          resourceType: change.newValue!.Type,\n          metadata: evaluateCfnTemplate.metadataFor(logicalId),\n        },\n      },\n    };\n  } else if (!change.newValue) {\n    return {\n      hotswappable: false,\n      logicalId,\n      change: {\n        reason: NonHotswappableReason.RESOURCE_DELETION,\n        description: `resource '${logicalId}' was destroyed by this deployment`,\n        subject: {\n          type: 'Resource',\n          logicalId,\n          resourceType: change.oldValue.Type,\n          metadata: evaluateCfnTemplate.metadataFor(logicalId),\n        },\n      },\n    };\n  }\n\n  // a resource has had its type changed\n  if (change.newValue.Type !== change.oldValue.Type) {\n    return {\n      hotswappable: false,\n      change: {\n        reason: NonHotswappableReason.RESOURCE_TYPE_CHANGED,\n        description: `resource '${logicalId}' had its type changed from '${change.oldValue?.Type}' to '${change.newValue?.Type}'`,\n        subject: {\n          type: 'Resource',\n          logicalId,\n          resourceType: change.newValue.Type,\n          metadata: evaluateCfnTemplate.metadataFor(logicalId),\n        },\n      },\n    };\n  }\n\n  return {\n    logicalId,\n    oldValue: change.oldValue,\n    newValue: change.newValue,\n    propertyUpdates: change.propertyUpdates,\n    metadata: evaluateCfnTemplate.metadataFor(logicalId),\n  };\n}\n\nasync function applyAllHotswapOperations(sdk: SDK, ioSpan: IMessageSpan<any>, hotswappableChanges: HotswapOperation[]): Promise<void[]> {\n  if (hotswappableChanges.length === 0) {\n    return Promise.resolve([]);\n  }\n\n  await ioSpan.defaults.info(`\\n${ICON} hotswapping resources:`);\n  const limit = pLimit(10);\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  return Promise.all(hotswappableChanges.map(hotswapOperation => limit(() => {\n    return applyHotswapOperation(sdk, ioSpan, hotswapOperation);\n  })));\n}\n\nasync function applyHotswapOperation(sdk: SDK, ioSpan: IMessageSpan<any>, hotswapOperation: HotswapOperation): Promise<void> {\n  // note the type of service that was successfully hotswapped in the User-Agent\n  const customUserAgent = `cdk-hotswap/success-${hotswapOperation.service}`;\n  sdk.appendCustomUserAgent(customUserAgent);\n  const resourceText = (r: AffectedResource) => r.description ?? `${r.resourceType} '${r.physicalName ?? r.logicalId}'`;\n\n  await ioSpan.notify(IO.CDK_TOOLKIT_I5402.msg(\n    hotswapOperation.change.resources.map(r => format(`   ${ICON} %s`, chalk.bold(resourceText(r)))).join('\\n'),\n    hotswapOperation.change,\n  ));\n\n  // if the SDK call fails, an error will be thrown by the SDK\n  // and will prevent the green 'hotswapped!' text from being displayed\n  try {\n    await hotswapOperation.apply(sdk);\n  } catch (e: any) {\n    if (e.name === 'TimeoutError' || e.name === 'AbortError') {\n      const result: WaiterResult = JSON.parse(formatErrorMessage(e));\n      const error = new ToolkitError(formatWaiterErrorResult(result));\n      error.name = e.name;\n      throw error;\n    }\n    throw e;\n  }\n\n  await ioSpan.notify(IO.CDK_TOOLKIT_I5403.msg(\n    hotswapOperation.change.resources.map(r => format(`   ${ICON} %s %s`, chalk.bold(resourceText(r)), chalk.green('hotswapped!'))).join('\\n'),\n    hotswapOperation.change,\n  ));\n\n  sdk.removeCustomUserAgent(customUserAgent);\n}\n\nfunction formatWaiterErrorResult(result: WaiterResult) {\n  const main = [\n    `Resource is not in the expected state due to waiter status: ${result.state}`,\n    result.reason ? `${result.reason}.` : '',\n  ].join('. ');\n\n  if (result.observedResponses != null) {\n    const observedResponses = Object\n      .entries(result.observedResponses)\n      .map(([msg, count]) => `  - ${msg} (${count})`)\n      .join('\\n');\n\n    return `${main} Observed responses:\\n${observedResponses}`;\n  }\n\n  return main;\n}\n\nasync function logRejectedChanges(\n  ioSpan: IMessageSpan<any>,\n  rejectedChanges: RejectedChange[],\n  hotswapMode: HotswapMode,\n): Promise<void> {\n  if (rejectedChanges.length === 0) {\n    return;\n  }\n  /**\n   * EKS Services can have a task definition that doesn't refer to the task definition being updated.\n   * We have to log this as a non-hotswappable change to the task definition, but when we do,\n   * we wind up hotswapping the task definition and logging it as a non-hotswappable change.\n   *\n   * This logic prevents us from logging that change as non-hotswappable when we hotswap it.\n   */\n  if (hotswapMode === 'hotswap-only') {\n    rejectedChanges = rejectedChanges.filter((change) => change.hotswapOnlyVisible === true);\n\n    if (rejectedChanges.length === 0) {\n      return;\n    }\n  }\n\n  const messages = ['']; // start with empty line\n\n  if (hotswapMode === 'hotswap-only') {\n    messages.push(format('%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found. To reconcile these using CloudFormation, specify --hotswap-fallback')));\n  } else {\n    messages.push(format('%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found:')));\n  }\n\n  for (const { change } of rejectedChanges) {\n    messages.push('    ' + nonHotswappableChangeMessage(change));\n  }\n  messages.push(''); // newline\n\n  await ioSpan.defaults.info(messages.join('\\n'));\n}\n\n/**\n * Formats a NonHotswappableChange\n */\nfunction nonHotswappableChangeMessage(change: NonHotswappableChange): string {\n  const subject = change.subject;\n  const reason = change.description ?? change.reason;\n\n  switch (subject.type) {\n    case 'Output':\n      return format(\n        'output: %s, reason: %s',\n        chalk.bold(subject.logicalId),\n        chalk.red(reason),\n      );\n    case 'Resource':\n      return nonHotswappableResourceMessage(subject, reason);\n  }\n}\n\n/**\n * Formats a non-hotswappable resource subject\n */\nfunction nonHotswappableResourceMessage(subject: ResourceSubject, reason: string): string {\n  if (subject.rejectedProperties?.length) {\n    return format(\n      'resource: %s, type: %s, rejected changes: %s, reason: %s',\n      chalk.bold(subject.logicalId),\n      chalk.bold(subject.resourceType),\n      chalk.bold(subject.rejectedProperties),\n      chalk.red(reason),\n    );\n  }\n\n  return format(\n    'resource: %s, type: %s, reason: %s',\n    chalk.bold(subject.logicalId),\n    chalk.bold(subject.resourceType),\n    chalk.red(reason),\n  );\n}\n"]}
|
|
@@ -3,8 +3,12 @@ import type * as make from './message-maker';
|
|
|
3
3
|
import type { Duration } from '../../../payloads/types';
|
|
4
4
|
import type { IActionAwareIoHost } from '../io-host';
|
|
5
5
|
import type { IoDefaultMessages } from './io-default-messages';
|
|
6
|
+
/**
|
|
7
|
+
* These data fields are automatically added by ending a span
|
|
8
|
+
*/
|
|
6
9
|
export interface SpanEnd {
|
|
7
10
|
readonly duration: number;
|
|
11
|
+
readonly counters?: Record<string, number>;
|
|
8
12
|
}
|
|
9
13
|
/**
|
|
10
14
|
* Describes a specific span
|
|
@@ -17,26 +21,36 @@ export interface SpanDefinition<S extends object, E extends SpanEnd> {
|
|
|
17
21
|
readonly start: make.IoMessageMaker<S>;
|
|
18
22
|
readonly end: make.IoMessageMaker<E>;
|
|
19
23
|
}
|
|
24
|
+
/**
|
|
25
|
+
* Arguments to the span.end() function
|
|
26
|
+
*
|
|
27
|
+
* `SpanEndArguments<T>` are the fields that a user still needs to supply, it
|
|
28
|
+
* fields in the type `T` that aren't also in `SpanEnd`. `SpanEnd` represents
|
|
29
|
+
* fields that are automatically added by the underlying `end` function.
|
|
30
|
+
*
|
|
31
|
+
* Fields that are already in `SpanEnd` are still rendered as optionals, so you
|
|
32
|
+
* can override them (but you don't have to).
|
|
33
|
+
*
|
|
34
|
+
* - Does the following: fields that are shared between `T` and `SpanEnd` are
|
|
35
|
+
* made optional, and the rest of the keys of `T` are required.
|
|
36
|
+
*
|
|
37
|
+
* - If `T` is fully subsumed by the `SpanEnd` type, then an object type with
|
|
38
|
+
* all fields optional, OR 'void' so you can avoid passing an argument at all.
|
|
39
|
+
*/
|
|
40
|
+
type SpanEndArguments<T> = keyof T extends keyof SpanEnd ? (Pick<Partial<SpanEnd>, keyof T & keyof SpanEnd> | void) : Optional<T, keyof T & keyof SpanEnd>;
|
|
20
41
|
/**
|
|
21
42
|
* Used in conditional types to check if a type (e.g. after omitting fields) is an empty object
|
|
22
43
|
* This is needed because counter-intuitive neither `object` nor `{}` represent that.
|
|
23
44
|
*/
|
|
24
|
-
type EmptyObject =
|
|
25
|
-
[index: string | number | symbol]: never;
|
|
26
|
-
};
|
|
45
|
+
type EmptyObject = Record<string, never>;
|
|
27
46
|
/**
|
|
28
47
|
* Helper type to force a parameter to be not present of the computed type is an empty object
|
|
29
48
|
*/
|
|
30
49
|
type VoidWhenEmpty<T> = T extends EmptyObject ? void : T;
|
|
31
|
-
/**
|
|
32
|
-
* Helper type to force a parameter to be an empty object if the computed type is an empty object
|
|
33
|
-
* This is weird, but some computed types (e.g. using `Omit`) don't end up enforcing this.
|
|
34
|
-
*/
|
|
35
|
-
type ForceEmpty<T> = T extends EmptyObject ? EmptyObject : T;
|
|
36
50
|
/**
|
|
37
51
|
* Make some properties optional
|
|
38
52
|
*/
|
|
39
|
-
type Optional<T, K extends keyof T> =
|
|
53
|
+
type Optional<T, K extends keyof T> = Omit<T, K> & Pick<Partial<T>, K>;
|
|
40
54
|
/**
|
|
41
55
|
* Ending the span returns the observed duration
|
|
42
56
|
*/
|
|
@@ -68,15 +82,31 @@ export interface IMessageSpan<E extends SpanEnd> extends IActionAwareIoHost {
|
|
|
68
82
|
/**
|
|
69
83
|
* End the span with a payload
|
|
70
84
|
*/
|
|
71
|
-
end(payload:
|
|
85
|
+
end(payload: SpanEndArguments<E>): Promise<ElapsedTime>;
|
|
72
86
|
/**
|
|
73
|
-
* End the span with a payload
|
|
87
|
+
* End the span with a message and payload
|
|
74
88
|
*/
|
|
75
|
-
end(payload:
|
|
89
|
+
end(message: string, payload: SpanEndArguments<E>): Promise<ElapsedTime>;
|
|
76
90
|
/**
|
|
77
|
-
*
|
|
91
|
+
* Increment a counter
|
|
92
|
+
*/
|
|
93
|
+
incCounter(name: string, delta?: number): void;
|
|
94
|
+
/**
|
|
95
|
+
* Return a new timer object
|
|
96
|
+
*
|
|
97
|
+
* It will be added into the span data when it's stopped. All open timers are
|
|
98
|
+
* automatically stopped when the span is ended.
|
|
99
|
+
*
|
|
100
|
+
* Timers are ultimately added to the `counters` array with `<name>_ms` and
|
|
101
|
+
* `<name>_cnt` keys.
|
|
78
102
|
*/
|
|
79
|
-
|
|
103
|
+
startTimer(name: string): ITimer;
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* A timer to time an operation in a span.
|
|
107
|
+
*/
|
|
108
|
+
export interface ITimer {
|
|
109
|
+
stop(): void;
|
|
80
110
|
}
|
|
81
111
|
/**
|
|
82
112
|
* Helper class to make spans around blocks of work
|
|
@@ -37,6 +37,8 @@ class MessageSpan {
|
|
|
37
37
|
spanId;
|
|
38
38
|
startTime;
|
|
39
39
|
timingMsgTemplate;
|
|
40
|
+
counters = {};
|
|
41
|
+
openTimers = new Set();
|
|
40
42
|
constructor(ioHelper, definition, makeHelper) {
|
|
41
43
|
this.definition = definition;
|
|
42
44
|
this.ioHelper = ioHelper;
|
|
@@ -64,18 +66,38 @@ class MessageSpan {
|
|
|
64
66
|
}
|
|
65
67
|
async end(x, y) {
|
|
66
68
|
const duration = this.time();
|
|
69
|
+
for (const t of this.openTimers) {
|
|
70
|
+
t.stop();
|
|
71
|
+
}
|
|
72
|
+
this.openTimers.clear();
|
|
67
73
|
const endInput = parseArgs(x, y);
|
|
68
74
|
const endMsg = endInput.message ?? util.format(this.timingMsgTemplate, this.definition.name, duration.asSec);
|
|
69
75
|
const endPayload = endInput.payload;
|
|
70
76
|
await this.notify(this.definition.end.msg(endMsg, {
|
|
71
77
|
duration: duration.asMs,
|
|
78
|
+
...(Object.keys(this.counters).length > 0 ? { counters: this.counters } : {}),
|
|
72
79
|
...endPayload,
|
|
73
80
|
}));
|
|
74
81
|
return duration;
|
|
75
82
|
}
|
|
83
|
+
incCounter(name, delta = 1) {
|
|
84
|
+
this.counters[name] = (this.counters[name] ?? 0) + delta;
|
|
85
|
+
}
|
|
76
86
|
async requestResponse(msg) {
|
|
77
87
|
return this.ioHelper.requestResponse(withSpanId(this.spanId, msg));
|
|
78
88
|
}
|
|
89
|
+
startTimer(name) {
|
|
90
|
+
const start = Date.now();
|
|
91
|
+
const t = {
|
|
92
|
+
stop: () => {
|
|
93
|
+
this.openTimers.delete(t);
|
|
94
|
+
this.incCounter(`${name}_ms`, Math.floor(Date.now() - start) / 1000);
|
|
95
|
+
this.incCounter(`${name}_cnt`, 1);
|
|
96
|
+
},
|
|
97
|
+
};
|
|
98
|
+
this.openTimers.add(t);
|
|
99
|
+
return t;
|
|
100
|
+
}
|
|
79
101
|
time() {
|
|
80
102
|
const elapsedTime = new Date().getTime() - this.startTime;
|
|
81
103
|
return {
|
|
@@ -102,4 +124,4 @@ function withSpanId(span, message) {
|
|
|
102
124
|
span,
|
|
103
125
|
};
|
|
104
126
|
}
|
|
105
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"span.js","sourceRoot":"","sources":["span.ts"],"names":[],"mappings":"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","sourcesContent":["import * as util from 'node:util';\nimport * as uuid from 'uuid';\nimport type { ActionLessMessage, ActionLessRequest, IoHelper } from './io-helper';\nimport type * as make from './message-maker';\nimport type { Duration } from '../../../payloads/types';\nimport { formatTime } from '../../../util';\nimport type { IActionAwareIoHost } from '../io-host';\nimport type { IoDefaultMessages } from './io-default-messages';\n\nexport interface SpanEnd {\n  readonly duration: number;\n}\n\n/**\n * Describes a specific span\n *\n * A span definition is a pair of `IoMessageMaker`s to create a start and end message of the span respectively.\n * It also has a display name, that is used for auto-generated message text when they are not provided.\n */\nexport interface SpanDefinition<S extends object, E extends SpanEnd> {\n  readonly name: string;\n  readonly start: make.IoMessageMaker<S>;\n  readonly end: make.IoMessageMaker<E>;\n}\n\n/**\n * Used in conditional types to check if a type (e.g. after omitting fields) is an empty object\n * This is needed because counter-intuitive neither `object` nor `{}` represent that.\n */\ntype EmptyObject = {\n  [index: string | number | symbol]: never;\n};\n\n/**\n * Helper type to force a parameter to be not present of the computed type is an empty object\n */\ntype VoidWhenEmpty<T> = T extends EmptyObject ? void : T;\n\n/**\n * Helper type to force a parameter to be an empty object if the computed type is an empty object\n * This is weird, but some computed types (e.g. using `Omit`) don't end up enforcing this.\n */\ntype ForceEmpty<T> = T extends EmptyObject ? EmptyObject : T;\n\n/**\n * Make some properties optional\n */\ntype Optional<T, K extends keyof T> = Pick<Partial<T>, K> & Omit<T, K>;\n\n/**\n * Ending the span returns the observed duration\n */\nexport interface ElapsedTime {\n  readonly asMs: number;\n  readonly asSec: number;\n}\n\n/**\n * A message span that can be ended and read times from\n */\nexport interface IMessageSpan<E extends SpanEnd> extends IActionAwareIoHost {\n  /**\n   * An IoHelper wrapped around the span.\n   */\n  readonly asHelper: IoHelper;\n  /**\n   * An IoDefaultMessages wrapped around the span.\n   */\n  readonly defaults: IoDefaultMessages;\n  /**\n   * Get the time elapsed since the start\n   */\n  elapsedTime(): Promise<ElapsedTime>;\n  /**\n   * Sends a simple, generic message with the current timing\n   * For more complex intermediate messages, get the `elapsedTime` and use `notify`\n   */\n  timing(maker: make.IoMessageMaker<Duration>, message?: string): Promise<ElapsedTime>;\n  /**\n   * End the span with a payload\n   */\n  end(payload: VoidWhenEmpty<Omit<E, keyof SpanEnd>>): Promise<ElapsedTime>;\n  /**\n   * End the span with a payload, overwriting\n   */\n  end(payload: VoidWhenEmpty<Optional<E, keyof SpanEnd>>): Promise<ElapsedTime>;\n  /**\n   * End the span with a message and payload\n   */\n  end(message: string, payload: ForceEmpty<Optional<E, keyof SpanEnd>>): Promise<ElapsedTime>;\n}\n\n/**\n * Helper class to make spans around blocks of work\n *\n * Blocks are enclosed by a start and end message.\n * All messages of the span share a unique id.\n * The end message contains the time passed between start and end.\n */\nexport class SpanMaker<S extends object, E extends SpanEnd> {\n  private readonly definition: SpanDefinition<S, E>;\n  private readonly ioHelper: IoHelper;\n  private makeHelper: (ioHost: IActionAwareIoHost) => IoHelper;\n\n  public constructor(ioHelper: IoHelper, definition: SpanDefinition<S, E>, makeHelper: (ioHost: IActionAwareIoHost) => IoHelper) {\n    this.definition = definition;\n    this.ioHelper = ioHelper;\n    this.makeHelper = makeHelper;\n  }\n\n  /**\n   * Starts the span and initially notifies the IoHost\n   * @returns a message span\n   */\n  public async begin(payload: VoidWhenEmpty<S>): Promise<IMessageSpan<E>>;\n  public async begin(message: string, payload: S): Promise<IMessageSpan<E>>;\n  public async begin(a: any, b?: S): Promise<IMessageSpan<E>> {\n    const span = new MessageSpan(this.ioHelper, this.definition, this.makeHelper);\n    const startInput = parseArgs<S>(a, b);\n    const startMsg = startInput.message ?? `Starting ${this.definition.name} ...`;\n    const startPayload = startInput.payload;\n    await span.notify(this.definition.start.msg(startMsg, startPayload));\n\n    return span;\n  }\n}\n\nclass MessageSpan<S extends object, E extends SpanEnd> implements IMessageSpan<E> {\n  public readonly asHelper: IoHelper;\n\n  private readonly definition: SpanDefinition<S, E>;\n  private readonly ioHelper: IoHelper;\n  private readonly spanId: string;\n  private readonly startTime: number;\n  private readonly timingMsgTemplate: string;\n\n  public constructor(ioHelper: IoHelper, definition: SpanDefinition<S, E>, makeHelper: (ioHost: IActionAwareIoHost) => IoHelper) {\n    this.definition = definition;\n    this.ioHelper = ioHelper;\n    this.spanId = uuid.v4();\n    this.startTime = new Date().getTime();\n    this.timingMsgTemplate = '\\n✨  %s time: %ds\\n';\n    this.asHelper = makeHelper(this);\n  }\n\n  public get defaults(): IoDefaultMessages {\n    return this.asHelper.defaults;\n  }\n\n  public async elapsedTime(): Promise<ElapsedTime> {\n    return this.time();\n  }\n  public async timing(maker: make.IoMessageMaker<Duration>, message?: string): Promise<ElapsedTime> {\n    const duration = this.time();\n    const timingMsg = message ? message : util.format(this.timingMsgTemplate, this.definition.name, duration.asSec);\n    await this.notify(maker.msg(timingMsg, {\n      duration: duration.asMs,\n    }));\n    return duration;\n  }\n  public async notify(msg: ActionLessMessage<unknown>): Promise<void> {\n    return this.ioHelper.notify(withSpanId(this.spanId, msg));\n  }\n  public async end(x: any, y?: ForceEmpty<Optional<E, keyof SpanEnd>>): Promise<ElapsedTime> {\n    const duration = this.time();\n\n    const endInput = parseArgs<ForceEmpty<Optional<E, keyof SpanEnd>>>(x, y);\n    const endMsg = endInput.message ?? util.format(this.timingMsgTemplate, this.definition.name, duration.asSec);\n    const endPayload = endInput.payload;\n\n    await this.notify(this.definition.end.msg(\n      endMsg, {\n        duration: duration.asMs,\n        ...endPayload,\n      } as E));\n\n    return duration;\n  }\n\n  public async requestResponse<T>(msg: ActionLessRequest<unknown, T>): Promise<T> {\n    return this.ioHelper.requestResponse(withSpanId(this.spanId, msg));\n  }\n\n  private time() {\n    const elapsedTime = new Date().getTime() - this.startTime;\n    return {\n      asMs: elapsedTime,\n      asSec: formatTime(elapsedTime),\n    };\n  }\n}\n\nfunction parseArgs<S extends object>(first: any, second?: S): { message: string | undefined; payload: S } {\n  const firstIsMessage = typeof first === 'string';\n\n  // When the first argument is a string or we have a second argument, then the first arg is the message\n  const message = (firstIsMessage || second) ? first : undefined;\n\n  // When the first argument is a string or we have a second argument,\n  // then the second arg is the payload, otherwise the first arg is the payload\n  const payload = (firstIsMessage || second) ? second : first;\n\n  return {\n    message,\n    payload,\n  };\n}\n\nfunction withSpanId<T extends object>(span: string, message: T): T & { span: string } {\n  return {\n    ...message,\n    span,\n  };\n}\n"]}
|
|
127
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"span.js","sourceRoot":"","sources":["span.ts"],"names":[],"mappings":"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","sourcesContent":["import * as util from 'node:util';\nimport * as uuid from 'uuid';\nimport type { ActionLessMessage, ActionLessRequest, IoHelper } from './io-helper';\nimport type * as make from './message-maker';\nimport type { Duration } from '../../../payloads/types';\nimport { formatTime } from '../../../util';\nimport type { IActionAwareIoHost } from '../io-host';\nimport type { IoDefaultMessages } from './io-default-messages';\n\n/**\n * These data fields are automatically added by ending a span\n */\nexport interface SpanEnd {\n  readonly duration: number;\n  readonly counters?: Record<string, number>;\n}\n\n/**\n * Describes a specific span\n *\n * A span definition is a pair of `IoMessageMaker`s to create a start and end message of the span respectively.\n * It also has a display name, that is used for auto-generated message text when they are not provided.\n */\nexport interface SpanDefinition<S extends object, E extends SpanEnd> {\n  readonly name: string;\n  readonly start: make.IoMessageMaker<S>;\n  readonly end: make.IoMessageMaker<E>;\n}\n\n/**\n * Arguments to the span.end() function\n *\n * `SpanEndArguments<T>` are the fields that a user still needs to supply, it\n * fields in the type `T` that aren't also in `SpanEnd`. `SpanEnd` represents\n * fields that are automatically added by the underlying `end` function.\n *\n * Fields that are already in `SpanEnd` are still rendered as optionals, so you\n * can override them (but you don't have to).\n *\n * - Does the following: fields that are shared between `T` and `SpanEnd` are\n *   made optional, and the rest of the keys of `T` are required.\n *\n * - If `T` is fully subsumed by the `SpanEnd` type, then an object type with\n *   all fields optional, OR 'void' so you can avoid passing an argument at all.\n */\ntype SpanEndArguments<T> = keyof T extends keyof SpanEnd\n  ? (Pick<Partial<SpanEnd>, keyof T & keyof SpanEnd> | void)\n  : Optional<T, keyof T & keyof SpanEnd>;\n\n/**\n * Used in conditional types to check if a type (e.g. after omitting fields) is an empty object\n * This is needed because counter-intuitive neither `object` nor `{}` represent that.\n */\ntype EmptyObject = Record<string, never>;\n\n/**\n * Helper type to force a parameter to be not present of the computed type is an empty object\n */\ntype VoidWhenEmpty<T> = T extends EmptyObject ? void : T;\n\n/**\n * Make some properties optional\n */\ntype Optional<T, K extends keyof T> = Omit<T, K> & Pick<Partial<T>, K>;\n\n/**\n * Ending the span returns the observed duration\n */\nexport interface ElapsedTime {\n  readonly asMs: number;\n  readonly asSec: number;\n}\n\n/**\n * A message span that can be ended and read times from\n */\nexport interface IMessageSpan<E extends SpanEnd> extends IActionAwareIoHost {\n  /**\n   * An IoHelper wrapped around the span.\n   */\n  readonly asHelper: IoHelper;\n  /**\n   * An IoDefaultMessages wrapped around the span.\n   */\n  readonly defaults: IoDefaultMessages;\n\n  /**\n   * Get the time elapsed since the start\n   */\n  elapsedTime(): Promise<ElapsedTime>;\n  /**\n   * Sends a simple, generic message with the current timing\n   * For more complex intermediate messages, get the `elapsedTime` and use `notify`\n   */\n  timing(maker: make.IoMessageMaker<Duration>, message?: string): Promise<ElapsedTime>;\n  /**\n   * End the span with a payload\n   */\n  end(payload: SpanEndArguments<E>): Promise<ElapsedTime>;\n  /**\n   * End the span with a message and payload\n   */\n  end(message: string, payload: SpanEndArguments<E>): Promise<ElapsedTime>;\n\n  /**\n   * Increment a counter\n   */\n  incCounter(name: string, delta?: number): void;\n\n  /**\n   * Return a new timer object\n   *\n   * It will be added into the span data when it's stopped. All open timers are\n   * automatically stopped when the span is ended.\n   *\n   * Timers are ultimately added to the `counters` array with `<name>_ms` and\n   * `<name>_cnt` keys.\n   */\n  startTimer(name: string): ITimer;\n}\n\n/**\n * A timer to time an operation in a span.\n */\nexport interface ITimer {\n  stop(): void;\n}\n\n/**\n * Helper class to make spans around blocks of work\n *\n * Blocks are enclosed by a start and end message.\n * All messages of the span share a unique id.\n * The end message contains the time passed between start and end.\n */\nexport class SpanMaker<S extends object, E extends SpanEnd> {\n  private readonly definition: SpanDefinition<S, E>;\n  private readonly ioHelper: IoHelper;\n  private makeHelper: (ioHost: IActionAwareIoHost) => IoHelper;\n\n  public constructor(ioHelper: IoHelper, definition: SpanDefinition<S, E>, makeHelper: (ioHost: IActionAwareIoHost) => IoHelper) {\n    this.definition = definition;\n    this.ioHelper = ioHelper;\n    this.makeHelper = makeHelper;\n  }\n\n  /**\n   * Starts the span and initially notifies the IoHost\n   * @returns a message span\n   */\n  public async begin(payload: VoidWhenEmpty<S>): Promise<IMessageSpan<E>>;\n  public async begin(message: string, payload: S): Promise<IMessageSpan<E>>;\n  public async begin(a: any, b?: S): Promise<IMessageSpan<E>> {\n    const span = new MessageSpan(this.ioHelper, this.definition, this.makeHelper);\n    const startInput = parseArgs<S>(a, b);\n    const startMsg = startInput.message ?? `Starting ${this.definition.name} ...`;\n    const startPayload = startInput.payload;\n    await span.notify(this.definition.start.msg(startMsg, startPayload));\n\n    return span;\n  }\n}\n\nclass MessageSpan<S extends object, E extends SpanEnd> implements IMessageSpan<E> {\n  public readonly asHelper: IoHelper;\n\n  private readonly definition: SpanDefinition<S, E>;\n  private readonly ioHelper: IoHelper;\n  private readonly spanId: string;\n  private readonly startTime: number;\n  private readonly timingMsgTemplate: string;\n  private readonly counters: Record<string, number> = {};\n  private readonly openTimers = new Set<ITimer>();\n\n  public constructor(ioHelper: IoHelper, definition: SpanDefinition<S, E>, makeHelper: (ioHost: IActionAwareIoHost) => IoHelper) {\n    this.definition = definition;\n    this.ioHelper = ioHelper;\n    this.spanId = uuid.v4();\n    this.startTime = new Date().getTime();\n    this.timingMsgTemplate = '\\n✨  %s time: %ds\\n';\n    this.asHelper = makeHelper(this);\n  }\n\n  public get defaults(): IoDefaultMessages {\n    return this.asHelper.defaults;\n  }\n\n  public async elapsedTime(): Promise<ElapsedTime> {\n    return this.time();\n  }\n  public async timing(maker: make.IoMessageMaker<Duration>, message?: string): Promise<ElapsedTime> {\n    const duration = this.time();\n    const timingMsg = message ? message : util.format(this.timingMsgTemplate, this.definition.name, duration.asSec);\n    await this.notify(maker.msg(timingMsg, {\n      duration: duration.asMs,\n    }));\n    return duration;\n  }\n  public async notify(msg: ActionLessMessage<unknown>): Promise<void> {\n    return this.ioHelper.notify(withSpanId(this.spanId, msg));\n  }\n  public async end(x: any, y?: SpanEndArguments<E>): Promise<ElapsedTime> {\n    const duration = this.time();\n\n    for (const t of this.openTimers) {\n      t.stop();\n    }\n    this.openTimers.clear();\n\n    const endInput = parseArgs<SpanEndArguments<E>>(x, y);\n    const endMsg = endInput.message ?? util.format(this.timingMsgTemplate, this.definition.name, duration.asSec);\n    const endPayload = endInput.payload;\n\n    await this.notify(this.definition.end.msg(\n      endMsg, {\n        duration: duration.asMs,\n        ...(Object.keys(this.counters).length > 0 ? { counters: this.counters } : {}),\n        ...endPayload,\n      } as E));\n\n    return duration;\n  }\n\n  public incCounter(name: string, delta: number = 1): void {\n    this.counters[name] = (this.counters[name] ?? 0) + delta;\n  }\n\n  public async requestResponse<T>(msg: ActionLessRequest<unknown, T>): Promise<T> {\n    return this.ioHelper.requestResponse(withSpanId(this.spanId, msg));\n  }\n\n  public startTimer(name: string): ITimer {\n    const start = Date.now();\n\n    const t: ITimer = {\n      stop: () => {\n        this.openTimers.delete(t);\n        this.incCounter(`${name}_ms`, Math.floor(Date.now() - start) / 1000);\n        this.incCounter(`${name}_cnt`, 1);\n      },\n    };\n    this.openTimers.add(t);\n    return t;\n  }\n\n  private time() {\n    const elapsedTime = new Date().getTime() - this.startTime;\n    return {\n      asMs: elapsedTime,\n      asSec: formatTime(elapsedTime),\n    };\n  }\n}\n\nfunction parseArgs<S>(first: any, second?: S): { message: string | undefined; payload: S } {\n  const firstIsMessage = typeof first === 'string';\n\n  // When the first argument is a string or we have a second argument, then the first arg is the message\n  const message = (firstIsMessage || second) ? first : undefined;\n\n  // When the first argument is a string or we have a second argument,\n  // then the second arg is the payload, otherwise the first arg is the payload\n  const payload = (firstIsMessage || second) ? second : first;\n\n  return {\n    message,\n    payload,\n  };\n}\n\nfunction withSpanId<T extends object>(span: string, message: T): T & { span: string } {\n  return {\n    ...message,\n    span,\n  };\n}\n"]}
|
package/lib/api/tree.js
CHANGED
|
@@ -18,7 +18,7 @@ async function loadTree(assembly, trace) {
|
|
|
18
18
|
try {
|
|
19
19
|
const outdir = assembly.directory;
|
|
20
20
|
const fileName = assembly.tree()?.file;
|
|
21
|
-
return fileName ? fs.readJSONSync(path.join(outdir, fileName)).tree :
|
|
21
|
+
return fileName ? fs.readJSONSync(path.join(outdir, fileName)).tree : undefined;
|
|
22
22
|
}
|
|
23
23
|
catch (e) {
|
|
24
24
|
await trace(`Failed to get tree.json file: ${e}. Proceeding with empty tree.`);
|
|
@@ -34,4 +34,4 @@ async function loadTreeFromDir(outdir, trace) {
|
|
|
34
34
|
return undefined;
|
|
35
35
|
}
|
|
36
36
|
}
|
|
37
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
37
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJlZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInRyZWUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUE4QkEsb0JBTUM7QUFFRCw0QkFTQztBQUVELDBDQU9DO0FBeERELGtDQUFrQztBQUVsQywrQkFBK0I7QUF5Qi9COztHQUVHO0FBQ0gsU0FBZ0IsSUFBSSxDQUFDLElBQW1DLEVBQUUsU0FBNEM7SUFDcEcsT0FBTyxJQUFJLElBQUksSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLGNBQWMsRUFBRSxDQUFDLENBQUM7SUFFN0QsU0FBUyxjQUFjO1FBQ3JCLE9BQU8sTUFBTSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsUUFBUSxJQUFJLEVBQUUsQ0FBQyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsU0FBUyxDQUFDLENBQUMsQ0FBQztJQUNuRixDQUFDO0FBQ0gsQ0FBQztBQUVNLEtBQUssVUFBVSxRQUFRLENBQUMsUUFBdUIsRUFBRSxLQUFxQztJQUMzRixJQUFJLENBQUM7UUFDSCxNQUFNLE1BQU0sR0FBRyxRQUFRLENBQUMsU0FBUyxDQUFDO1FBQ2xDLE1BQU0sUUFBUSxHQUFHLFFBQVEsQ0FBQyxJQUFJLEVBQUUsRUFBRSxJQUFJLENBQUM7UUFDdkMsT0FBTyxRQUFRLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsUUFBUSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQztJQUNsRixDQUFDO0lBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztRQUNYLE1BQU0sS0FBSyxDQUFDLGlDQUFpQyxDQUFDLCtCQUErQixDQUFDLENBQUM7UUFDL0UsT0FBTyxTQUFTLENBQUM7SUFDbkIsQ0FBQztBQUNILENBQUM7QUFFTSxLQUFLLFVBQVUsZUFBZSxDQUFDLE1BQWMsRUFBRSxLQUFxQztJQUN6RixJQUFJLENBQUM7UUFDSCxPQUFPLEVBQUUsQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsV0FBVyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUM7SUFDOUQsQ0FBQztJQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7UUFDWCxNQUFNLEtBQUssQ0FBQyxpQ0FBaUMsQ0FBQywrQkFBK0IsQ0FBQyxDQUFDO1FBQy9FLE9BQU8sU0FBUyxDQUFDO0lBQ25CLENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgcGF0aCBmcm9tICdub2RlOnBhdGgnO1xuaW1wb3J0IHR5cGUgeyBDbG91ZEFzc2VtYmx5IH0gZnJvbSAnQGF3cy1jZGsvY2xvdWQtYXNzZW1ibHktYXBpJztcbmltcG9ydCAqIGFzIGZzIGZyb20gJ2ZzLWV4dHJhJztcblxuLyoqXG4gKiBTb3VyY2UgaW5mb3JtYXRpb24gb24gYSBjb25zdHJ1Y3QgKGNsYXNzIGZxbiBhbmQgdmVyc2lvbilcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBDb25zdHJ1Y3RJbmZvIHtcbiAgcmVhZG9ubHkgZnFuOiBzdHJpbmc7XG4gIHJlYWRvbmx5IHZlcnNpb246IHN0cmluZztcbn1cblxuLyoqXG4gKiBBIG5vZGUgaW4gdGhlIGNvbnN0cnVjdCB0cmVlLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvbnN0cnVjdFRyZWVOb2RlIHtcbiAgcmVhZG9ubHkgaWQ6IHN0cmluZztcbiAgcmVhZG9ubHkgcGF0aDogc3RyaW5nO1xuICByZWFkb25seSBjaGlsZHJlbj86IHsgW2tleTogc3RyaW5nXTogQ29uc3RydWN0VHJlZU5vZGUgfTtcbiAgcmVhZG9ubHkgYXR0cmlidXRlcz86IHsgW2tleTogc3RyaW5nXTogYW55IH07XG5cbiAgLyoqXG4gICAqIEluZm9ybWF0aW9uIG9uIHRoZSBjb25zdHJ1Y3QgY2xhc3MgdGhhdCBsZWQgdG8gdGhpcyBub2RlLCBpZiBhdmFpbGFibGVcbiAgICovXG4gIHJlYWRvbmx5IGNvbnN0cnVjdEluZm8/OiBDb25zdHJ1Y3RJbmZvO1xufVxuXG4vKipcbiAqIFdoZXRoZXIgdGhlIHByb3ZpZGVkIHByZWRpY2F0ZSBpcyB0cnVlIGZvciBhdCBsZWFzdCBvbmUgZWxlbWVudCBpbiB0aGUgY29uc3RydWN0IChzdWItKXRyZWUuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBzb21lKG5vZGU6IENvbnN0cnVjdFRyZWVOb2RlIHwgdW5kZWZpbmVkLCBwcmVkaWNhdGU6IChuOiBDb25zdHJ1Y3RUcmVlTm9kZSkgPT4gYm9vbGVhbik6IGJvb2xlYW4ge1xuICByZXR1cm4gbm9kZSAhPSBudWxsICYmIChwcmVkaWNhdGUobm9kZSkgfHwgZmluZEluQ2hpbGRyZW4oKSk7XG5cbiAgZnVuY3Rpb24gZmluZEluQ2hpbGRyZW4oKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIE9iamVjdC52YWx1ZXMobm9kZT8uY2hpbGRyZW4gPz8ge30pLnNvbWUoY2hpbGQgPT4gc29tZShjaGlsZCwgcHJlZGljYXRlKSk7XG4gIH1cbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGxvYWRUcmVlKGFzc2VtYmx5OiBDbG91ZEFzc2VtYmx5LCB0cmFjZTogKG1zZzogc3RyaW5nKSA9PiBQcm9taXNlPHZvaWQ+KTogUHJvbWlzZTxDb25zdHJ1Y3RUcmVlTm9kZSB8IHVuZGVmaW5lZD4ge1xuICB0cnkge1xuICAgIGNvbnN0IG91dGRpciA9IGFzc2VtYmx5LmRpcmVjdG9yeTtcbiAgICBjb25zdCBmaWxlTmFtZSA9IGFzc2VtYmx5LnRyZWUoKT8uZmlsZTtcbiAgICByZXR1cm4gZmlsZU5hbWUgPyBmcy5yZWFkSlNPTlN5bmMocGF0aC5qb2luKG91dGRpciwgZmlsZU5hbWUpKS50cmVlIDogdW5kZWZpbmVkO1xuICB9IGNhdGNoIChlKSB7XG4gICAgYXdhaXQgdHJhY2UoYEZhaWxlZCB0byBnZXQgdHJlZS5qc29uIGZpbGU6ICR7ZX0uIFByb2NlZWRpbmcgd2l0aCBlbXB0eSB0cmVlLmApO1xuICAgIHJldHVybiB1bmRlZmluZWQ7XG4gIH1cbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGxvYWRUcmVlRnJvbURpcihvdXRkaXI6IHN0cmluZywgdHJhY2U6IChtc2c6IHN0cmluZykgPT4gUHJvbWlzZTx2b2lkPik6IFByb21pc2U8Q29uc3RydWN0VHJlZU5vZGUgfCB1bmRlZmluZWQ+IHtcbiAgdHJ5IHtcbiAgICByZXR1cm4gZnMucmVhZEpTT05TeW5jKHBhdGguam9pbihvdXRkaXIsICd0cmVlLmpzb24nKSkudHJlZTtcbiAgfSBjYXRjaCAoZSkge1xuICAgIGF3YWl0IHRyYWNlKGBGYWlsZWQgdG8gZ2V0IHRyZWUuanNvbiBmaWxlOiAke2V9LiBQcm9jZWVkaW5nIHdpdGggZW1wdHkgdHJlZS5gKTtcbiAgICByZXR1cm4gdW5kZWZpbmVkO1xuICB9XG59XG4iXX0=
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import * as cxapi from '@aws-cdk/cloud-assembly-api';
|
|
2
|
+
import { WorkGraph } from './work-graph';
|
|
3
|
+
import type { IoHelper } from '../io/private';
|
|
4
|
+
/**
|
|
5
|
+
* Build a WorkGraph for destroy with reversed dependencies.
|
|
6
|
+
*
|
|
7
|
+
* In deploy order, if A depends on B, B is deployed first. For destroy,
|
|
8
|
+
* the arrows are reversed: A must be destroyed before B.
|
|
9
|
+
*/
|
|
10
|
+
export declare function buildDestroyWorkGraph(stacks: cxapi.CloudFormationStackArtifact[], ioHelper: IoHelper): WorkGraph;
|
|
11
|
+
//# sourceMappingURL=build-destroy-work-graph.d.ts.map
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.buildDestroyWorkGraph = buildDestroyWorkGraph;
|
|
4
|
+
const cxapi = require("@aws-cdk/cloud-assembly-api");
|
|
5
|
+
const work_graph_1 = require("./work-graph");
|
|
6
|
+
const work_graph_types_1 = require("./work-graph-types");
|
|
7
|
+
/**
|
|
8
|
+
* Build a WorkGraph for destroy with reversed dependencies.
|
|
9
|
+
*
|
|
10
|
+
* In deploy order, if A depends on B, B is deployed first. For destroy,
|
|
11
|
+
* the arrows are reversed: A must be destroyed before B.
|
|
12
|
+
*/
|
|
13
|
+
function buildDestroyWorkGraph(stacks, ioHelper) {
|
|
14
|
+
const graph = new work_graph_1.WorkGraph({}, ioHelper);
|
|
15
|
+
const selectedIds = new Set(stacks.map((s) => s.id));
|
|
16
|
+
for (const stack of stacks) {
|
|
17
|
+
graph.addNodes({
|
|
18
|
+
type: 'stack',
|
|
19
|
+
id: stack.id,
|
|
20
|
+
dependencies: new Set(),
|
|
21
|
+
stack,
|
|
22
|
+
deploymentState: work_graph_types_1.DeploymentState.PENDING,
|
|
23
|
+
priority: 0,
|
|
24
|
+
});
|
|
25
|
+
}
|
|
26
|
+
for (const stack of stacks) {
|
|
27
|
+
for (const dep of stack.dependencies) {
|
|
28
|
+
if (cxapi.CloudFormationStackArtifact.isCloudFormationStackArtifact(dep) && selectedIds.has(dep.id)) {
|
|
29
|
+
graph.addDependency(dep.id, stack.id);
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
return graph;
|
|
34
|
+
}
|
|
35
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnVpbGQtZGVzdHJveS13b3JrLWdyYXBoLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiYnVpbGQtZGVzdHJveS13b3JrLWdyYXBoLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBV0Esc0RBd0JDO0FBbkNELHFEQUFxRDtBQUNyRCw2Q0FBeUM7QUFDekMseURBQXFEO0FBR3JEOzs7OztHQUtHO0FBQ0gsU0FBZ0IscUJBQXFCLENBQUMsTUFBMkMsRUFBRSxRQUFrQjtJQUNuRyxNQUFNLEtBQUssR0FBRyxJQUFJLHNCQUFTLENBQUMsRUFBRSxFQUFFLFFBQVEsQ0FBQyxDQUFDO0lBQzFDLE1BQU0sV0FBVyxHQUFHLElBQUksR0FBRyxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO0lBRXJELEtBQUssTUFBTSxLQUFLLElBQUksTUFBTSxFQUFFLENBQUM7UUFDM0IsS0FBSyxDQUFDLFFBQVEsQ0FBQztZQUNiLElBQUksRUFBRSxPQUFPO1lBQ2IsRUFBRSxFQUFFLEtBQUssQ0FBQyxFQUFFO1lBQ1osWUFBWSxFQUFFLElBQUksR0FBRyxFQUFVO1lBQy9CLEtBQUs7WUFDTCxlQUFlLEVBQUUsa0NBQWUsQ0FBQyxPQUFPO1lBQ3hDLFFBQVEsRUFBRSxDQUFDO1NBQ1osQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUVELEtBQUssTUFBTSxLQUFLLElBQUksTUFBTSxFQUFFLENBQUM7UUFDM0IsS0FBSyxNQUFNLEdBQUcsSUFBSSxLQUFLLENBQUMsWUFBWSxFQUFFLENBQUM7WUFDckMsSUFBSSxLQUFLLENBQUMsMkJBQTJCLENBQUMsNkJBQTZCLENBQUMsR0FBRyxDQUFDLElBQUksV0FBVyxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQztnQkFDcEcsS0FBSyxDQUFDLGFBQWEsQ0FBQyxHQUFHLENBQUMsRUFBRSxFQUFFLEtBQUssQ0FBQyxFQUFFLENBQUMsQ0FBQztZQUN4QyxDQUFDO1FBQ0gsQ0FBQztJQUNILENBQUM7SUFFRCxPQUFPLEtBQUssQ0FBQztBQUNmLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBjeGFwaSBmcm9tICdAYXdzLWNkay9jbG91ZC1hc3NlbWJseS1hcGknO1xuaW1wb3J0IHsgV29ya0dyYXBoIH0gZnJvbSAnLi93b3JrLWdyYXBoJztcbmltcG9ydCB7IERlcGxveW1lbnRTdGF0ZSB9IGZyb20gJy4vd29yay1ncmFwaC10eXBlcyc7XG5pbXBvcnQgdHlwZSB7IElvSGVscGVyIH0gZnJvbSAnLi4vaW8vcHJpdmF0ZSc7XG5cbi8qKlxuICogQnVpbGQgYSBXb3JrR3JhcGggZm9yIGRlc3Ryb3kgd2l0aCByZXZlcnNlZCBkZXBlbmRlbmNpZXMuXG4gKlxuICogSW4gZGVwbG95IG9yZGVyLCBpZiBBIGRlcGVuZHMgb24gQiwgQiBpcyBkZXBsb3llZCBmaXJzdC4gRm9yIGRlc3Ryb3ksXG4gKiB0aGUgYXJyb3dzIGFyZSByZXZlcnNlZDogQSBtdXN0IGJlIGRlc3Ryb3llZCBiZWZvcmUgQi5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGJ1aWxkRGVzdHJveVdvcmtHcmFwaChzdGFja3M6IGN4YXBpLkNsb3VkRm9ybWF0aW9uU3RhY2tBcnRpZmFjdFtdLCBpb0hlbHBlcjogSW9IZWxwZXIpOiBXb3JrR3JhcGgge1xuICBjb25zdCBncmFwaCA9IG5ldyBXb3JrR3JhcGgoe30sIGlvSGVscGVyKTtcbiAgY29uc3Qgc2VsZWN0ZWRJZHMgPSBuZXcgU2V0KHN0YWNrcy5tYXAoKHMpID0+IHMuaWQpKTtcblxuICBmb3IgKGNvbnN0IHN0YWNrIG9mIHN0YWNrcykge1xuICAgIGdyYXBoLmFkZE5vZGVzKHtcbiAgICAgIHR5cGU6ICdzdGFjaycsXG4gICAgICBpZDogc3RhY2suaWQsXG4gICAgICBkZXBlbmRlbmNpZXM6IG5ldyBTZXQ8c3RyaW5nPigpLFxuICAgICAgc3RhY2ssXG4gICAgICBkZXBsb3ltZW50U3RhdGU6IERlcGxveW1lbnRTdGF0ZS5QRU5ESU5HLFxuICAgICAgcHJpb3JpdHk6IDAsXG4gICAgfSk7XG4gIH1cblxuICBmb3IgKGNvbnN0IHN0YWNrIG9mIHN0YWNrcykge1xuICAgIGZvciAoY29uc3QgZGVwIG9mIHN0YWNrLmRlcGVuZGVuY2llcykge1xuICAgICAgaWYgKGN4YXBpLkNsb3VkRm9ybWF0aW9uU3RhY2tBcnRpZmFjdC5pc0Nsb3VkRm9ybWF0aW9uU3RhY2tBcnRpZmFjdChkZXApICYmIHNlbGVjdGVkSWRzLmhhcyhkZXAuaWQpKSB7XG4gICAgICAgIGdyYXBoLmFkZERlcGVuZGVuY3koZGVwLmlkLCBzdGFjay5pZCk7XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgcmV0dXJuIGdyYXBoO1xufVxuIl19
|
|
@@ -14,7 +14,8 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
14
14
|
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
15
|
};
|
|
16
16
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./build-destroy-work-graph"), exports);
|
|
17
18
|
__exportStar(require("./work-graph"), exports);
|
|
18
19
|
__exportStar(require("./work-graph-builder"), exports);
|
|
19
20
|
__exportStar(require("./work-graph-types"), exports);
|
|
20
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
21
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsNkRBQTJDO0FBQzNDLCtDQUE2QjtBQUM3Qix1REFBcUM7QUFDckMscURBQW1DIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9idWlsZC1kZXN0cm95LXdvcmstZ3JhcGgnO1xuZXhwb3J0ICogZnJvbSAnLi93b3JrLWdyYXBoJztcbmV4cG9ydCAqIGZyb20gJy4vd29yay1ncmFwaC1idWlsZGVyJztcbmV4cG9ydCAqIGZyb20gJy4vd29yay1ncmFwaC10eXBlcyc7XG4iXX0=
|