@aws-cdk/toolkit-lib 1.18.0 → 1.19.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/build-info.json CHANGED
@@ -1,4 +1,4 @@
1
1
  {
2
- "comment": "Generated at 2026-03-09T13:28:43Z by build-info.sh",
3
- "commit": "970da46"
2
+ "comment": "Generated at 2026-03-18T18:26:21Z by build-info.sh",
3
+ "commit": "48e9b5d"
4
4
  }
@@ -10,5 +10,11 @@ export interface DestroyOptions {
10
10
  * The arn of the IAM role to use for the stack destroy operation
11
11
  */
12
12
  readonly roleArn?: string;
13
+ /**
14
+ * Maximum number of simultaneous destroys (dependency permitting) to execute.
15
+ *
16
+ * @default 1
17
+ */
18
+ readonly concurrency?: number;
13
19
  }
14
20
  //# sourceMappingURL=index.d.ts.map
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBTdGFja1NlbGVjdG9yIH0gZnJvbSAnLi4vLi4vYXBpL2Nsb3VkLWFzc2VtYmx5JztcblxuZXhwb3J0IGludGVyZmFjZSBEZXN0cm95T3B0aW9ucyB7XG4gIC8qKlxuICAgKiBDcml0ZXJpYSBmb3Igc2VsZWN0aW5nIHN0YWNrcyB0byBkZXBsb3lcbiAgICpcbiAgICogQGRlZmF1bHQgLSBBbGwgc3RhY2tzXG4gICAqL1xuICByZWFkb25seSBzdGFja3M/OiBTdGFja1NlbGVjdG9yO1xuXG4gIC8qKlxuICAgKiBUaGUgYXJuIG9mIHRoZSBJQU0gcm9sZSB0byB1c2UgZm9yIHRoZSBzdGFjayBkZXN0cm95IG9wZXJhdGlvblxuICAgKi9cbiAgcmVhZG9ubHkgcm9sZUFybj86IHN0cmluZztcbn1cbiJdfQ==
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBTdGFja1NlbGVjdG9yIH0gZnJvbSAnLi4vLi4vYXBpL2Nsb3VkLWFzc2VtYmx5JztcblxuZXhwb3J0IGludGVyZmFjZSBEZXN0cm95T3B0aW9ucyB7XG4gIC8qKlxuICAgKiBDcml0ZXJpYSBmb3Igc2VsZWN0aW5nIHN0YWNrcyB0byBkZXBsb3lcbiAgICpcbiAgICogQGRlZmF1bHQgLSBBbGwgc3RhY2tzXG4gICAqL1xuICByZWFkb25seSBzdGFja3M/OiBTdGFja1NlbGVjdG9yO1xuXG4gIC8qKlxuICAgKiBUaGUgYXJuIG9mIHRoZSBJQU0gcm9sZSB0byB1c2UgZm9yIHRoZSBzdGFjayBkZXN0cm95IG9wZXJhdGlvblxuICAgKi9cbiAgcmVhZG9ubHkgcm9sZUFybj86IHN0cmluZztcblxuICAvKipcbiAgICogTWF4aW11bSBudW1iZXIgb2Ygc2ltdWx0YW5lb3VzIGRlc3Ryb3lzIChkZXBlbmRlbmN5IHBlcm1pdHRpbmcpIHRvIGV4ZWN1dGUuXG4gICAqXG4gICAqIEBkZWZhdWx0IDFcbiAgICovXG4gIHJlYWRvbmx5IGNvbmN1cnJlbmN5PzogbnVtYmVyO1xufVxuIl19
@@ -111,6 +111,10 @@ export declare class ExecutionEnvironment implements AsyncDisposable {
111
111
  * `process.env` later.
112
112
  */
113
113
  export declare function writeContextToEnv(env: Env, context: Context, completeness: 'add-process-env-later' | 'env-is-complete'): () => Promise<void>;
114
+ /**
115
+ * Find the `aws-cdk-lib` library version by inspecting construct tree debug information
116
+ */
117
+ export declare function findConstructLibraryVersion(tree: ConstructTreeNode | undefined): string | undefined;
114
118
  /**
115
119
  * Checks if the framework supports context overflow
116
120
  */
@@ -2,6 +2,7 @@
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.ExecutionEnvironment = void 0;
4
4
  exports.writeContextToEnv = writeContextToEnv;
5
+ exports.findConstructLibraryVersion = findConstructLibraryVersion;
5
6
  exports.frameworkSupportsContextOverflow = frameworkSupportsContextOverflow;
6
7
  exports.assemblyFromDirectory = assemblyFromDirectory;
7
8
  exports.settingsFromSynthOptions = settingsFromSynthOptions;
@@ -215,9 +216,7 @@ function writeContextToEnv(env, context, completeness) {
215
216
  *
216
217
  * @param assembly - the assembly to check
217
218
  */
218
- async function checkContextOverflowSupport(assembly, ioHelper) {
219
- const traceFn = (msg) => ioHelper.defaults.trace(msg);
220
- const tree = await (0, tree_1.loadTree)(assembly, traceFn);
219
+ async function checkContextOverflowSupport(tree, ioHelper) {
221
220
  // We're dealing with an old version of the framework here. It is unaware of the temporary
222
221
  // file, which means that it will ignore the context overflow.
223
222
  if (!frameworkSupportsContextOverflow(tree)) {
@@ -225,18 +224,27 @@ async function checkContextOverflowSupport(assembly, ioHelper) {
225
224
  }
226
225
  }
227
226
  /**
228
- * Checks if the framework supports context overflow
227
+ * Find the `aws-cdk-lib` library version by inspecting construct tree debug information
229
228
  */
230
- function frameworkSupportsContextOverflow(tree) {
231
- return !(0, tree_1.some)(tree, node => {
229
+ function findConstructLibraryVersion(tree) {
230
+ let ret = undefined;
231
+ (0, tree_1.some)(tree, node => {
232
232
  const fqn = node.constructInfo?.fqn;
233
233
  const version = node.constructInfo?.version;
234
- return (fqn === 'aws-cdk-lib.App' // v2 app
235
- && version !== '0.0.0' // ignore developer builds
236
- && version != null && (0, semver_1.lte)(version, '2.38.0') // last version not supporting large context
237
- ) // v2
238
- || fqn === '@aws-cdk/core.App'; // v1 app => not supported
234
+ if (fqn?.startsWith('aws-cdk-lib.') || fqn?.startsWith('@aws-cdk/core.')) {
235
+ ret = version;
236
+ return true;
237
+ }
238
+ return false;
239
239
  });
240
+ return ret !== '0.0.0' ? ret : undefined;
241
+ }
242
+ /**
243
+ * Checks if the framework supports context overflow
244
+ */
245
+ function frameworkSupportsContextOverflow(tree) {
246
+ const version = findConstructLibraryVersion(tree);
247
+ return !version || !(0, semver_1.lte)(version, '2.38.0');
240
248
  }
241
249
  /**
242
250
  * Safely create an assembly from a cloud assembly directory
@@ -249,7 +257,8 @@ async function assemblyFromDirectory(assemblyDir, ioHelper, loadOptions = {}) {
249
257
  // We sort as we deploy
250
258
  topoSort: false,
251
259
  });
252
- await checkContextOverflowSupport(assembly, ioHelper);
260
+ const tree = await (0, tree_1.loadTree)(assembly, ioHelper.defaults.trace.bind(ioHelper.defaults));
261
+ await checkContextOverflowSupport(tree, ioHelper);
253
262
  return assembly;
254
263
  }
255
264
  catch (err) {
@@ -282,4 +291,4 @@ function settingsFromSynthOptions(synthOpts = {}) {
282
291
  function parametersFromSynthOptions(synthOptions) {
283
292
  return (0, environment_1.synthParametersFromSettings)(settingsFromSynthOptions(synthOptions ?? {}));
284
293
  }
285
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"prepare-source.js","sourceRoot":"","sources":["prepare-source.ts"],"names":[],"mappings":"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","sourcesContent":["import '../../../private/dispose-polyfill';\nimport * as os from 'node:os';\nimport * as path from 'node:path';\nimport { format } from 'node:util';\nimport { CloudAssembly } from '@aws-cdk/cloud-assembly-api';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as fs from 'fs-extra';\nimport { lte } from 'semver';\nimport type { ToolkitServices } from '../../../toolkit/private';\nimport { ToolkitError } from '../../../toolkit/toolkit-error';\nimport { splitBySize, versionNumber } from '../../../util';\nimport type { SdkProvider } from '../../aws-auth/private';\nimport type { IoHelper } from '../../io/private';\nimport { IO } from '../../io/private';\nimport type { IReadLock, IWriteLock } from '../../rwlock';\nimport { RWLock } from '../../rwlock';\nimport { Settings } from '../../settings';\nimport type { ConstructTreeNode } from '../../tree';\nimport { loadTree, some } from '../../tree';\nimport type { Context, Env } from '../environment';\nimport { prepareDefaultEnvironment, spaceAvailableForContext, guessExecutable, synthParametersFromSettings } from '../environment';\nimport type { AppSynthOptions, LoadAssemblyOptions } from '../source-builder';\n\nexport interface ExecutionEnvironmentOptions {\n  /**\n   * The directory the cloud assembly will be written to.\n   *\n   * @default - use a temporary directory as output directory, this will be cleaned up when this object is disposed\n   */\n  readonly outdir?: string;\n\n  /**\n   * Resolve and add environment variables for the app's default environment.\n   *\n   * This will make a call to STS, which is not always desirable e.g. if the env is explicitly specified.\n   */\n  readonly resolveDefaultAppEnv: boolean;\n}\n\nexport class ExecutionEnvironment implements AsyncDisposable {\n  /**\n   * Create an ExecutionEnvironment\n   *\n   * An ExecutionEnvironment holds a writer lock on the given directory which will\n   * be cleaned up when the object is disposed.\n   *\n   * A temporary directory will be created if none is supplied, which will be cleaned\n   * up when this object is disposed.\n   *\n   * If `markSuccessful()` is called, the writer lock is converted to a reader lock\n   * and temporary directories will not be cleaned up anymore.\n   */\n  public static async create(services: ToolkitServices, options: ExecutionEnvironmentOptions) {\n    let outDirIsTemporary = false;\n    let dir = options.outdir;\n    if (!dir) {\n      outDirIsTemporary = true;\n      dir = fs.mkdtempSync(path.join(fs.realpathSync(os.tmpdir()), 'cdk.out'));\n    }\n    const lock = await new RWLock(dir).acquireWrite();\n\n    const opts = {\n      outdir: dir,\n      resolveDefaultAppEnv: options.resolveDefaultAppEnv,\n    };\n\n    return new ExecutionEnvironment(services, opts, {\n      lock,\n      outDirIsTemporary,\n    });\n  }\n\n  /**\n   * Should the outdir be disposed of.\n   */\n  public get shouldDisposeOutDir(): boolean {\n    return this.shouldClean;\n  }\n\n  /**\n   * The directory the cloud assembly will be written to.\n   */\n  public readonly outdir: string;\n\n  private readonly options: Required<ExecutionEnvironmentOptions>;\n  private readonly ioHelper: IoHelper;\n  private readonly sdkProvider: SdkProvider;\n  private readonly debugFn: (msg: string) => Promise<void>;\n  private lock: IWriteLock | undefined;\n  private shouldClean: boolean;\n\n  private constructor(\n    services: ToolkitServices,\n    options: Required<ExecutionEnvironmentOptions>,\n    { lock, outDirIsTemporary }: {\n      readonly outDirIsTemporary: boolean;\n      readonly lock: IWriteLock;\n    },\n  ) {\n    this.ioHelper = services.ioHelper;\n    this.sdkProvider = services.sdkProvider;\n    this.debugFn = (msg: string) => this.ioHelper.defaults.debug(msg);\n    this.lock = lock;\n    this.shouldClean = outDirIsTemporary;\n    this.outdir = options.outdir;\n    this.options = options;\n  }\n\n  public async [Symbol.asyncDispose]() {\n    await this.lock?.release();\n\n    if (this.shouldDisposeOutDir) {\n      await fs.rm(this.outdir, { recursive: true, force: true });\n    }\n  }\n\n  /**\n   * Mark the execution as successful, which stops the writer lock from being released upon disposal\n   */\n  public async markSuccessful() {\n    if (!this.lock) {\n      throw new TypeError('Cannot mark successful more than once');\n    }\n    const readLock = await this.lock.convertToReaderLock();\n    this.lock = undefined;\n    this.shouldClean = false;\n    return { readLock };\n  }\n\n  /**\n   * Begin an execution in this environment\n   *\n   * This will acquire a write lock on the given environment. The write lock\n   * will be released automatically when the return object is disposed, unless it\n   * is converted to a reader lock.\n   */\n  public async beginExecution(): Promise<{ writeToReadLock(): Promise<IReadLock> } & AsyncDisposable> {\n    const lock = await new RWLock(this.outdir).acquireWrite();\n\n    let converted = false;\n    return {\n      async writeToReadLock() {\n        converted = true;\n        return lock.convertToReaderLock();\n      },\n      [Symbol.asyncDispose]: async () => {\n        // Release if not converted\n        if (!converted) {\n          await lock.release();\n        }\n      },\n    };\n  }\n\n  /**\n   * Guess the executable from the command-line argument\n   *\n   * Only do this if the file is NOT marked as executable. If it is,\n   * we'll defer to the shebang inside the file itself.\n   *\n   * If we're on Windows, we ALWAYS take the handler, since it's hard to\n   * verify if registry associations have or have not been set up for this\n   * file type, so we'll assume the worst and take control.\n   */\n  public guessExecutable(app: string): Promise<string> {\n    return guessExecutable(app, this.debugFn);\n  }\n\n  /**\n   * If we don't have region/account defined in context, we fall back to the default SDK behavior\n   * where region is retrieved from ~/.aws/config and account is based on default credentials provider\n   * chain and then STS is queried.\n   *\n   * This is done opportunistically: for example, if we can't access STS for some reason or the region\n   * is not configured, the context value will be 'null' and there could failures down the line. In\n   * some cases, synthesis does not require region/account information at all, so that might be perfectly\n   * fine in certain scenarios.\n   */\n  public async defaultEnvVars(): Promise<Env> {\n    const debugFn = (msg: string) => this.ioHelper.notify(IO.CDK_ASSEMBLY_I0010.msg(msg));\n    const env = this.options.resolveDefaultAppEnv ? await prepareDefaultEnvironment(this.sdkProvider, debugFn) : {};\n\n    env[cxapi.OUTDIR_ENV] = this.outdir;\n    await debugFn(format('outdir:', this.outdir));\n\n    // CLI version information\n    env[cxapi.CLI_ASM_VERSION_ENV] = cxschema.Manifest.version();\n    env[cxapi.CLI_VERSION_ENV] = versionNumber();\n\n    await debugFn(format('env:', env));\n    return env;\n  }\n\n  /**\n   * Run code from a different working directory\n   */\n  public async changeDir<T>(block: () => Promise<T>, workingDir?: string) {\n    const originalWorkingDir = process.cwd();\n    try {\n      if (workingDir) {\n        process.chdir(workingDir);\n      }\n\n      return await block();\n    } finally {\n      if (workingDir) {\n        process.chdir(originalWorkingDir);\n      }\n    }\n  }\n}\n\n/**\n * Serializes the given context to a set if environment variables environment variables\n *\n * Needs to know the size of the rest of the env because that's necessary to do\n * an overflow computation on Windows. This function will mutate the given\n * environment in-place. It should be called as the very last operation on the\n * environment, because afterwards is might be at the maximum size.\n *\n * This *would* have returned an `IAsyncDisposable` but that requires messing\n * with TypeScript type definitions to use it in aws-cdk, so returning an\n * explicit cleanup function is easier.\n *\n * `completeness` indicates whether this `env` block represents the full `env`\n * that will be passed to a subprocess, or whether it will be mixed into\n * `process.env` later.\n */\nexport function writeContextToEnv(env: Env, context: Context, completeness: 'add-process-env-later' | 'env-is-complete') {\n  let contextOverflowLocation = null;\n\n  // On Windows, all envvars together must fit in a 32k block (<https://devblogs.microsoft.com/oldnewthing/20100203-00>)\n  // On Linux, a single entry may not exceed 131k; but we're treating it as all together because that's safe\n  // and it's a single execution path for both platforms.\n  const envVariableSizeLimit = os.platform() === 'win32' ? 32760 : 131072;\n\n  const completeEnv = { ...completeness === 'add-process-env-later' ? process.env : {}, ...env };\n  const [smallContext, overflow] = splitBySize(context, spaceAvailableForContext(completeEnv, envVariableSizeLimit));\n\n  // Store the safe part in the environment variable\n  env[cxapi.CONTEXT_ENV] = JSON.stringify(smallContext);\n\n  // If there was any overflow, write it to a temporary file\n  if (Object.keys(overflow ?? {}).length > 0) {\n    const contextDir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdk-context'));\n    contextOverflowLocation = path.join(contextDir, 'context-overflow.json');\n    fs.writeJSONSync(contextOverflowLocation, overflow);\n    env[cxapi.CONTEXT_OVERFLOW_LOCATION_ENV] = contextOverflowLocation;\n  }\n\n  return async () => {\n    if (contextOverflowLocation) {\n      await fs.promises.rm(path.dirname(contextOverflowLocation), { recursive: true, force: true });\n    }\n  };\n}\n\n/**\n * Checks if a given assembly supports context overflow, warn otherwise.\n *\n * @param assembly - the assembly to check\n */\nasync function checkContextOverflowSupport(assembly: CloudAssembly, ioHelper: IoHelper): Promise<void> {\n  const traceFn = (msg: string) => ioHelper.defaults.trace(msg);\n  const tree = await loadTree(assembly, traceFn);\n\n  // We're dealing with an old version of the framework here. It is unaware of the temporary\n  // file, which means that it will ignore the context overflow.\n  if (!frameworkSupportsContextOverflow(tree)) {\n    await ioHelper.notify(IO.CDK_ASSEMBLY_W0010.msg('Part of the context could not be sent to the application. Please update the AWS CDK library to the latest version.'));\n  }\n}\n\n/**\n * Checks if the framework supports context overflow\n */\nexport function frameworkSupportsContextOverflow(tree: ConstructTreeNode | undefined): boolean {\n  return !some(tree, node => {\n    const fqn = node.constructInfo?.fqn;\n    const version = node.constructInfo?.version;\n    return (\n      fqn === 'aws-cdk-lib.App' // v2 app\n      && version !== '0.0.0' // ignore developer builds\n      && version != null && lte(version, '2.38.0') // last version not supporting large context\n    ) // v2\n    || fqn === '@aws-cdk/core.App'; // v1 app => not supported\n  });\n}\n\n/**\n * Safely create an assembly from a cloud assembly directory\n */\nexport async function assemblyFromDirectory(assemblyDir: string, ioHelper: IoHelper, loadOptions: LoadAssemblyOptions = {}) {\n  try {\n    const assembly = new CloudAssembly(assemblyDir, {\n      skipVersionCheck: !(loadOptions.checkVersion ?? true),\n      skipEnumCheck: !(loadOptions.checkEnums ?? true),\n      // We sort as we deploy\n      topoSort: false,\n    });\n    await checkContextOverflowSupport(assembly, ioHelper);\n    return assembly;\n  } catch (err: any) {\n    if (err.message.includes(cxschema.VERSION_MISMATCH)) {\n      // this means the CLI version is too old.\n      // we instruct the user to upgrade.\n      const message = 'This AWS CDK Toolkit is not compatible with the AWS CDK library used by your application. Please upgrade to the latest version.';\n      await ioHelper.notify(IO.CDK_ASSEMBLY_E1111.msg(message, { error: err }));\n      throw new ToolkitError(`${message}\\n(${err.message}`);\n    }\n    throw err;\n  }\n}\n\nexport function settingsFromSynthOptions(synthOpts: AppSynthOptions = {}): Settings {\n  return new Settings({\n    debug: false,\n    pathMetadata: true,\n    versionReporting: true,\n    assetMetadata: true,\n    assetStaging: true,\n    ...synthOpts,\n  }, true);\n}\n\n/**\n * Turn synthesis options into context/environment variables that will go to the CDK app\n *\n * These are parameters that control the synthesis operation, configurable by the user\n * from the outside of the app.\n */\nexport function parametersFromSynthOptions(synthOptions?: AppSynthOptions) {\n  return synthParametersFromSettings(settingsFromSynthOptions(synthOptions ?? {}));\n}\n"]}
294
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"prepare-source.js","sourceRoot":"","sources":["prepare-source.ts"],"names":[],"mappings":"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","sourcesContent":["import '../../../private/dispose-polyfill';\nimport * as os from 'node:os';\nimport * as path from 'node:path';\nimport { format } from 'node:util';\nimport { CloudAssembly } from '@aws-cdk/cloud-assembly-api';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as fs from 'fs-extra';\nimport { lte } from 'semver';\nimport type { ToolkitServices } from '../../../toolkit/private';\nimport { ToolkitError } from '../../../toolkit/toolkit-error';\nimport { splitBySize, versionNumber } from '../../../util';\nimport type { SdkProvider } from '../../aws-auth/private';\nimport type { IoHelper } from '../../io/private';\nimport { IO } from '../../io/private';\nimport type { IReadLock, IWriteLock } from '../../rwlock';\nimport { RWLock } from '../../rwlock';\nimport { Settings } from '../../settings';\nimport type { ConstructTreeNode } from '../../tree';\nimport { loadTree, some } from '../../tree';\nimport type { Context, Env } from '../environment';\nimport { prepareDefaultEnvironment, spaceAvailableForContext, guessExecutable, synthParametersFromSettings } from '../environment';\nimport type { AppSynthOptions, LoadAssemblyOptions } from '../source-builder';\n\nexport interface ExecutionEnvironmentOptions {\n  /**\n   * The directory the cloud assembly will be written to.\n   *\n   * @default - use a temporary directory as output directory, this will be cleaned up when this object is disposed\n   */\n  readonly outdir?: string;\n\n  /**\n   * Resolve and add environment variables for the app's default environment.\n   *\n   * This will make a call to STS, which is not always desirable e.g. if the env is explicitly specified.\n   */\n  readonly resolveDefaultAppEnv: boolean;\n}\n\nexport class ExecutionEnvironment implements AsyncDisposable {\n  /**\n   * Create an ExecutionEnvironment\n   *\n   * An ExecutionEnvironment holds a writer lock on the given directory which will\n   * be cleaned up when the object is disposed.\n   *\n   * A temporary directory will be created if none is supplied, which will be cleaned\n   * up when this object is disposed.\n   *\n   * If `markSuccessful()` is called, the writer lock is converted to a reader lock\n   * and temporary directories will not be cleaned up anymore.\n   */\n  public static async create(services: ToolkitServices, options: ExecutionEnvironmentOptions) {\n    let outDirIsTemporary = false;\n    let dir = options.outdir;\n    if (!dir) {\n      outDirIsTemporary = true;\n      dir = fs.mkdtempSync(path.join(fs.realpathSync(os.tmpdir()), 'cdk.out'));\n    }\n    const lock = await new RWLock(dir).acquireWrite();\n\n    const opts = {\n      outdir: dir,\n      resolveDefaultAppEnv: options.resolveDefaultAppEnv,\n    };\n\n    return new ExecutionEnvironment(services, opts, {\n      lock,\n      outDirIsTemporary,\n    });\n  }\n\n  /**\n   * Should the outdir be disposed of.\n   */\n  public get shouldDisposeOutDir(): boolean {\n    return this.shouldClean;\n  }\n\n  /**\n   * The directory the cloud assembly will be written to.\n   */\n  public readonly outdir: string;\n\n  private readonly options: Required<ExecutionEnvironmentOptions>;\n  private readonly ioHelper: IoHelper;\n  private readonly sdkProvider: SdkProvider;\n  private readonly debugFn: (msg: string) => Promise<void>;\n  private lock: IWriteLock | undefined;\n  private shouldClean: boolean;\n\n  private constructor(\n    services: ToolkitServices,\n    options: Required<ExecutionEnvironmentOptions>,\n    { lock, outDirIsTemporary }: {\n      readonly outDirIsTemporary: boolean;\n      readonly lock: IWriteLock;\n    },\n  ) {\n    this.ioHelper = services.ioHelper;\n    this.sdkProvider = services.sdkProvider;\n    this.debugFn = (msg: string) => this.ioHelper.defaults.debug(msg);\n    this.lock = lock;\n    this.shouldClean = outDirIsTemporary;\n    this.outdir = options.outdir;\n    this.options = options;\n  }\n\n  public async [Symbol.asyncDispose]() {\n    await this.lock?.release();\n\n    if (this.shouldDisposeOutDir) {\n      await fs.rm(this.outdir, { recursive: true, force: true });\n    }\n  }\n\n  /**\n   * Mark the execution as successful, which stops the writer lock from being released upon disposal\n   */\n  public async markSuccessful() {\n    if (!this.lock) {\n      throw new TypeError('Cannot mark successful more than once');\n    }\n    const readLock = await this.lock.convertToReaderLock();\n    this.lock = undefined;\n    this.shouldClean = false;\n    return { readLock };\n  }\n\n  /**\n   * Begin an execution in this environment\n   *\n   * This will acquire a write lock on the given environment. The write lock\n   * will be released automatically when the return object is disposed, unless it\n   * is converted to a reader lock.\n   */\n  public async beginExecution(): Promise<{ writeToReadLock(): Promise<IReadLock> } & AsyncDisposable> {\n    const lock = await new RWLock(this.outdir).acquireWrite();\n\n    let converted = false;\n    return {\n      async writeToReadLock() {\n        converted = true;\n        return lock.convertToReaderLock();\n      },\n      [Symbol.asyncDispose]: async () => {\n        // Release if not converted\n        if (!converted) {\n          await lock.release();\n        }\n      },\n    };\n  }\n\n  /**\n   * Guess the executable from the command-line argument\n   *\n   * Only do this if the file is NOT marked as executable. If it is,\n   * we'll defer to the shebang inside the file itself.\n   *\n   * If we're on Windows, we ALWAYS take the handler, since it's hard to\n   * verify if registry associations have or have not been set up for this\n   * file type, so we'll assume the worst and take control.\n   */\n  public guessExecutable(app: string): Promise<string> {\n    return guessExecutable(app, this.debugFn);\n  }\n\n  /**\n   * If we don't have region/account defined in context, we fall back to the default SDK behavior\n   * where region is retrieved from ~/.aws/config and account is based on default credentials provider\n   * chain and then STS is queried.\n   *\n   * This is done opportunistically: for example, if we can't access STS for some reason or the region\n   * is not configured, the context value will be 'null' and there could failures down the line. In\n   * some cases, synthesis does not require region/account information at all, so that might be perfectly\n   * fine in certain scenarios.\n   */\n  public async defaultEnvVars(): Promise<Env> {\n    const debugFn = (msg: string) => this.ioHelper.notify(IO.CDK_ASSEMBLY_I0010.msg(msg));\n    const env = this.options.resolveDefaultAppEnv ? await prepareDefaultEnvironment(this.sdkProvider, debugFn) : {};\n\n    env[cxapi.OUTDIR_ENV] = this.outdir;\n    await debugFn(format('outdir:', this.outdir));\n\n    // CLI version information\n    env[cxapi.CLI_ASM_VERSION_ENV] = cxschema.Manifest.version();\n    env[cxapi.CLI_VERSION_ENV] = versionNumber();\n\n    await debugFn(format('env:', env));\n    return env;\n  }\n\n  /**\n   * Run code from a different working directory\n   */\n  public async changeDir<T>(block: () => Promise<T>, workingDir?: string) {\n    const originalWorkingDir = process.cwd();\n    try {\n      if (workingDir) {\n        process.chdir(workingDir);\n      }\n\n      return await block();\n    } finally {\n      if (workingDir) {\n        process.chdir(originalWorkingDir);\n      }\n    }\n  }\n}\n\n/**\n * Serializes the given context to a set if environment variables environment variables\n *\n * Needs to know the size of the rest of the env because that's necessary to do\n * an overflow computation on Windows. This function will mutate the given\n * environment in-place. It should be called as the very last operation on the\n * environment, because afterwards is might be at the maximum size.\n *\n * This *would* have returned an `IAsyncDisposable` but that requires messing\n * with TypeScript type definitions to use it in aws-cdk, so returning an\n * explicit cleanup function is easier.\n *\n * `completeness` indicates whether this `env` block represents the full `env`\n * that will be passed to a subprocess, or whether it will be mixed into\n * `process.env` later.\n */\nexport function writeContextToEnv(env: Env, context: Context, completeness: 'add-process-env-later' | 'env-is-complete') {\n  let contextOverflowLocation = null;\n\n  // On Windows, all envvars together must fit in a 32k block (<https://devblogs.microsoft.com/oldnewthing/20100203-00>)\n  // On Linux, a single entry may not exceed 131k; but we're treating it as all together because that's safe\n  // and it's a single execution path for both platforms.\n  const envVariableSizeLimit = os.platform() === 'win32' ? 32760 : 131072;\n\n  const completeEnv = { ...completeness === 'add-process-env-later' ? process.env : {}, ...env };\n  const [smallContext, overflow] = splitBySize(context, spaceAvailableForContext(completeEnv, envVariableSizeLimit));\n\n  // Store the safe part in the environment variable\n  env[cxapi.CONTEXT_ENV] = JSON.stringify(smallContext);\n\n  // If there was any overflow, write it to a temporary file\n  if (Object.keys(overflow ?? {}).length > 0) {\n    const contextDir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdk-context'));\n    contextOverflowLocation = path.join(contextDir, 'context-overflow.json');\n    fs.writeJSONSync(contextOverflowLocation, overflow);\n    env[cxapi.CONTEXT_OVERFLOW_LOCATION_ENV] = contextOverflowLocation;\n  }\n\n  return async () => {\n    if (contextOverflowLocation) {\n      await fs.promises.rm(path.dirname(contextOverflowLocation), { recursive: true, force: true });\n    }\n  };\n}\n\n/**\n * Checks if a given assembly supports context overflow, warn otherwise.\n *\n * @param assembly - the assembly to check\n */\nasync function checkContextOverflowSupport(tree: ConstructTreeNode | undefined, ioHelper: IoHelper): Promise<void> {\n  // We're dealing with an old version of the framework here. It is unaware of the temporary\n  // file, which means that it will ignore the context overflow.\n  if (!frameworkSupportsContextOverflow(tree)) {\n    await ioHelper.notify(IO.CDK_ASSEMBLY_W0010.msg('Part of the context could not be sent to the application. Please update the AWS CDK library to the latest version.'));\n  }\n}\n\n/**\n * Find the `aws-cdk-lib` library version by inspecting construct tree debug information\n */\nexport function findConstructLibraryVersion(tree: ConstructTreeNode | undefined): string | undefined {\n  let ret: string | undefined = undefined;\n\n  some(tree, node => {\n    const fqn = node.constructInfo?.fqn;\n    const version = node.constructInfo?.version;\n    if (fqn?.startsWith('aws-cdk-lib.') || fqn?.startsWith('@aws-cdk/core.')) {\n      ret = version;\n      return true;\n    }\n    return false;\n  });\n\n  return ret !== '0.0.0' ? ret : undefined;\n}\n\n/**\n * Checks if the framework supports context overflow\n */\nexport function frameworkSupportsContextOverflow(tree: ConstructTreeNode | undefined): boolean {\n  const version = findConstructLibraryVersion(tree);\n\n  return !version || !lte(version, '2.38.0');\n}\n\n/**\n * Safely create an assembly from a cloud assembly directory\n */\nexport async function assemblyFromDirectory(assemblyDir: string, ioHelper: IoHelper, loadOptions: LoadAssemblyOptions = {}) {\n  try {\n    const assembly = new CloudAssembly(assemblyDir, {\n      skipVersionCheck: !(loadOptions.checkVersion ?? true),\n      skipEnumCheck: !(loadOptions.checkEnums ?? true),\n      // We sort as we deploy\n      topoSort: false,\n    });\n\n    const tree = await loadTree(assembly, ioHelper.defaults.trace.bind(ioHelper.defaults));\n    await checkContextOverflowSupport(tree, ioHelper);\n\n    return assembly;\n  } catch (err: any) {\n    if (err.message.includes(cxschema.VERSION_MISMATCH)) {\n      // this means the CLI version is too old.\n      // we instruct the user to upgrade.\n      const message = 'This AWS CDK Toolkit is not compatible with the AWS CDK library used by your application. Please upgrade to the latest version.';\n      await ioHelper.notify(IO.CDK_ASSEMBLY_E1111.msg(message, { error: err }));\n      throw new ToolkitError(`${message}\\n(${err.message}`);\n    }\n    throw err;\n  }\n}\n\nexport function settingsFromSynthOptions(synthOpts: AppSynthOptions = {}): Settings {\n  return new Settings({\n    debug: false,\n    pathMetadata: true,\n    versionReporting: true,\n    assetMetadata: true,\n    assetStaging: true,\n    ...synthOpts,\n  }, true);\n}\n\n/**\n * Turn synthesis options into context/environment variables that will go to the CDK app\n *\n * These are parameters that control the synthesis operation, configurable by the user\n * from the outside of the app.\n */\nexport function parametersFromSynthOptions(synthOptions?: AppSynthOptions) {\n  return synthParametersFromSettings(settingsFromSynthOptions(synthOptions ?? {}));\n}\n"]}
@@ -112,7 +112,7 @@ class DiffFormatter {
112
112
  // detect and filter out mangled characters from the diff
113
113
  if (diff.differenceCount && !options.strict) {
114
114
  const mangledNewTemplate = JSON.parse((0, cloudformation_diff_1.mangleLikeCloudFormation)(JSON.stringify(this.newTemplate.template)));
115
- const mangledDiff = (0, cloudformation_diff_1.fullDiff)(this.oldTemplate, mangledNewTemplate, this.changeSet);
115
+ const mangledDiff = (0, cloudformation_diff_1.fullDiff)(oldTemplate, mangledNewTemplate, this.changeSet);
116
116
  filteredChangesCount = Math.max(0, diff.differenceCount - mangledDiff.differenceCount);
117
117
  if (filteredChangesCount > 0) {
118
118
  diff = mangledDiff;
@@ -237,4 +237,4 @@ function obscureDiff(diff) {
237
237
  });
238
238
  }
239
239
  }
240
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diff-formatter.js","sourceRoot":"","sources":["diff-formatter.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'node:util';\nimport type * as cxapi from '@aws-cdk/cloud-assembly-api';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport {\n  formatDifferences,\n  formatSecurityChanges,\n  fullDiff,\n  mangleLikeCloudFormation,\n  type ResourceDifference,\n  type TemplateDiff,\n} from '@aws-cdk/cloudformation-diff';\nimport * as chalk from 'chalk';\nimport { PermissionChangeType } from '../../payloads';\nimport type { NestedStackTemplates } from '../cloudformation';\nimport { StringWriteStream } from '../streams';\n\n/**\n * Output of formatSecurityDiff\n */\ninterface FormatSecurityDiffOutput {\n  /**\n   * Complete formatted security diff\n   */\n  readonly formattedDiff: string;\n\n  /**\n   * The type of permission changes in the security diff.\n   * The IoHost will use this to decide whether or not to print.\n   */\n  readonly permissionChangeType: PermissionChangeType;\n}\n\n/**\n * Output of formatStackDiff\n */\ninterface FormatStackDiffOutput {\n  /**\n   * Number of stacks with diff changes\n   */\n  readonly numStacksWithChanges: number;\n\n  /**\n   * Complete formatted diff\n   */\n  readonly formattedDiff: string;\n}\n\n/**\n * Props for the Diff Formatter\n */\ninterface DiffFormatterProps {\n  /**\n   * The relevant information for the Template that is being diffed.\n   * Includes the old/current state of the stack as well as the new state.\n   */\n  readonly templateInfo: TemplateInfo;\n}\n\n/**\n * Properties specific to formatting the stack diff\n */\ninterface FormatStackDiffOptions {\n  /**\n   * do not filter out AWS::CDK::Metadata or Rules\n   *\n   * @default false\n   */\n  readonly strict?: boolean;\n\n  /**\n   * lines of context to use in arbitrary JSON diff\n   *\n   * @default 3\n   */\n  readonly contextLines?: number;\n\n  /**\n   * silences \\'There were no differences\\' messages\n   *\n   * @default false\n   */\n  readonly quiet?: boolean;\n}\n\ninterface ReusableStackDiffOptions extends FormatStackDiffOptions {\n}\n\n/**\n * Information on a template's old/new state\n * that is used for diff.\n */\nexport interface TemplateInfo {\n  /**\n   * The old/existing template\n   */\n  readonly oldTemplate: any;\n\n  /**\n   * The new template\n   */\n  readonly newTemplate: cxapi.CloudFormationStackArtifact;\n\n  /**\n   * A CloudFormation ChangeSet to help the diff operation.\n   * Probably created via `createDiffChangeSet`.\n   *\n   * @default undefined\n   */\n  readonly changeSet?: any;\n\n  /**\n   * Whether or not there are any imported resources\n   *\n   * @default false\n   */\n  readonly isImport?: boolean;\n\n  /**\n   * Any nested stacks included in the template\n   *\n   * @default {}\n   */\n  readonly nestedStacks?: {\n    [nestedStackLogicalId: string]: NestedStackTemplates;\n  };\n\n  /**\n   * Mappings of old locations to new locations. If these are provided,\n   * for all resources that were moved, their corresponding addition\n   * and removal lines will be augmented with the location they were\n   * moved fom and to, respectively.\n   */\n  readonly mappings?: Record<string, string>;\n}\n\n/**\n * Class for formatting the diff output\n */\nexport class DiffFormatter {\n  private readonly oldTemplate: any;\n  private readonly newTemplate: cxapi.CloudFormationStackArtifact;\n  private readonly stackName: string;\n  private readonly changeSet?: any;\n  private readonly nestedStacks: { [nestedStackLogicalId: string]: NestedStackTemplates } | undefined;\n  private readonly isImport: boolean;\n  private readonly mappings: Record<string, string>;\n\n  /**\n   * Stores the TemplateDiffs that get calculated in this DiffFormatter,\n   * indexed by the stack name.\n   */\n  private _diffs: { [name: string]: TemplateDiff } = {};\n\n  constructor(props: DiffFormatterProps) {\n    this.oldTemplate = props.templateInfo.oldTemplate;\n    this.newTemplate = props.templateInfo.newTemplate;\n    this.stackName = props.templateInfo.newTemplate.displayName ?? props.templateInfo.newTemplate.stackName;\n    this.changeSet = props.templateInfo.changeSet;\n    this.nestedStacks = props.templateInfo.nestedStacks;\n    this.isImport = props.templateInfo.isImport ?? false;\n    this.mappings = props.templateInfo.mappings ?? {};\n  }\n\n  public get diffs() {\n    return this._diffs;\n  }\n\n  /**\n   * Get or creates the diff of a stack.\n   * If it creates the diff, it stores the result in a map for\n   * easier retrieval later.\n   */\n  private diff(stackName?: string, oldTemplate?: any, mappings: Record<string, string> = {}) {\n    const realStackName = stackName ?? this.stackName;\n\n    if (!this._diffs[realStackName]) {\n      const templateDiff = fullDiff(\n        oldTemplate ?? this.oldTemplate,\n        this.newTemplate.template,\n        this.changeSet,\n        this.isImport,\n      );\n\n      const setMove = (change: ResourceDifference, direction: 'from' | 'to', location?: string)=> {\n        if (location != null) {\n          const [sourceStackName, sourceLogicalId] = location.split('.');\n          change.move = {\n            direction,\n            stackName: sourceStackName,\n            resourceLogicalId: sourceLogicalId,\n          };\n        }\n      };\n\n      templateDiff.resources.forEachDifference((id, change) => {\n        const location = `${realStackName}.${id}`;\n        if (change.isAddition && Object.values(mappings).includes(location)) {\n          setMove(change, 'from', Object.keys(mappings).find(k => mappings[k] === location));\n        } else if (change.isRemoval && Object.keys(mappings).includes(location)) {\n          setMove(change, 'to', mappings[location]);\n        }\n      });\n\n      this._diffs[realStackName] = templateDiff;\n    }\n    return this._diffs[realStackName];\n  }\n\n  /**\n   * Return whether the diff has security-impacting changes that need confirmation.\n   *\n   * If no stackName is given, then the root stack name is used.\n   */\n  private permissionType(): PermissionChangeType {\n    const diff = this.diff();\n\n    if (diff.permissionsBroadened) {\n      return PermissionChangeType.BROADENING;\n    } else if (diff.permissionsAnyChanges) {\n      return PermissionChangeType.NON_BROADENING;\n    } else {\n      return PermissionChangeType.NONE;\n    }\n  }\n\n  /**\n   * Format the stack diff\n   */\n  public formatStackDiff(options: FormatStackDiffOptions = {}): FormatStackDiffOutput {\n    return this.formatStackDiffHelper(\n      this.oldTemplate,\n      this.stackName,\n      this.nestedStacks,\n      options,\n      this.mappings,\n    );\n  }\n\n  private formatStackDiffHelper(\n    oldTemplate: any,\n    stackName: string,\n    nestedStackTemplates: { [nestedStackLogicalId: string]: NestedStackTemplates } | undefined,\n    options: ReusableStackDiffOptions,\n    mappings: Record<string, string> = {},\n  ) {\n    let diff = this.diff(stackName, oldTemplate, mappings);\n\n    // The stack diff is formatted via `Formatter`, which takes in a stream\n    // and sends its output directly to that stream. To facilitate use of the\n    // global CliIoHost, we create our own stream to capture the output of\n    // `Formatter` and return the output as a string for the consumer of\n    // `formatStackDiff` to decide what to do with it.\n    const stream = new StringWriteStream();\n\n    let numStacksWithChanges = 0;\n    let formattedDiff = '';\n    let filteredChangesCount = 0;\n    try {\n      // must output the stack name if there are differences, even if quiet\n      if (stackName && (!options.quiet || !diff.isEmpty)) {\n        stream.write(format(`Stack ${chalk.bold(stackName)}\\n`));\n      }\n\n      if (!options.quiet && this.isImport) {\n        stream.write('Parameters and rules created during migration do not affect resource configuration.\\n');\n      }\n\n      // detect and filter out mangled characters from the diff\n      if (diff.differenceCount && !options.strict) {\n        const mangledNewTemplate = JSON.parse(mangleLikeCloudFormation(JSON.stringify(this.newTemplate.template)));\n        const mangledDiff = fullDiff(this.oldTemplate, mangledNewTemplate, this.changeSet);\n        filteredChangesCount = Math.max(0, diff.differenceCount - mangledDiff.differenceCount);\n        if (filteredChangesCount > 0) {\n          diff = mangledDiff;\n        }\n      }\n\n      // filter out 'AWS::CDK::Metadata' resources from the template\n      // filter out 'CheckBootstrapVersion' rules from the template\n      if (!options.strict) {\n        obscureDiff(diff);\n      }\n\n      if (!diff.isEmpty) {\n        numStacksWithChanges++;\n\n        // formatDifferences updates the stream with the formatted stack diff\n        formatDifferences(stream, diff, {\n          ...logicalIdMapFromTemplate(this.oldTemplate),\n          ...buildLogicalToPathMap(this.newTemplate),\n        }, options.contextLines);\n      } else if (!options.quiet) {\n        stream.write(chalk.green('There were no differences\\n'));\n      }\n\n      if (filteredChangesCount > 0) {\n        stream.write(chalk.yellow(`Omitted ${filteredChangesCount} changes because they are likely mangled non-ASCII characters. Use --strict to print them.\\n`));\n      }\n    } finally {\n      // store the stream containing a formatted stack diff\n      formattedDiff = stream.toString();\n      stream.end();\n    }\n\n    for (const nestedStackLogicalId of Object.keys(nestedStackTemplates ?? {})) {\n      if (!nestedStackTemplates) {\n        break;\n      }\n      const nestedStack = nestedStackTemplates[nestedStackLogicalId];\n\n      (this.newTemplate as any)._template = nestedStack.generatedTemplate;\n      const nextDiff = this.formatStackDiffHelper(\n        nestedStack.deployedTemplate,\n        nestedStack.physicalName ?? nestedStackLogicalId,\n        nestedStack.nestedStackTemplates,\n        options,\n        this.mappings,\n      );\n      numStacksWithChanges += nextDiff.numStacksWithChanges;\n      formattedDiff += nextDiff.formattedDiff;\n    }\n\n    return {\n      numStacksWithChanges,\n      formattedDiff,\n    };\n  }\n\n  /**\n   * Format the security diff\n   */\n  public formatSecurityDiff(): FormatSecurityDiffOutput {\n    const diff = this.diff();\n\n    // The security diff is formatted via `Formatter`, which takes in a stream\n    // and sends its output directly to that stream. To faciliate use of the\n    // global CliIoHost, we create our own stream to capture the output of\n    // `Formatter` and return the output as a string for the consumer of\n    // `formatSecurityDiff` to decide what to do with it.\n    const stream = new StringWriteStream();\n\n    stream.write(format(`Stack ${chalk.bold(this.stackName)}\\n`));\n\n    try {\n      // formatSecurityChanges updates the stream with the formatted security diff\n      formatSecurityChanges(stream, diff, buildLogicalToPathMap(this.newTemplate));\n    } finally {\n      stream.end();\n    }\n    // store the stream containing a formatted stack diff\n    const formattedDiff = stream.toString();\n    return { formattedDiff, permissionChangeType: this.permissionType() };\n  }\n}\n\nfunction buildLogicalToPathMap(stack: cxapi.CloudFormationStackArtifact) {\n  const map: { [id: string]: string } = {};\n  for (const md of stack.findMetadataByType(cxschema.ArtifactMetadataEntryType.LOGICAL_ID)) {\n    map[md.data as string] = md.path;\n  }\n  return map;\n}\n\nfunction logicalIdMapFromTemplate(template: any) {\n  const ret: Record<string, string> = {};\n\n  for (const [logicalId, resource] of Object.entries(template.Resources ?? {})) {\n    const path = (resource as any)?.Metadata?.['aws:cdk:path'];\n    if (path) {\n      ret[logicalId] = path;\n    }\n  }\n  return ret;\n}\n\n/**\n * Remove any template elements that we don't want to show users.\n * This is currently:\n * - AWS::CDK::Metadata resource\n * - CheckBootstrapVersion Rule\n */\nfunction obscureDiff(diff: TemplateDiff) {\n  if (diff.unknown) {\n    // see https://github.com/aws/aws-cdk/issues/17942\n    diff.unknown = diff.unknown.filter(change => {\n      if (!change) {\n        return true;\n      }\n      if (change.newValue?.CheckBootstrapVersion) {\n        return false;\n      }\n      if (change.oldValue?.CheckBootstrapVersion) {\n        return false;\n      }\n      return true;\n    });\n  }\n\n  if (diff.resources) {\n    diff.resources = diff.resources.filter(change => {\n      if (!change) {\n        return true;\n      }\n      if (change.newResourceType === 'AWS::CDK::Metadata') {\n        return false;\n      }\n      if (change.oldResourceType === 'AWS::CDK::Metadata') {\n        return false;\n      }\n      return true;\n    });\n  }\n}\n"]}
240
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diff-formatter.js","sourceRoot":"","sources":["diff-formatter.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'node:util';\nimport type * as cxapi from '@aws-cdk/cloud-assembly-api';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport {\n  formatDifferences,\n  formatSecurityChanges,\n  fullDiff,\n  mangleLikeCloudFormation,\n  type ResourceDifference,\n  type TemplateDiff,\n} from '@aws-cdk/cloudformation-diff';\nimport * as chalk from 'chalk';\nimport { PermissionChangeType } from '../../payloads';\nimport type { NestedStackTemplates } from '../cloudformation';\nimport { StringWriteStream } from '../streams';\n\n/**\n * Output of formatSecurityDiff\n */\ninterface FormatSecurityDiffOutput {\n  /**\n   * Complete formatted security diff\n   */\n  readonly formattedDiff: string;\n\n  /**\n   * The type of permission changes in the security diff.\n   * The IoHost will use this to decide whether or not to print.\n   */\n  readonly permissionChangeType: PermissionChangeType;\n}\n\n/**\n * Output of formatStackDiff\n */\ninterface FormatStackDiffOutput {\n  /**\n   * Number of stacks with diff changes\n   */\n  readonly numStacksWithChanges: number;\n\n  /**\n   * Complete formatted diff\n   */\n  readonly formattedDiff: string;\n}\n\n/**\n * Props for the Diff Formatter\n */\ninterface DiffFormatterProps {\n  /**\n   * The relevant information for the Template that is being diffed.\n   * Includes the old/current state of the stack as well as the new state.\n   */\n  readonly templateInfo: TemplateInfo;\n}\n\n/**\n * Properties specific to formatting the stack diff\n */\ninterface FormatStackDiffOptions {\n  /**\n   * do not filter out AWS::CDK::Metadata or Rules\n   *\n   * @default false\n   */\n  readonly strict?: boolean;\n\n  /**\n   * lines of context to use in arbitrary JSON diff\n   *\n   * @default 3\n   */\n  readonly contextLines?: number;\n\n  /**\n   * silences \\'There were no differences\\' messages\n   *\n   * @default false\n   */\n  readonly quiet?: boolean;\n}\n\ninterface ReusableStackDiffOptions extends FormatStackDiffOptions {\n}\n\n/**\n * Information on a template's old/new state\n * that is used for diff.\n */\nexport interface TemplateInfo {\n  /**\n   * The old/existing template\n   */\n  readonly oldTemplate: any;\n\n  /**\n   * The new template\n   */\n  readonly newTemplate: cxapi.CloudFormationStackArtifact;\n\n  /**\n   * A CloudFormation ChangeSet to help the diff operation.\n   * Probably created via `createDiffChangeSet`.\n   *\n   * @default undefined\n   */\n  readonly changeSet?: any;\n\n  /**\n   * Whether or not there are any imported resources\n   *\n   * @default false\n   */\n  readonly isImport?: boolean;\n\n  /**\n   * Any nested stacks included in the template\n   *\n   * @default {}\n   */\n  readonly nestedStacks?: {\n    [nestedStackLogicalId: string]: NestedStackTemplates;\n  };\n\n  /**\n   * Mappings of old locations to new locations. If these are provided,\n   * for all resources that were moved, their corresponding addition\n   * and removal lines will be augmented with the location they were\n   * moved fom and to, respectively.\n   */\n  readonly mappings?: Record<string, string>;\n}\n\n/**\n * Class for formatting the diff output\n */\nexport class DiffFormatter {\n  private readonly oldTemplate: any;\n  private readonly newTemplate: cxapi.CloudFormationStackArtifact;\n  private readonly stackName: string;\n  private readonly changeSet?: any;\n  private readonly nestedStacks: { [nestedStackLogicalId: string]: NestedStackTemplates } | undefined;\n  private readonly isImport: boolean;\n  private readonly mappings: Record<string, string>;\n\n  /**\n   * Stores the TemplateDiffs that get calculated in this DiffFormatter,\n   * indexed by the stack name.\n   */\n  private _diffs: { [name: string]: TemplateDiff } = {};\n\n  constructor(props: DiffFormatterProps) {\n    this.oldTemplate = props.templateInfo.oldTemplate;\n    this.newTemplate = props.templateInfo.newTemplate;\n    this.stackName = props.templateInfo.newTemplate.displayName ?? props.templateInfo.newTemplate.stackName;\n    this.changeSet = props.templateInfo.changeSet;\n    this.nestedStacks = props.templateInfo.nestedStacks;\n    this.isImport = props.templateInfo.isImport ?? false;\n    this.mappings = props.templateInfo.mappings ?? {};\n  }\n\n  public get diffs() {\n    return this._diffs;\n  }\n\n  /**\n   * Get or creates the diff of a stack.\n   * If it creates the diff, it stores the result in a map for\n   * easier retrieval later.\n   */\n  private diff(stackName?: string, oldTemplate?: any, mappings: Record<string, string> = {}) {\n    const realStackName = stackName ?? this.stackName;\n\n    if (!this._diffs[realStackName]) {\n      const templateDiff = fullDiff(\n        oldTemplate ?? this.oldTemplate,\n        this.newTemplate.template,\n        this.changeSet,\n        this.isImport,\n      );\n\n      const setMove = (change: ResourceDifference, direction: 'from' | 'to', location?: string)=> {\n        if (location != null) {\n          const [sourceStackName, sourceLogicalId] = location.split('.');\n          change.move = {\n            direction,\n            stackName: sourceStackName,\n            resourceLogicalId: sourceLogicalId,\n          };\n        }\n      };\n\n      templateDiff.resources.forEachDifference((id, change) => {\n        const location = `${realStackName}.${id}`;\n        if (change.isAddition && Object.values(mappings).includes(location)) {\n          setMove(change, 'from', Object.keys(mappings).find(k => mappings[k] === location));\n        } else if (change.isRemoval && Object.keys(mappings).includes(location)) {\n          setMove(change, 'to', mappings[location]);\n        }\n      });\n\n      this._diffs[realStackName] = templateDiff;\n    }\n    return this._diffs[realStackName];\n  }\n\n  /**\n   * Return whether the diff has security-impacting changes that need confirmation.\n   *\n   * If no stackName is given, then the root stack name is used.\n   */\n  private permissionType(): PermissionChangeType {\n    const diff = this.diff();\n\n    if (diff.permissionsBroadened) {\n      return PermissionChangeType.BROADENING;\n    } else if (diff.permissionsAnyChanges) {\n      return PermissionChangeType.NON_BROADENING;\n    } else {\n      return PermissionChangeType.NONE;\n    }\n  }\n\n  /**\n   * Format the stack diff\n   */\n  public formatStackDiff(options: FormatStackDiffOptions = {}): FormatStackDiffOutput {\n    return this.formatStackDiffHelper(\n      this.oldTemplate,\n      this.stackName,\n      this.nestedStacks,\n      options,\n      this.mappings,\n    );\n  }\n\n  private formatStackDiffHelper(\n    oldTemplate: any,\n    stackName: string,\n    nestedStackTemplates: { [nestedStackLogicalId: string]: NestedStackTemplates } | undefined,\n    options: ReusableStackDiffOptions,\n    mappings: Record<string, string> = {},\n  ) {\n    let diff = this.diff(stackName, oldTemplate, mappings);\n\n    // The stack diff is formatted via `Formatter`, which takes in a stream\n    // and sends its output directly to that stream. To facilitate use of the\n    // global CliIoHost, we create our own stream to capture the output of\n    // `Formatter` and return the output as a string for the consumer of\n    // `formatStackDiff` to decide what to do with it.\n    const stream = new StringWriteStream();\n\n    let numStacksWithChanges = 0;\n    let formattedDiff = '';\n    let filteredChangesCount = 0;\n    try {\n      // must output the stack name if there are differences, even if quiet\n      if (stackName && (!options.quiet || !diff.isEmpty)) {\n        stream.write(format(`Stack ${chalk.bold(stackName)}\\n`));\n      }\n\n      if (!options.quiet && this.isImport) {\n        stream.write('Parameters and rules created during migration do not affect resource configuration.\\n');\n      }\n\n      // detect and filter out mangled characters from the diff\n      if (diff.differenceCount && !options.strict) {\n        const mangledNewTemplate = JSON.parse(mangleLikeCloudFormation(JSON.stringify(this.newTemplate.template)));\n        const mangledDiff = fullDiff(oldTemplate, mangledNewTemplate, this.changeSet);\n        filteredChangesCount = Math.max(0, diff.differenceCount - mangledDiff.differenceCount);\n        if (filteredChangesCount > 0) {\n          diff = mangledDiff;\n        }\n      }\n\n      // filter out 'AWS::CDK::Metadata' resources from the template\n      // filter out 'CheckBootstrapVersion' rules from the template\n      if (!options.strict) {\n        obscureDiff(diff);\n      }\n\n      if (!diff.isEmpty) {\n        numStacksWithChanges++;\n\n        // formatDifferences updates the stream with the formatted stack diff\n        formatDifferences(stream, diff, {\n          ...logicalIdMapFromTemplate(this.oldTemplate),\n          ...buildLogicalToPathMap(this.newTemplate),\n        }, options.contextLines);\n      } else if (!options.quiet) {\n        stream.write(chalk.green('There were no differences\\n'));\n      }\n\n      if (filteredChangesCount > 0) {\n        stream.write(chalk.yellow(`Omitted ${filteredChangesCount} changes because they are likely mangled non-ASCII characters. Use --strict to print them.\\n`));\n      }\n    } finally {\n      // store the stream containing a formatted stack diff\n      formattedDiff = stream.toString();\n      stream.end();\n    }\n\n    for (const nestedStackLogicalId of Object.keys(nestedStackTemplates ?? {})) {\n      if (!nestedStackTemplates) {\n        break;\n      }\n      const nestedStack = nestedStackTemplates[nestedStackLogicalId];\n\n      (this.newTemplate as any)._template = nestedStack.generatedTemplate;\n      const nextDiff = this.formatStackDiffHelper(\n        nestedStack.deployedTemplate,\n        nestedStack.physicalName ?? nestedStackLogicalId,\n        nestedStack.nestedStackTemplates,\n        options,\n        this.mappings,\n      );\n      numStacksWithChanges += nextDiff.numStacksWithChanges;\n      formattedDiff += nextDiff.formattedDiff;\n    }\n\n    return {\n      numStacksWithChanges,\n      formattedDiff,\n    };\n  }\n\n  /**\n   * Format the security diff\n   */\n  public formatSecurityDiff(): FormatSecurityDiffOutput {\n    const diff = this.diff();\n\n    // The security diff is formatted via `Formatter`, which takes in a stream\n    // and sends its output directly to that stream. To faciliate use of the\n    // global CliIoHost, we create our own stream to capture the output of\n    // `Formatter` and return the output as a string for the consumer of\n    // `formatSecurityDiff` to decide what to do with it.\n    const stream = new StringWriteStream();\n\n    stream.write(format(`Stack ${chalk.bold(this.stackName)}\\n`));\n\n    try {\n      // formatSecurityChanges updates the stream with the formatted security diff\n      formatSecurityChanges(stream, diff, buildLogicalToPathMap(this.newTemplate));\n    } finally {\n      stream.end();\n    }\n    // store the stream containing a formatted stack diff\n    const formattedDiff = stream.toString();\n    return { formattedDiff, permissionChangeType: this.permissionType() };\n  }\n}\n\nfunction buildLogicalToPathMap(stack: cxapi.CloudFormationStackArtifact) {\n  const map: { [id: string]: string } = {};\n  for (const md of stack.findMetadataByType(cxschema.ArtifactMetadataEntryType.LOGICAL_ID)) {\n    map[md.data as string] = md.path;\n  }\n  return map;\n}\n\nfunction logicalIdMapFromTemplate(template: any) {\n  const ret: Record<string, string> = {};\n\n  for (const [logicalId, resource] of Object.entries(template.Resources ?? {})) {\n    const path = (resource as any)?.Metadata?.['aws:cdk:path'];\n    if (path) {\n      ret[logicalId] = path;\n    }\n  }\n  return ret;\n}\n\n/**\n * Remove any template elements that we don't want to show users.\n * This is currently:\n * - AWS::CDK::Metadata resource\n * - CheckBootstrapVersion Rule\n */\nfunction obscureDiff(diff: TemplateDiff) {\n  if (diff.unknown) {\n    // see https://github.com/aws/aws-cdk/issues/17942\n    diff.unknown = diff.unknown.filter(change => {\n      if (!change) {\n        return true;\n      }\n      if (change.newValue?.CheckBootstrapVersion) {\n        return false;\n      }\n      if (change.oldValue?.CheckBootstrapVersion) {\n        return false;\n      }\n      return true;\n    });\n  }\n\n  if (diff.resources) {\n    diff.resources = diff.resources.filter(change => {\n      if (!change) {\n        return true;\n      }\n      if (change.newResourceType === 'AWS::CDK::Metadata') {\n        return false;\n      }\n      if (change.oldResourceType === 'AWS::CDK::Metadata') {\n        return false;\n      }\n      return true;\n    });\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> = Pick<Partial<T>, K> & Omit<T, K>;
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: VoidWhenEmpty<Omit<E, keyof SpanEnd>>): Promise<ElapsedTime>;
85
+ end(payload: SpanEndArguments<E>): Promise<ElapsedTime>;
72
86
  /**
73
- * End the span with a payload, overwriting
87
+ * End the span with a message and payload
74
88
  */
75
- end(payload: VoidWhenEmpty<Optional<E, keyof SpanEnd>>): Promise<ElapsedTime>;
89
+ end(message: string, payload: SpanEndArguments<E>): Promise<ElapsedTime>;
76
90
  /**
77
- * End the span with a message and payload
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
- end(message: string, payload: ForceEmpty<Optional<E, keyof SpanEnd>>): Promise<ElapsedTime>;
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":";;;AAAA,kCAAkC;AAClC,6BAA6B;AAI7B,wCAA2C;AAuF3C;;;;;;GAMG;AACH,MAAa,SAAS;IACH,UAAU,CAAuB;IACjC,QAAQ,CAAW;IAC5B,UAAU,CAA2C;IAE7D,YAAmB,QAAkB,EAAE,UAAgC,EAAE,UAAoD;QAC3H,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC;QAC7B,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC;IAC/B,CAAC;IAQM,KAAK,CAAC,KAAK,CAAC,CAAM,EAAE,CAAK;QAC9B,MAAM,IAAI,GAAG,IAAI,WAAW,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;QAC9E,MAAM,UAAU,GAAG,SAAS,CAAI,CAAC,EAAE,CAAC,CAAC,CAAC;QACtC,MAAM,QAAQ,GAAG,UAAU,CAAC,OAAO,IAAI,YAAY,IAAI,CAAC,UAAU,CAAC,IAAI,MAAM,CAAC;QAC9E,MAAM,YAAY,GAAG,UAAU,CAAC,OAAO,CAAC;QACxC,MAAM,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,YAAY,CAAC,CAAC,CAAC;QAErE,OAAO,IAAI,CAAC;IACd,CAAC;CACF;AA1BD,8BA0BC;AAED,MAAM,WAAW;IACC,QAAQ,CAAW;IAElB,UAAU,CAAuB;IACjC,QAAQ,CAAW;IACnB,MAAM,CAAS;IACf,SAAS,CAAS;IAClB,iBAAiB,CAAS;IAE3C,YAAmB,QAAkB,EAAE,UAAgC,EAAE,UAAoD;QAC3H,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC;QAC7B,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,EAAE,EAAE,CAAC;QACxB,IAAI,CAAC,SAAS,GAAG,IAAI,IAAI,EAAE,CAAC,OAAO,EAAE,CAAC;QACtC,IAAI,CAAC,iBAAiB,GAAG,qBAAqB,CAAC;QAC/C,IAAI,CAAC,QAAQ,GAAG,UAAU,CAAC,IAAI,CAAC,CAAC;IACnC,CAAC;IAED,IAAW,QAAQ;QACjB,OAAO,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC;IAChC,CAAC;IAEM,KAAK,CAAC,WAAW;QACtB,OAAO,IAAI,CAAC,IAAI,EAAE,CAAC;IACrB,CAAC;IACM,KAAK,CAAC,MAAM,CAAC,KAAoC,EAAE,OAAgB;QACxE,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC;QAC7B,MAAM,SAAS,GAAG,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,iBAAiB,EAAE,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,QAAQ,CAAC,KAAK,CAAC,CAAC;QAChH,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,EAAE;YACrC,QAAQ,EAAE,QAAQ,CAAC,IAAI;SACxB,CAAC,CAAC,CAAC;QACJ,OAAO,QAAQ,CAAC;IAClB,CAAC;IACM,KAAK,CAAC,MAAM,CAAC,GAA+B;QACjD,OAAO,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC;IAC5D,CAAC;IACM,KAAK,CAAC,GAAG,CAAC,CAAM,EAAE,CAA0C;QACjE,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC;QAE7B,MAAM,QAAQ,GAAG,SAAS,CAAyC,CAAC,EAAE,CAAC,CAAC,CAAC;QACzE,MAAM,MAAM,GAAG,QAAQ,CAAC,OAAO,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,iBAAiB,EAAE,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,QAAQ,CAAC,KAAK,CAAC,CAAC;QAC7G,MAAM,UAAU,GAAG,QAAQ,CAAC,OAAO,CAAC;QAEpC,MAAM,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,GAAG,CACvC,MAAM,EAAE;YACN,QAAQ,EAAE,QAAQ,CAAC,IAAI;YACvB,GAAG,UAAU;SACT,CAAC,CAAC,CAAC;QAEX,OAAO,QAAQ,CAAC;IAClB,CAAC;IAEM,KAAK,CAAC,eAAe,CAAI,GAAkC;QAChE,OAAO,IAAI,CAAC,QAAQ,CAAC,eAAe,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC;IACrE,CAAC;IAEO,IAAI;QACV,MAAM,WAAW,GAAG,IAAI,IAAI,EAAE,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC;QAC1D,OAAO;YACL,IAAI,EAAE,WAAW;YACjB,KAAK,EAAE,IAAA,iBAAU,EAAC,WAAW,CAAC;SAC/B,CAAC;IACJ,CAAC;CACF;AAED,SAAS,SAAS,CAAmB,KAAU,EAAE,MAAU;IACzD,MAAM,cAAc,GAAG,OAAO,KAAK,KAAK,QAAQ,CAAC;IAEjD,sGAAsG;IACtG,MAAM,OAAO,GAAG,CAAC,cAAc,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC;IAE/D,oEAAoE;IACpE,6EAA6E;IAC7E,MAAM,OAAO,GAAG,CAAC,cAAc,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC;IAE5D,OAAO;QACL,OAAO;QACP,OAAO;KACR,CAAC;AACJ,CAAC;AAED,SAAS,UAAU,CAAmB,IAAY,EAAE,OAAU;IAC5D,OAAO;QACL,GAAG,OAAO;QACV,IAAI;KACL,CAAC;AACJ,CAAC","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"]}