UNPKG

aws-cdk

Version:

AWS CDK CLI, the command line tool for CDK apps

151 lines 21.7 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.BackgroundStackRefresh = exports.ActiveAssetCache = void 0; exports.refreshStacks = refreshStacks; const api_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api"); const private_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private"); class ActiveAssetCache { constructor() { this.stacks = new Set(); } rememberStack(stackTemplate) { this.stacks.add(stackTemplate); } contains(asset) { for (const stack of this.stacks) { if (stack.includes(asset)) { return true; } } return false; } } exports.ActiveAssetCache = ActiveAssetCache; async function paginateSdkCall(cb) { let finished = false; let nextToken; while (!finished) { nextToken = await cb(nextToken); if (nextToken === undefined) { finished = true; } } } /** * Fetches all relevant stack templates from CloudFormation. It ignores the following stacks: * - stacks in DELETE_COMPLETE or DELETE_IN_PROGRESS stage * - stacks that are using a different bootstrap qualifier */ async function fetchAllStackTemplates(cfn, ioHelper, qualifier) { const stackNames = []; await paginateSdkCall(async (nextToken) => { const stacks = await cfn.listStacks({ NextToken: nextToken }); // We ignore stacks with these statuses because their assets are no longer live const ignoredStatues = ['CREATE_FAILED', 'DELETE_COMPLETE', 'DELETE_IN_PROGRESS', 'DELETE_FAILED', 'REVIEW_IN_PROGRESS']; stackNames.push(...(stacks.StackSummaries ?? []) .filter((s) => !ignoredStatues.includes(s.StackStatus)) .map((s) => s.StackId ?? s.StackName)); return stacks.NextToken; }); await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg(`Parsing through ${stackNames.length} stacks`)); const templates = []; for (const stack of stackNames) { let summary; summary = await cfn.getTemplateSummary({ StackName: stack, }); if (bootstrapFilter(summary.Parameters, qualifier)) { // This stack is definitely bootstrapped to a different qualifier so we can safely ignore it continue; } else { const template = await cfn.getTemplate({ StackName: stack, }); templates.push((template.TemplateBody ?? '') + JSON.stringify(summary?.Parameters)); } } await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg('Done parsing through stacks')); return templates; } /** * Filter out stacks that we KNOW are using a different bootstrap qualifier * This is mostly necessary for the integration tests that can run the same app (with the same assets) * under different qualifiers. * This is necessary because a stack under a different bootstrap could coincidentally reference the same hash * and cause a false negative (cause an asset to be preserved when its isolated) * This is intentionally done in a way where we ONLY filter out stacks that are meant for a different qualifier * because we are okay with false positives. */ function bootstrapFilter(parameters, qualifier) { const bootstrapVersion = parameters?.find((p) => p.ParameterKey === 'BootstrapVersion'); const splitBootstrapVersion = bootstrapVersion?.DefaultValue?.split('/'); // We find the qualifier in a specific part of the bootstrap version parameter return (qualifier && splitBootstrapVersion && splitBootstrapVersion.length == 4 && splitBootstrapVersion[2] != qualifier); } async function refreshStacks(cfn, ioHelper, activeAssets, qualifier) { try { const stacks = await fetchAllStackTemplates(cfn, ioHelper, qualifier); for (const stack of stacks) { activeAssets.rememberStack(stack); } } catch (err) { throw new api_1.ToolkitError(`Error refreshing stacks: ${err}`); } } /** * Class that controls scheduling of the background stack refresh */ class BackgroundStackRefresh { constructor(props) { this.props = props; this.queuedPromises = []; this.lastRefreshTime = Date.now(); } start() { // Since start is going to be called right after the first invocation of refreshStacks, // lets wait some time before beginning the background refresh. this.timeout = setTimeout(() => this.refresh(), 300000); // 5 minutes } async refresh() { const startTime = Date.now(); await refreshStacks(this.props.cfn, this.props.ioHelper, this.props.activeAssets, this.props.qualifier); this.justRefreshedStacks(); // If the last invocation of refreshStacks takes <5 minutes, the next invocation starts 5 minutes after the last one started. // If the last invocation of refreshStacks takes >5 minutes, the next invocation starts immediately. this.timeout = setTimeout(() => this.refresh(), Math.max(startTime + 300000 - Date.now(), 0)); } justRefreshedStacks() { this.lastRefreshTime = Date.now(); for (const p of this.queuedPromises.splice(0, this.queuedPromises.length)) { p(undefined); } } /** * Checks if the last successful background refresh happened within the specified time frame. * If the last refresh is older than the specified time frame, it returns a Promise that resolves * when the next background refresh completes or rejects if the refresh takes too long. */ noOlderThan(ms) { const horizon = Date.now() - ms; // The last refresh happened within the time frame if (this.lastRefreshTime >= horizon) { return Promise.resolve(); } // The last refresh happened earlier than the time frame // We will wait for the latest refresh to land or reject if it takes too long return Promise.race([ new Promise(resolve => this.queuedPromises.push(resolve)), new Promise((_, reject) => setTimeout(() => reject(new api_1.ToolkitError('refreshStacks took too long; the background thread likely threw an error')), ms)), ]); } stop() { clearTimeout(this.timeout); } } exports.BackgroundStackRefresh = BackgroundStackRefresh; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-refresh.js","sourceRoot":"","sources":["stack-refresh.ts"],"names":[],"mappings":"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","sourcesContent":["import type { ParameterDeclaration } from '@aws-sdk/client-cloudformation';\nimport { ToolkitError } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport { IO, type IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport type { ICloudFormationClient } from '../aws-auth';\n\nexport class ActiveAssetCache {\n  private readonly stacks: Set<string> = new Set();\n\n  public rememberStack(stackTemplate: string) {\n    this.stacks.add(stackTemplate);\n  }\n\n  public contains(asset: string): boolean {\n    for (const stack of this.stacks) {\n      if (stack.includes(asset)) {\n        return true;\n      }\n    }\n    return false;\n  }\n}\n\nasync function paginateSdkCall(cb: (nextToken?: string) => Promise<string | undefined>) {\n  let finished = false;\n  let nextToken: string | undefined;\n  while (!finished) {\n    nextToken = await cb(nextToken);\n    if (nextToken === undefined) {\n      finished = true;\n    }\n  }\n}\n\n/**\n * Fetches all relevant stack templates from CloudFormation. It ignores the following stacks:\n * - stacks in DELETE_COMPLETE or DELETE_IN_PROGRESS stage\n * - stacks that are using a different bootstrap qualifier\n */\nasync function fetchAllStackTemplates(cfn: ICloudFormationClient, ioHelper: IoHelper, qualifier?: string) {\n  const stackNames: string[] = [];\n  await paginateSdkCall(async (nextToken) => {\n    const stacks = await cfn.listStacks({ NextToken: nextToken });\n\n    // We ignore stacks with these statuses because their assets are no longer live\n    const ignoredStatues = ['CREATE_FAILED', 'DELETE_COMPLETE', 'DELETE_IN_PROGRESS', 'DELETE_FAILED', 'REVIEW_IN_PROGRESS'];\n    stackNames.push(\n      ...(stacks.StackSummaries ?? [])\n        .filter((s: any) => !ignoredStatues.includes(s.StackStatus))\n        .map((s: any) => s.StackId ?? s.StackName),\n    );\n\n    return stacks.NextToken;\n  });\n\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(`Parsing through ${stackNames.length} stacks`));\n\n  const templates: string[] = [];\n  for (const stack of stackNames) {\n    let summary;\n    summary = await cfn.getTemplateSummary({\n      StackName: stack,\n    });\n\n    if (bootstrapFilter(summary.Parameters, qualifier)) {\n      // This stack is definitely bootstrapped to a different qualifier so we can safely ignore it\n      continue;\n    } else {\n      const template = await cfn.getTemplate({\n        StackName: stack,\n      });\n\n      templates.push((template.TemplateBody ?? '') + JSON.stringify(summary?.Parameters));\n    }\n  }\n\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg('Done parsing through stacks'));\n\n  return templates;\n}\n\n/**\n * Filter out stacks that we KNOW are using a different bootstrap qualifier\n * This is mostly necessary for the integration tests that can run the same app (with the same assets)\n * under different qualifiers.\n * This is necessary because a stack under a different bootstrap could coincidentally reference the same hash\n * and cause a false negative (cause an asset to be preserved when its isolated)\n * This is intentionally done in a way where we ONLY filter out stacks that are meant for a different qualifier\n * because we are okay with false positives.\n */\nfunction bootstrapFilter(parameters?: ParameterDeclaration[], qualifier?: string) {\n  const bootstrapVersion = parameters?.find((p) => p.ParameterKey === 'BootstrapVersion');\n  const splitBootstrapVersion = bootstrapVersion?.DefaultValue?.split('/');\n  // We find the qualifier in a specific part of the bootstrap version parameter\n  return (qualifier &&\n          splitBootstrapVersion &&\n          splitBootstrapVersion.length == 4 &&\n          splitBootstrapVersion[2] != qualifier);\n}\n\nexport async function refreshStacks(cfn: ICloudFormationClient, ioHelper: IoHelper, activeAssets: ActiveAssetCache, qualifier?: string) {\n  try {\n    const stacks = await fetchAllStackTemplates(cfn, ioHelper, qualifier);\n    for (const stack of stacks) {\n      activeAssets.rememberStack(stack);\n    }\n  } catch (err) {\n    throw new ToolkitError(`Error refreshing stacks: ${err}`);\n  }\n}\n\n/**\n * Background Stack Refresh properties\n */\nexport interface BackgroundStackRefreshProps {\n  /**\n   * The CFN SDK handler\n   */\n  readonly cfn: ICloudFormationClient;\n\n  /**\n   * Used to send messages.\n   */\n  readonly ioHelper: IoHelper;\n\n  /**\n   * Active Asset storage\n   */\n  readonly activeAssets: ActiveAssetCache;\n\n  /**\n   * Stack bootstrap qualifier\n   */\n  readonly qualifier?: string;\n}\n\n/**\n * Class that controls scheduling of the background stack refresh\n */\nexport class BackgroundStackRefresh {\n  private timeout?: NodeJS.Timeout;\n  private lastRefreshTime: number;\n  private queuedPromises: Array<(value: unknown) => void> = [];\n\n  constructor(private readonly props: BackgroundStackRefreshProps) {\n    this.lastRefreshTime = Date.now();\n  }\n\n  public start() {\n    // Since start is going to be called right after the first invocation of refreshStacks,\n    // lets wait some time before beginning the background refresh.\n    this.timeout = setTimeout(() => this.refresh(), 300_000); // 5 minutes\n  }\n\n  private async refresh() {\n    const startTime = Date.now();\n\n    await refreshStacks(this.props.cfn, this.props.ioHelper, this.props.activeAssets, this.props.qualifier);\n    this.justRefreshedStacks();\n\n    // If the last invocation of refreshStacks takes <5 minutes, the next invocation starts 5 minutes after the last one started.\n    // If the last invocation of refreshStacks takes >5 minutes, the next invocation starts immediately.\n    this.timeout = setTimeout(() => this.refresh(), Math.max(startTime + 300_000 - Date.now(), 0));\n  }\n\n  private justRefreshedStacks() {\n    this.lastRefreshTime = Date.now();\n    for (const p of this.queuedPromises.splice(0, this.queuedPromises.length)) {\n      p(undefined);\n    }\n  }\n\n  /**\n   * Checks if the last successful background refresh happened within the specified time frame.\n   * If the last refresh is older than the specified time frame, it returns a Promise that resolves\n   * when the next background refresh completes or rejects if the refresh takes too long.\n   */\n  public noOlderThan(ms: number) {\n    const horizon = Date.now() - ms;\n\n    // The last refresh happened within the time frame\n    if (this.lastRefreshTime >= horizon) {\n      return Promise.resolve();\n    }\n\n    // The last refresh happened earlier than the time frame\n    // We will wait for the latest refresh to land or reject if it takes too long\n    return Promise.race([\n      new Promise(resolve => this.queuedPromises.push(resolve)),\n      new Promise((_, reject) => setTimeout(() => reject(new ToolkitError('refreshStacks took too long; the background thread likely threw an error')), ms)),\n    ]);\n  }\n\n  public stop() {\n    clearTimeout(this.timeout);\n  }\n}\n"]}