aws-cdk 2.1003.0 → 2.1004.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (68) hide show
  1. package/build-info.json +2 -2
  2. package/lib/api/aws-auth/sdk-logger.d.ts +3 -3
  3. package/lib/api/aws-auth/sdk-logger.js +5 -10
  4. package/lib/api/bootstrap/bootstrap-environment.d.ts +3 -3
  5. package/lib/api/bootstrap/bootstrap-environment.js +9 -9
  6. package/lib/api/bootstrap/deploy-bootstrap.d.ts +4 -4
  7. package/lib/api/bootstrap/deploy-bootstrap.js +11 -11
  8. package/lib/api/cxapp/cloud-assembly.d.ts +1 -16
  9. package/lib/api/cxapp/cloud-assembly.js +3 -3
  10. package/lib/api/deployments/asset-publishing.d.ts +5 -5
  11. package/lib/api/deployments/asset-publishing.js +7 -8
  12. package/lib/api/deployments/assets.d.ts +2 -2
  13. package/lib/api/deployments/assets.js +9 -9
  14. package/lib/api/deployments/checks.d.ts +2 -2
  15. package/lib/api/deployments/checks.js +4 -4
  16. package/lib/api/deployments/cloudformation.d.ts +7 -7
  17. package/lib/api/deployments/cloudformation.js +27 -27
  18. package/lib/api/deployments/deploy-stack.d.ts +3 -3
  19. package/lib/api/deployments/deploy-stack.js +52 -55
  20. package/lib/api/deployments/deployments.d.ts +9 -5
  21. package/lib/api/deployments/deployments.js +22 -24
  22. package/lib/api/deployments/hotswap-deployments.d.ts +2 -2
  23. package/lib/api/deployments/hotswap-deployments.js +12 -12
  24. package/lib/api/environment/environment-access.d.ts +3 -4
  25. package/lib/api/environment/environment-access.js +6 -7
  26. package/lib/api/environment/environment-resources.d.ts +5 -5
  27. package/lib/api/environment/environment-resources.js +12 -12
  28. package/lib/api/garbage-collection/garbage-collector.d.ts +3 -4
  29. package/lib/api/garbage-collection/garbage-collector.js +37 -38
  30. package/lib/api/garbage-collection/progress-printer.d.ts +3 -4
  31. package/lib/api/garbage-collection/progress-printer.js +5 -6
  32. package/lib/api/garbage-collection/stack-refresh.d.ts +3 -3
  33. package/lib/api/garbage-collection/stack-refresh.js +7 -7
  34. package/lib/api/logs/find-cloudwatch-logs.d.ts +2 -2
  35. package/lib/api/logs/find-cloudwatch-logs.js +5 -5
  36. package/lib/api/logs/logs-monitor.d.ts +27 -4
  37. package/lib/api/logs/logs-monitor.js +51 -30
  38. package/lib/api/resource-import/importer.d.ts +3 -5
  39. package/lib/api/resource-import/importer.js +12 -13
  40. package/lib/api/resource-import/migrator.d.ts +3 -5
  41. package/lib/api/resource-import/migrator.js +7 -9
  42. package/lib/api/stack-events/stack-activity-monitor.d.ts +5 -63
  43. package/lib/api/stack-events/stack-activity-monitor.js +7 -8
  44. package/lib/api/stack-events/stack-progress-monitor.d.ts +1 -14
  45. package/lib/api/stack-events/stack-progress-monitor.js +1 -1
  46. package/lib/api/toolkit-info.d.ts +2 -2
  47. package/lib/api/toolkit-info.js +5 -5
  48. package/lib/api/work-graph/work-graph-builder.d.ts +3 -4
  49. package/lib/api/work-graph/work-graph-builder.js +5 -6
  50. package/lib/api/work-graph/work-graph.d.ts +3 -8
  51. package/lib/api/work-graph/work-graph.js +6 -7
  52. package/lib/cli/activity-printer/base.d.ts +3 -4
  53. package/lib/cli/activity-printer/base.js +6 -5
  54. package/lib/cli/activity-printer/current.js +1 -1
  55. package/lib/cli/activity-printer/history.d.ts +1 -1
  56. package/lib/cli/activity-printer/history.js +1 -1
  57. package/lib/cli/cdk-toolkit.js +17 -21
  58. package/lib/cli/cli.js +6 -6
  59. package/lib/cli/messages.d.ts +11 -9
  60. package/lib/cli/messages.js +7 -14
  61. package/lib/index.js +4856 -4134
  62. package/lib/list-stacks.d.ts +1 -1
  63. package/lib/list-stacks.js +1 -1
  64. package/lib/logging.d.ts +3 -1
  65. package/lib/logging.js +2 -1
  66. package/lib/toolkit/cli-io-host.d.ts +37 -125
  67. package/lib/toolkit/cli-io-host.js +44 -74
  68. package/package.json +3 -9
@@ -1,11 +1,11 @@
1
- import { IoMessaging } from '../../toolkit/cli-io-host';
1
+ import { IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';
2
2
  import { ICloudFormationClient } from '../aws-auth';
3
3
  export declare class ActiveAssetCache {
4
4
  private readonly stacks;
5
5
  rememberStack(stackTemplate: string): void;
6
6
  contains(asset: string): boolean;
7
7
  }
8
- export declare function refreshStacks(cfn: ICloudFormationClient, { ioHost, action }: IoMessaging, activeAssets: ActiveAssetCache, qualifier?: string): Promise<void>;
8
+ export declare function refreshStacks(cfn: ICloudFormationClient, ioHelper: IoHelper, activeAssets: ActiveAssetCache, qualifier?: string): Promise<void>;
9
9
  /**
10
10
  * Background Stack Refresh properties
11
11
  */
@@ -17,7 +17,7 @@ export interface BackgroundStackRefreshProps {
17
17
  /**
18
18
  * Used to send messages.
19
19
  */
20
- readonly msg: IoMessaging;
20
+ readonly ioHelper: IoHelper;
21
21
  /**
22
22
  * Active Asset storage
23
23
  */
@@ -36,7 +36,7 @@ async function paginateSdkCall(cb) {
36
36
  * - stacks in DELETE_COMPLETE or DELETE_IN_PROGRESS stage
37
37
  * - stacks that are using a different bootstrap qualifier
38
38
  */
39
- async function fetchAllStackTemplates(cfn, { ioHost, action }, qualifier) {
39
+ async function fetchAllStackTemplates(cfn, ioHelper, qualifier) {
40
40
  var _a;
41
41
  const stackNames = [];
42
42
  await paginateSdkCall(async (nextToken) => {
@@ -49,7 +49,7 @@ async function fetchAllStackTemplates(cfn, { ioHost, action }, qualifier) {
49
49
  .map((s) => { var _a; return (_a = s.StackId) !== null && _a !== void 0 ? _a : s.StackName; }));
50
50
  return stacks.NextToken;
51
51
  });
52
- await ioHost.notify((0, messages_1.debug)(action, `Parsing through ${stackNames.length} stacks`));
52
+ await ioHelper.notify((0, messages_1.debug)(`Parsing through ${stackNames.length} stacks`));
53
53
  const templates = [];
54
54
  for (const stack of stackNames) {
55
55
  let summary;
@@ -67,7 +67,7 @@ async function fetchAllStackTemplates(cfn, { ioHost, action }, qualifier) {
67
67
  templates.push(((_a = template.TemplateBody) !== null && _a !== void 0 ? _a : '') + JSON.stringify(summary === null || summary === void 0 ? void 0 : summary.Parameters));
68
68
  }
69
69
  }
70
- await ioHost.notify((0, messages_1.debug)(action, 'Done parsing through stacks'));
70
+ await ioHelper.notify((0, messages_1.debug)('Done parsing through stacks'));
71
71
  return templates;
72
72
  }
73
73
  /**
@@ -89,9 +89,9 @@ function bootstrapFilter(parameters, qualifier) {
89
89
  splitBootstrapVersion.length == 4 &&
90
90
  splitBootstrapVersion[2] != qualifier);
91
91
  }
92
- async function refreshStacks(cfn, { ioHost, action }, activeAssets, qualifier) {
92
+ async function refreshStacks(cfn, ioHelper, activeAssets, qualifier) {
93
93
  try {
94
- const stacks = await fetchAllStackTemplates(cfn, { ioHost, action }, qualifier);
94
+ const stacks = await fetchAllStackTemplates(cfn, ioHelper, qualifier);
95
95
  for (const stack of stacks) {
96
96
  activeAssets.rememberStack(stack);
97
97
  }
@@ -116,7 +116,7 @@ class BackgroundStackRefresh {
116
116
  }
117
117
  async refresh() {
118
118
  const startTime = Date.now();
119
- await refreshStacks(this.props.cfn, this.props.msg, this.props.activeAssets, this.props.qualifier);
119
+ await refreshStacks(this.props.cfn, this.props.ioHelper, this.props.activeAssets, this.props.qualifier);
120
120
  this.justRefreshedStacks();
121
121
  // If the last invocation of refreshStacks takes <5 minutes, the next invocation starts 5 minutes after the last one started.
122
122
  // If the last invocation of refreshStacks takes >5 minutes, the next invocation starts immediately.
@@ -151,4 +151,4 @@ class BackgroundStackRefresh {
151
151
  }
152
152
  }
153
153
  exports.BackgroundStackRefresh = BackgroundStackRefresh;
154
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-refresh.js","sourceRoot":"","sources":["stack-refresh.ts"],"names":[],"mappings":"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","sourcesContent":["import { ParameterDeclaration } from '@aws-sdk/client-cloudformation';\nimport { debug } from '../../cli/messages';\nimport { IoMessaging } from '../../toolkit/cli-io-host';\nimport { ToolkitError } from '../../toolkit/error';\nimport { 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, { ioHost, action }: IoMessaging, 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 ioHost.notify(debug(action, `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 ioHost.notify(debug(action, '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, { ioHost, action }: IoMessaging, activeAssets: ActiveAssetCache, qualifier?: string) {\n  try {\n    const stacks = await fetchAllStackTemplates(cfn, { ioHost, action }, 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 msg: IoMessaging;\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.msg, 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"]}
154
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-refresh.js","sourceRoot":"","sources":["stack-refresh.ts"],"names":[],"mappings":"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","sourcesContent":["import { ParameterDeclaration } from '@aws-sdk/client-cloudformation';\nimport { IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport { debug } from '../../cli/messages';\nimport { ToolkitError } from '../../toolkit/error';\nimport { 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(debug(`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(debug('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"]}
@@ -1,5 +1,5 @@
1
1
  import type { CloudFormationStackArtifact, Environment } from '@aws-cdk/cx-api';
2
- import { IoMessaging } from '../../toolkit/cli-io-host';
2
+ import { IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';
3
3
  import type { SDK, SdkProvider } from '../aws-auth';
4
4
  /**
5
5
  * Configuration needed to monitor CloudWatch Log Groups
@@ -22,4 +22,4 @@ export interface FoundLogGroupsResult {
22
22
  */
23
23
  readonly logGroupNames: string[];
24
24
  }
25
- export declare function findCloudWatchLogGroups(sdkProvider: SdkProvider, msg: IoMessaging, stackArtifact: CloudFormationStackArtifact): Promise<FoundLogGroupsResult>;
25
+ export declare function findCloudWatchLogGroups(sdkProvider: SdkProvider, ioHelper: IoHelper, stackArtifact: CloudFormationStackArtifact): Promise<FoundLogGroupsResult>;
@@ -1,7 +1,7 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.findCloudWatchLogGroups = findCloudWatchLogGroups;
4
- const logging_1 = require("../../logging");
4
+ const private_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private");
5
5
  const util_1 = require("../../util");
6
6
  const environment_1 = require("../environment");
7
7
  const evaluate_cloudformation_template_1 = require("../evaluate-cloudformation-template");
@@ -9,15 +9,15 @@ const mode_1 = require("../plugin/mode");
9
9
  const toolkit_info_1 = require("../toolkit-info");
10
10
  // resource types that have associated CloudWatch Log Groups that should _not_ be monitored
11
11
  const IGNORE_LOGS_RESOURCE_TYPES = ['AWS::EC2::FlowLog', 'AWS::CloudTrail::Trail', 'AWS::CodeBuild::Project'];
12
- async function findCloudWatchLogGroups(sdkProvider, msg, stackArtifact) {
12
+ async function findCloudWatchLogGroups(sdkProvider, ioHelper, stackArtifact) {
13
13
  let sdk;
14
14
  const resolvedEnv = await sdkProvider.resolveEnvironment(stackArtifact.environment);
15
15
  // try to assume the lookup role and fallback to the default credentials
16
16
  try {
17
- sdk = (await new environment_1.EnvironmentAccess(sdkProvider, toolkit_info_1.DEFAULT_TOOLKIT_STACK_NAME, msg).accessStackForLookup(stackArtifact)).sdk;
17
+ sdk = (await new environment_1.EnvironmentAccess(sdkProvider, toolkit_info_1.DEFAULT_TOOLKIT_STACK_NAME, ioHelper).accessStackForLookup(stackArtifact)).sdk;
18
18
  }
19
19
  catch (e) {
20
- (0, logging_1.debug)(`Failed to access SDK environment: ${(0, util_1.formatErrorMessage)(e)}`);
20
+ await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg(`Failed to access SDK environment: ${(0, util_1.formatErrorMessage)(e)}`));
21
21
  sdk = (await sdkProvider.forEnvironment(resolvedEnv, mode_1.Mode.ForReading)).sdk;
22
22
  }
23
23
  const listStackResources = new evaluate_cloudformation_template_1.LazyListStackResources(sdk, stackArtifact.stackName);
@@ -94,4 +94,4 @@ function findAllLogGroupNames(stackResources, evaluateCfnTemplate) {
94
94
  }
95
95
  return logGroupNames;
96
96
  }
97
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"find-cloudwatch-logs.js","sourceRoot":"","sources":["find-cloudwatch-logs.ts"],"names":[],"mappings":"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","sourcesContent":["import type { CloudFormationStackArtifact, Environment } from '@aws-cdk/cx-api';\nimport type { StackResourceSummary } from '@aws-sdk/client-cloudformation';\nimport { debug } from '../../logging';\nimport { IoMessaging } from '../../toolkit/cli-io-host';\nimport { formatErrorMessage } from '../../util';\nimport type { SDK, SdkProvider } from '../aws-auth';\nimport { EnvironmentAccess } from '../environment';\nimport { EvaluateCloudFormationTemplate, LazyListStackResources } from '../evaluate-cloudformation-template';\nimport { Mode } from '../plugin/mode';\nimport { DEFAULT_TOOLKIT_STACK_NAME } from '../toolkit-info';\n\n// resource types that have associated CloudWatch Log Groups that should _not_ be monitored\nconst IGNORE_LOGS_RESOURCE_TYPES = ['AWS::EC2::FlowLog', 'AWS::CloudTrail::Trail', 'AWS::CodeBuild::Project'];\n\n/**\n * Configuration needed to monitor CloudWatch Log Groups\n * found in a given CloudFormation Stack\n */\nexport interface FoundLogGroupsResult {\n  /**\n   * The resolved environment (account/region) that the log\n   * groups are deployed in\n   */\n  readonly env: Environment;\n\n  /**\n   * The SDK that can be used to read events from the CloudWatch\n   * Log Groups in the given environment\n   */\n  readonly sdk: SDK;\n\n  /**\n   * The names of the relevant CloudWatch Log Groups\n   * in the given CloudFormation template\n   */\n  readonly logGroupNames: string[];\n}\n\nexport async function findCloudWatchLogGroups(\n  sdkProvider: SdkProvider,\n  msg: IoMessaging,\n  stackArtifact: CloudFormationStackArtifact,\n): Promise<FoundLogGroupsResult> {\n  let sdk: SDK;\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stackArtifact.environment);\n  // try to assume the lookup role and fallback to the default credentials\n  try {\n    sdk = (await new EnvironmentAccess(sdkProvider, DEFAULT_TOOLKIT_STACK_NAME, msg).accessStackForLookup(stackArtifact)).sdk;\n  } catch (e: any) {\n    debug(`Failed to access SDK environment: ${formatErrorMessage(e)}`);\n    sdk = (await sdkProvider.forEnvironment(resolvedEnv, Mode.ForReading)).sdk;\n  }\n\n  const listStackResources = new LazyListStackResources(sdk, stackArtifact.stackName);\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    stackName: stackArtifact.stackName,\n    template: stackArtifact.template,\n    parameters: {},\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    sdk,\n  });\n\n  const stackResources = await listStackResources.listStackResources();\n  const logGroupNames = findAllLogGroupNames(stackResources, evaluateCfnTemplate);\n\n  return {\n    env: resolvedEnv,\n    sdk,\n    logGroupNames,\n  };\n}\n\n/**\n * Determine if a CloudWatch Log Group is associated\n * with an ignored resource\n */\nfunction isReferencedFromIgnoredResource(\n  logGroupResource: StackResourceSummary,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): boolean {\n  const resourcesReferencingLogGroup = evaluateCfnTemplate.findReferencesTo(logGroupResource.LogicalResourceId!);\n  return resourcesReferencingLogGroup.some((reference) => {\n    return IGNORE_LOGS_RESOURCE_TYPES.includes(reference.Type);\n  });\n}\n\ntype CloudWatchLogsResolver = (\n  resource: StackResourceSummary,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n) => string | undefined;\n\nconst cloudWatchLogsResolvers: Record<string, CloudWatchLogsResolver> = {\n  'AWS::Logs::LogGroup': (resource, evaluateCfnTemplate) => {\n    if (isReferencedFromIgnoredResource(resource, evaluateCfnTemplate)) {\n      return undefined;\n    }\n    return resource.PhysicalResourceId?.toString();\n  },\n\n  // Resource types that will create a CloudWatch log group with a specific name if one is not provided.\n  // The keys are CFN resource types, and the values are the name of the physical name property of that resource\n  // and the service name that is used in the automatically created CloudWatch log group.\n  'AWS::Lambda::Function': (resource, evaluateCfnTemplate) => {\n    const loggingConfig = evaluateCfnTemplate.getResourceProperty(resource.LogicalResourceId!, 'LoggingConfig');\n    if (loggingConfig?.LogGroup) {\n      // if LogGroup is a string then use it as the LogGroupName as it is referred by LogGroup.fromLogGroupArn in CDK\n      if (typeof loggingConfig.LogGroup === 'string') {\n        return loggingConfig.LogGroup;\n      }\n\n      // if { Ref: '...' } is used then try to resolve the LogGroupName from the referenced resource in the template\n      if (typeof loggingConfig.LogGroup === 'object') {\n        if (loggingConfig.LogGroup.Ref) {\n          return evaluateCfnTemplate.getResourceProperty(loggingConfig.LogGroup.Ref, 'LogGroupName');\n        }\n      }\n    }\n\n    return `/aws/lambda/${resource.PhysicalResourceId}`;\n  },\n};\n\n/**\n * Find all CloudWatch Log Groups in the deployed template.\n * This will find both explicitly created Log Groups (excluding those associated with ignored resources)\n * and Log Groups created implicitly (i.e. Lambda Functions)\n */\nfunction findAllLogGroupNames(\n  stackResources: StackResourceSummary[],\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): string[] {\n  const logGroupNames: string[] = [];\n\n  for (const resource of stackResources) {\n    const logGroupResolver = cloudWatchLogsResolvers[resource.ResourceType!];\n    if (logGroupResolver) {\n      const logGroupName = logGroupResolver(resource, evaluateCfnTemplate);\n      if (logGroupName) {\n        logGroupNames.push(logGroupName);\n      }\n    }\n  }\n\n  return logGroupNames;\n}\n"]}
97
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"find-cloudwatch-logs.js","sourceRoot":"","sources":["find-cloudwatch-logs.ts"],"names":[],"mappings":"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","sourcesContent":["import type { CloudFormationStackArtifact, Environment } from '@aws-cdk/cx-api';\nimport type { StackResourceSummary } from '@aws-sdk/client-cloudformation';\nimport { IO, IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport { formatErrorMessage } from '../../util';\nimport type { SDK, SdkProvider } from '../aws-auth';\nimport { EnvironmentAccess } from '../environment';\nimport { EvaluateCloudFormationTemplate, LazyListStackResources } from '../evaluate-cloudformation-template';\nimport { Mode } from '../plugin/mode';\nimport { DEFAULT_TOOLKIT_STACK_NAME } from '../toolkit-info';\n\n// resource types that have associated CloudWatch Log Groups that should _not_ be monitored\nconst IGNORE_LOGS_RESOURCE_TYPES = ['AWS::EC2::FlowLog', 'AWS::CloudTrail::Trail', 'AWS::CodeBuild::Project'];\n\n/**\n * Configuration needed to monitor CloudWatch Log Groups\n * found in a given CloudFormation Stack\n */\nexport interface FoundLogGroupsResult {\n  /**\n   * The resolved environment (account/region) that the log\n   * groups are deployed in\n   */\n  readonly env: Environment;\n\n  /**\n   * The SDK that can be used to read events from the CloudWatch\n   * Log Groups in the given environment\n   */\n  readonly sdk: SDK;\n\n  /**\n   * The names of the relevant CloudWatch Log Groups\n   * in the given CloudFormation template\n   */\n  readonly logGroupNames: string[];\n}\n\nexport async function findCloudWatchLogGroups(\n  sdkProvider: SdkProvider,\n  ioHelper: IoHelper,\n  stackArtifact: CloudFormationStackArtifact,\n): Promise<FoundLogGroupsResult> {\n  let sdk: SDK;\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stackArtifact.environment);\n  // try to assume the lookup role and fallback to the default credentials\n  try {\n    sdk = (await new EnvironmentAccess(sdkProvider, DEFAULT_TOOLKIT_STACK_NAME, ioHelper).accessStackForLookup(stackArtifact)).sdk;\n  } catch (e: any) {\n    await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(`Failed to access SDK environment: ${formatErrorMessage(e)}`));\n    sdk = (await sdkProvider.forEnvironment(resolvedEnv, Mode.ForReading)).sdk;\n  }\n\n  const listStackResources = new LazyListStackResources(sdk, stackArtifact.stackName);\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    stackName: stackArtifact.stackName,\n    template: stackArtifact.template,\n    parameters: {},\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    sdk,\n  });\n\n  const stackResources = await listStackResources.listStackResources();\n  const logGroupNames = findAllLogGroupNames(stackResources, evaluateCfnTemplate);\n\n  return {\n    env: resolvedEnv,\n    sdk,\n    logGroupNames,\n  };\n}\n\n/**\n * Determine if a CloudWatch Log Group is associated\n * with an ignored resource\n */\nfunction isReferencedFromIgnoredResource(\n  logGroupResource: StackResourceSummary,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): boolean {\n  const resourcesReferencingLogGroup = evaluateCfnTemplate.findReferencesTo(logGroupResource.LogicalResourceId!);\n  return resourcesReferencingLogGroup.some((reference) => {\n    return IGNORE_LOGS_RESOURCE_TYPES.includes(reference.Type);\n  });\n}\n\ntype CloudWatchLogsResolver = (\n  resource: StackResourceSummary,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n) => string | undefined;\n\nconst cloudWatchLogsResolvers: Record<string, CloudWatchLogsResolver> = {\n  'AWS::Logs::LogGroup': (resource, evaluateCfnTemplate) => {\n    if (isReferencedFromIgnoredResource(resource, evaluateCfnTemplate)) {\n      return undefined;\n    }\n    return resource.PhysicalResourceId?.toString();\n  },\n\n  // Resource types that will create a CloudWatch log group with a specific name if one is not provided.\n  // The keys are CFN resource types, and the values are the name of the physical name property of that resource\n  // and the service name that is used in the automatically created CloudWatch log group.\n  'AWS::Lambda::Function': (resource, evaluateCfnTemplate) => {\n    const loggingConfig = evaluateCfnTemplate.getResourceProperty(resource.LogicalResourceId!, 'LoggingConfig');\n    if (loggingConfig?.LogGroup) {\n      // if LogGroup is a string then use it as the LogGroupName as it is referred by LogGroup.fromLogGroupArn in CDK\n      if (typeof loggingConfig.LogGroup === 'string') {\n        return loggingConfig.LogGroup;\n      }\n\n      // if { Ref: '...' } is used then try to resolve the LogGroupName from the referenced resource in the template\n      if (typeof loggingConfig.LogGroup === 'object') {\n        if (loggingConfig.LogGroup.Ref) {\n          return evaluateCfnTemplate.getResourceProperty(loggingConfig.LogGroup.Ref, 'LogGroupName');\n        }\n      }\n    }\n\n    return `/aws/lambda/${resource.PhysicalResourceId}`;\n  },\n};\n\n/**\n * Find all CloudWatch Log Groups in the deployed template.\n * This will find both explicitly created Log Groups (excluding those associated with ignored resources)\n * and Log Groups created implicitly (i.e. Lambda Functions)\n */\nfunction findAllLogGroupNames(\n  stackResources: StackResourceSummary[],\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): string[] {\n  const logGroupNames: string[] = [];\n\n  for (const resource of stackResources) {\n    const logGroupResolver = cloudWatchLogsResolvers[resource.ResourceType!];\n    if (logGroupResolver) {\n      const logGroupName = logGroupResolver(resource, evaluateCfnTemplate);\n      if (logGroupName) {\n        logGroupNames.push(logGroupName);\n      }\n    }\n  }\n\n  return logGroupNames;\n}\n"]}
@@ -1,5 +1,18 @@
1
1
  import * as cxapi from '@aws-cdk/cx-api';
2
+ import { IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';
2
3
  import type { SDK } from '../aws-auth';
4
+ export interface CloudWatchLogEventMonitorProps {
5
+ /**
6
+ * The IoHost used for messaging
7
+ */
8
+ readonly ioHelper: IoHelper;
9
+ /**
10
+ * The time from which we start reading log messages
11
+ *
12
+ * @default - now
13
+ */
14
+ readonly startTime?: Date;
15
+ }
3
16
  export declare class CloudWatchLogEventMonitor {
4
17
  /**
5
18
  * Determines which events not to display
@@ -9,12 +22,21 @@ export declare class CloudWatchLogEventMonitor {
9
22
  * Map of environment (account:region) to LogGroupsAccessSettings
10
23
  */
11
24
  private readonly envsLogGroupsAccessSettings;
12
- private active;
13
- constructor(startTime?: Date);
25
+ /**
26
+ * After reading events from all CloudWatch log groups
27
+ * how long should we wait to read more events.
28
+ *
29
+ * If there is some error with reading events (i.e. Throttle)
30
+ * then this is also how long we wait until we try again
31
+ */
32
+ private readonly pollingInterval;
33
+ monitorId?: string;
34
+ private readonly ioHelper;
35
+ constructor(props: CloudWatchLogEventMonitorProps);
14
36
  /**
15
37
  * resume reading/printing events
16
38
  */
17
- activate(): void;
39
+ activate(): Promise<void>;
18
40
  /**
19
41
  * deactivates the monitor so no new events are read
20
42
  * use case for this is when we are in the middle of performing a deployment
@@ -24,7 +46,7 @@ export declare class CloudWatchLogEventMonitor {
24
46
  * Also resets the start time to be when the new deployment was triggered
25
47
  * and clears the list of tracked log groups
26
48
  */
27
- deactivate(): void;
49
+ deactivate(): Promise<void>;
28
50
  /**
29
51
  * Adds CloudWatch log groups to read log events from.
30
52
  * Since we could be watching multiple stacks that deploy to
@@ -33,6 +55,7 @@ export declare class CloudWatchLogEventMonitor {
33
55
  * that environment.
34
56
  */
35
57
  addLogGroups(env: cxapi.Environment, sdk: SDK, logGroupNames: string[]): void;
58
+ private logGroupNames;
36
59
  private scheduleNextTick;
37
60
  private tick;
38
61
  /**
@@ -3,32 +3,38 @@ Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.CloudWatchLogEventMonitor = void 0;
4
4
  const util = require("util");
5
5
  const chalk = require("chalk");
6
- const logging_1 = require("../../logging");
6
+ const uuid = require("uuid");
7
+ const private_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private");
7
8
  const util_1 = require("../../util");
8
- /**
9
- * After reading events from all CloudWatch log groups
10
- * how long should we wait to read more events.
11
- *
12
- * If there is some error with reading events (i.e. Throttle)
13
- * then this is also how long we wait until we try again
14
- */
15
- const SLEEP = 2000;
16
9
  class CloudWatchLogEventMonitor {
17
- constructor(startTime) {
18
- var _a;
10
+ constructor(props) {
11
+ var _a, _b;
19
12
  /**
20
13
  * Map of environment (account:region) to LogGroupsAccessSettings
21
14
  */
22
15
  this.envsLogGroupsAccessSettings = new Map();
23
- this.active = false;
24
- this.startTime = (_a = startTime === null || startTime === void 0 ? void 0 : startTime.getTime()) !== null && _a !== void 0 ? _a : Date.now();
16
+ /**
17
+ * After reading events from all CloudWatch log groups
18
+ * how long should we wait to read more events.
19
+ *
20
+ * If there is some error with reading events (i.e. Throttle)
21
+ * then this is also how long we wait until we try again
22
+ */
23
+ this.pollingInterval = 2000;
24
+ this.startTime = (_b = (_a = props.startTime) === null || _a === void 0 ? void 0 : _a.getTime()) !== null && _b !== void 0 ? _b : Date.now();
25
+ this.ioHelper = props.ioHelper;
25
26
  }
26
27
  /**
27
28
  * resume reading/printing events
28
29
  */
29
- activate() {
30
- this.active = true;
31
- this.scheduleNextTick(0);
30
+ async activate() {
31
+ this.monitorId = uuid.v4();
32
+ await this.ioHelper.notify(private_1.IO.CDK_TOOLKIT_I5032.msg('Start monitoring log groups', {
33
+ monitor: this.monitorId,
34
+ logGroupNames: this.logGroupNames(),
35
+ }));
36
+ await this.tick();
37
+ this.scheduleNextTick();
32
38
  }
33
39
  /**
34
40
  * deactivates the monitor so no new events are read
@@ -39,9 +45,14 @@ class CloudWatchLogEventMonitor {
39
45
  * Also resets the start time to be when the new deployment was triggered
40
46
  * and clears the list of tracked log groups
41
47
  */
42
- deactivate() {
43
- this.active = false;
48
+ async deactivate() {
49
+ const oldMonitorId = this.monitorId;
50
+ this.monitorId = undefined;
44
51
  this.startTime = Date.now();
52
+ await this.ioHelper.notify(private_1.IO.CDK_TOOLKIT_I5034.msg('Stopped monitoring log groups', {
53
+ monitor: oldMonitorId,
54
+ logGroupNames: this.logGroupNames(),
55
+ }));
45
56
  this.envsLogGroupsAccessSettings.clear();
46
57
  }
47
58
  /**
@@ -66,26 +77,36 @@ class CloudWatchLogEventMonitor {
66
77
  },
67
78
  });
68
79
  }
69
- scheduleNextTick(sleep) {
70
- setTimeout(() => void this.tick(), sleep);
80
+ logGroupNames() {
81
+ return Array.from(this.envsLogGroupsAccessSettings.values()).flatMap((settings) => Object.keys(settings.logGroupsStartTimes));
82
+ }
83
+ scheduleNextTick() {
84
+ if (!this.monitorId) {
85
+ return;
86
+ }
87
+ setTimeout(() => void this.tick(), this.pollingInterval);
71
88
  }
72
89
  async tick() {
73
90
  // excluding from codecoverage because this
74
91
  // doesn't always run (depends on timing)
75
- /* istanbul ignore next */
76
- if (!this.active) {
92
+ /* c8 ignore next */
93
+ if (!this.monitorId) {
77
94
  return;
78
95
  }
79
96
  try {
80
97
  const events = (0, util_1.flatten)(await this.readNewEvents());
81
- events.forEach((event) => {
82
- this.print(event);
83
- });
98
+ for (const event of events) {
99
+ await this.print(event);
100
+ }
101
+ // We might have been stop()ped while the network call was in progress.
102
+ if (!this.monitorId) {
103
+ return;
104
+ }
84
105
  }
85
106
  catch (e) {
86
- (0, logging_1.error)('Error occurred while monitoring logs: %s', e);
107
+ await this.ioHelper.notify(private_1.IO.CDK_TOOLKIT_E5035.msg('Error occurred while monitoring logs: %s', { error: e }));
87
108
  }
88
- this.scheduleNextTick(SLEEP);
109
+ this.scheduleNextTick();
89
110
  }
90
111
  /**
91
112
  * Reads all new log events from a set of CloudWatch Log Groups
@@ -105,8 +126,8 @@ class CloudWatchLogEventMonitor {
105
126
  /**
106
127
  * Print out a cloudwatch event
107
128
  */
108
- print(event) {
109
- (0, logging_1.info)(util.format('[%s] %s %s', chalk.blue(event.logGroupName), chalk.yellow(event.timestamp.toLocaleTimeString()), event.message.trim()));
129
+ async print(event) {
130
+ await this.ioHelper.notify(private_1.IO.CDK_TOOLKIT_I5033.msg(util.format('[%s] %s %s', chalk.blue(event.logGroupName), chalk.yellow(event.timestamp.toLocaleTimeString()), event.message.trim()), event));
110
131
  }
111
132
  /**
112
133
  * Reads all new log events from a CloudWatch Log Group
@@ -166,4 +187,4 @@ class CloudWatchLogEventMonitor {
166
187
  }
167
188
  }
168
189
  exports.CloudWatchLogEventMonitor = CloudWatchLogEventMonitor;
169
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logs-monitor.js","sourceRoot":"","sources":["logs-monitor.ts"],"names":[],"mappings":"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","sourcesContent":["import * as util from 'util';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport { info, error } from '../../logging';\nimport { flatten } from '../../util';\nimport type { SDK } from '../aws-auth';\n\n/**\n * After reading events from all CloudWatch log groups\n * how long should we wait to read more events.\n *\n * If there is some error with reading events (i.e. Throttle)\n * then this is also how long we wait until we try again\n */\nconst SLEEP = 2_000;\n\n/**\n * Represents a CloudWatch Log Event that will be\n * printed to the terminal\n */\ninterface CloudWatchLogEvent {\n  /**\n   * The log event message\n   */\n  readonly message: string;\n\n  /**\n   * The name of the log group\n   */\n  readonly logGroupName: string;\n\n  /**\n   * The time at which the event occurred\n   */\n  readonly timestamp: Date;\n}\n\n/**\n * Configuration tracking information on the log groups that are\n * being monitored\n */\ninterface LogGroupsAccessSettings {\n  /**\n   * The SDK for a given environment (account/region)\n   */\n  readonly sdk: SDK;\n\n  /**\n   * A map of log groups and associated startTime in a given account.\n   *\n   * The monitor will read events from the log group starting at the\n   * associated startTime\n   */\n  readonly logGroupsStartTimes: { [logGroupName: string]: number };\n}\n\nexport class CloudWatchLogEventMonitor {\n  /**\n   * Determines which events not to display\n   */\n  private startTime: number;\n\n  /**\n   * Map of environment (account:region) to LogGroupsAccessSettings\n   */\n  private readonly envsLogGroupsAccessSettings = new Map<string, LogGroupsAccessSettings>();\n\n  private active = false;\n\n  constructor(startTime?: Date) {\n    this.startTime = startTime?.getTime() ?? Date.now();\n  }\n\n  /**\n   * resume reading/printing events\n   */\n  public activate(): void {\n    this.active = true;\n    this.scheduleNextTick(0);\n  }\n\n  /**\n   * deactivates the monitor so no new events are read\n   * use case for this is when we are in the middle of performing a deployment\n   * and don't want to interweave all the logs together with the CFN\n   * deployment logs\n   *\n   * Also resets the start time to be when the new deployment was triggered\n   * and clears the list of tracked log groups\n   */\n  public deactivate(): void {\n    this.active = false;\n    this.startTime = Date.now();\n    this.envsLogGroupsAccessSettings.clear();\n  }\n\n  /**\n   * Adds CloudWatch log groups to read log events from.\n   * Since we could be watching multiple stacks that deploy to\n   * multiple environments (account+region), we need to store a list of log groups\n   * per env along with the SDK object that has access to read from\n   * that environment.\n   */\n  public addLogGroups(env: cxapi.Environment, sdk: SDK, logGroupNames: string[]): void {\n    const awsEnv = `${env.account}:${env.region}`;\n    const logGroupsStartTimes = logGroupNames.reduce(\n      (acc, groupName) => {\n        acc[groupName] = this.startTime;\n        return acc;\n      },\n      {} as { [logGroupName: string]: number },\n    );\n    this.envsLogGroupsAccessSettings.set(awsEnv, {\n      sdk,\n      logGroupsStartTimes: {\n        ...this.envsLogGroupsAccessSettings.get(awsEnv)?.logGroupsStartTimes,\n        ...logGroupsStartTimes,\n      },\n    });\n  }\n\n  private scheduleNextTick(sleep: number): void {\n    setTimeout(() => void this.tick(), sleep);\n  }\n\n  private async tick(): Promise<void> {\n    // excluding from codecoverage because this\n    // doesn't always run (depends on timing)\n    /* istanbul ignore next */\n    if (!this.active) {\n      return;\n    }\n    try {\n      const events = flatten(await this.readNewEvents());\n      events.forEach((event) => {\n        this.print(event);\n      });\n    } catch (e) {\n      error('Error occurred while monitoring logs: %s', e);\n    }\n\n    this.scheduleNextTick(SLEEP);\n  }\n\n  /**\n   * Reads all new log events from a set of CloudWatch Log Groups\n   * in parallel\n   */\n  private async readNewEvents(): Promise<Array<Array<CloudWatchLogEvent>>> {\n    const promises: Array<Promise<Array<CloudWatchLogEvent>>> = [];\n    for (const settings of this.envsLogGroupsAccessSettings.values()) {\n      for (const group of Object.keys(settings.logGroupsStartTimes)) {\n        promises.push(this.readEventsFromLogGroup(settings, group));\n      }\n    }\n    // Limited set of log groups\n    // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n    return Promise.all(promises);\n  }\n\n  /**\n   * Print out a cloudwatch event\n   */\n  private print(event: CloudWatchLogEvent): void {\n    info(\n      util.format(\n        '[%s] %s %s',\n        chalk.blue(event.logGroupName),\n        chalk.yellow(event.timestamp.toLocaleTimeString()),\n        event.message.trim(),\n      ),\n    );\n  }\n\n  /**\n   * Reads all new log events from a CloudWatch Log Group\n   * starting at either the time the hotswap was triggered or\n   * when the last event was read on the previous tick\n   */\n  private async readEventsFromLogGroup(\n    logGroupsAccessSettings: LogGroupsAccessSettings,\n    logGroupName: string,\n  ): Promise<Array<CloudWatchLogEvent>> {\n    const events: CloudWatchLogEvent[] = [];\n\n    // log events from some service are ingested faster than others\n    // so we need to track the start/end time for each log group individually\n    // to make sure that we process all events from each log group\n    const startTime = logGroupsAccessSettings.logGroupsStartTimes[logGroupName] ?? this.startTime;\n    let endTime = startTime;\n    try {\n      const response = await logGroupsAccessSettings.sdk.cloudWatchLogs().filterLogEvents({\n        logGroupName: logGroupName,\n        limit: 100,\n        startTime: startTime,\n      });\n      const filteredEvents = response.events ?? [];\n\n      for (const event of filteredEvents) {\n        if (event.message) {\n          events.push({\n            message: event.message,\n            logGroupName,\n            timestamp: event.timestamp ? new Date(event.timestamp) : new Date(),\n          });\n\n          if (event.timestamp && endTime < event.timestamp) {\n            endTime = event.timestamp;\n          }\n        }\n      }\n      // As long as there are _any_ events in the log group `filterLogEvents` will return a nextToken.\n      // This is true even if these events are before `startTime`. So if we have 100 events and a nextToken\n      // then assume that we have hit the limit and let the user know some messages have been suppressed.\n      // We are essentially showing them a sampling (10000 events printed out is not very useful)\n      if (filteredEvents.length === 100 && response.nextToken) {\n        events.push({\n          message: '>>> `watch` shows only the first 100 log messages - the rest have been truncated...',\n          logGroupName,\n          timestamp: new Date(endTime),\n        });\n      }\n    } catch (e: any) {\n      // with Lambda functions the CloudWatch is not created\n      // until something is logged, so just keep polling until\n      // there is somthing to find\n      if (e.name === 'ResourceNotFoundException') {\n        return [];\n      }\n      throw e;\n    }\n    logGroupsAccessSettings.logGroupsStartTimes[logGroupName] = endTime + 1;\n    return events;\n  }\n}\n"]}
190
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logs-monitor.js","sourceRoot":"","sources":["logs-monitor.ts"],"names":[],"mappings":"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","sourcesContent":["import * as util from 'util';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport * as uuid from 'uuid';\nimport type { CloudWatchLogEvent } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io';\nimport { IO, IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport { flatten } from '../../util';\nimport type { SDK } from '../aws-auth';\n\n/**\n * Configuration tracking information on the log groups that are\n * being monitored\n */\ninterface LogGroupsAccessSettings {\n  /**\n   * The SDK for a given environment (account/region)\n   */\n  readonly sdk: SDK;\n\n  /**\n   * A map of log groups and associated startTime in a given account.\n   *\n   * The monitor will read events from the log group starting at the\n   * associated startTime\n   */\n  readonly logGroupsStartTimes: { [logGroupName: string]: number };\n}\n\nexport interface CloudWatchLogEventMonitorProps {\n  /**\n   * The IoHost used for messaging\n   */\n  readonly ioHelper: IoHelper;\n\n  /**\n   * The time from which we start reading log messages\n   *\n   * @default - now\n   */\n  readonly startTime?: Date;\n}\n\nexport class CloudWatchLogEventMonitor {\n  /**\n   * Determines which events not to display\n   */\n  private startTime: number;\n\n  /**\n   * Map of environment (account:region) to LogGroupsAccessSettings\n   */\n  private readonly envsLogGroupsAccessSettings = new Map<string, LogGroupsAccessSettings>();\n\n  /**\n   * After reading events from all CloudWatch log groups\n   * how long should we wait to read more events.\n   *\n   * If there is some error with reading events (i.e. Throttle)\n   * then this is also how long we wait until we try again\n   */\n  private readonly pollingInterval: number = 2_000;\n\n  public monitorId?: string;\n  private readonly ioHelper: IoHelper;\n\n  constructor(props: CloudWatchLogEventMonitorProps) {\n    this.startTime = props.startTime?.getTime() ?? Date.now();\n    this.ioHelper = props.ioHelper;\n  }\n\n  /**\n   * resume reading/printing events\n   */\n  public async activate(): Promise<void> {\n    this.monitorId = uuid.v4();\n\n    await this.ioHelper.notify(IO.CDK_TOOLKIT_I5032.msg('Start monitoring log groups', {\n      monitor: this.monitorId,\n      logGroupNames: this.logGroupNames(),\n    }));\n\n    await this.tick();\n    this.scheduleNextTick();\n  }\n\n  /**\n   * deactivates the monitor so no new events are read\n   * use case for this is when we are in the middle of performing a deployment\n   * and don't want to interweave all the logs together with the CFN\n   * deployment logs\n   *\n   * Also resets the start time to be when the new deployment was triggered\n   * and clears the list of tracked log groups\n   */\n  public async deactivate(): Promise<void> {\n    const oldMonitorId = this.monitorId!;\n    this.monitorId = undefined;\n    this.startTime = Date.now();\n\n    await this.ioHelper.notify(IO.CDK_TOOLKIT_I5034.msg('Stopped monitoring log groups', {\n      monitor: oldMonitorId,\n      logGroupNames: this.logGroupNames(),\n    }));\n\n    this.envsLogGroupsAccessSettings.clear();\n  }\n\n  /**\n   * Adds CloudWatch log groups to read log events from.\n   * Since we could be watching multiple stacks that deploy to\n   * multiple environments (account+region), we need to store a list of log groups\n   * per env along with the SDK object that has access to read from\n   * that environment.\n   */\n  public addLogGroups(env: cxapi.Environment, sdk: SDK, logGroupNames: string[]): void {\n    const awsEnv = `${env.account}:${env.region}`;\n    const logGroupsStartTimes = logGroupNames.reduce(\n      (acc, groupName) => {\n        acc[groupName] = this.startTime;\n        return acc;\n      },\n      {} as { [logGroupName: string]: number },\n    );\n    this.envsLogGroupsAccessSettings.set(awsEnv, {\n      sdk,\n      logGroupsStartTimes: {\n        ...this.envsLogGroupsAccessSettings.get(awsEnv)?.logGroupsStartTimes,\n        ...logGroupsStartTimes,\n      },\n    });\n  }\n\n  private logGroupNames(): string[] {\n    return Array.from(this.envsLogGroupsAccessSettings.values()).flatMap((settings) => Object.keys(settings.logGroupsStartTimes));\n  }\n\n  private scheduleNextTick(): void {\n    if (!this.monitorId) {\n      return;\n    }\n\n    setTimeout(() => void this.tick(), this.pollingInterval);\n  }\n\n  private async tick(): Promise<void> {\n    // excluding from codecoverage because this\n    // doesn't always run (depends on timing)\n    /* c8 ignore next */\n    if (!this.monitorId) {\n      return;\n    }\n\n    try {\n      const events = flatten(await this.readNewEvents());\n      for (const event of events) {\n        await this.print(event);\n      }\n\n      // We might have been stop()ped while the network call was in progress.\n      if (!this.monitorId) {\n        return;\n      }\n    } catch (e: any) {\n      await this.ioHelper.notify(IO.CDK_TOOLKIT_E5035.msg('Error occurred while monitoring logs: %s', { error: e }));\n    }\n\n    this.scheduleNextTick();\n  }\n\n  /**\n   * Reads all new log events from a set of CloudWatch Log Groups\n   * in parallel\n   */\n  private async readNewEvents(): Promise<Array<Array<CloudWatchLogEvent>>> {\n    const promises: Array<Promise<Array<CloudWatchLogEvent>>> = [];\n    for (const settings of this.envsLogGroupsAccessSettings.values()) {\n      for (const group of Object.keys(settings.logGroupsStartTimes)) {\n        promises.push(this.readEventsFromLogGroup(settings, group));\n      }\n    }\n    // Limited set of log groups\n    // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n    return Promise.all(promises);\n  }\n\n  /**\n   * Print out a cloudwatch event\n   */\n  private async print(event: CloudWatchLogEvent): Promise<void> {\n    await this.ioHelper.notify(IO.CDK_TOOLKIT_I5033.msg(\n      util.format(\n        '[%s] %s %s',\n        chalk.blue(event.logGroupName),\n        chalk.yellow(event.timestamp.toLocaleTimeString()),\n        event.message.trim(),\n      ),\n      event,\n    ));\n  }\n\n  /**\n   * Reads all new log events from a CloudWatch Log Group\n   * starting at either the time the hotswap was triggered or\n   * when the last event was read on the previous tick\n   */\n  private async readEventsFromLogGroup(\n    logGroupsAccessSettings: LogGroupsAccessSettings,\n    logGroupName: string,\n  ): Promise<Array<CloudWatchLogEvent>> {\n    const events: CloudWatchLogEvent[] = [];\n\n    // log events from some service are ingested faster than others\n    // so we need to track the start/end time for each log group individually\n    // to make sure that we process all events from each log group\n    const startTime = logGroupsAccessSettings.logGroupsStartTimes[logGroupName] ?? this.startTime;\n    let endTime = startTime;\n    try {\n      const response = await logGroupsAccessSettings.sdk.cloudWatchLogs().filterLogEvents({\n        logGroupName: logGroupName,\n        limit: 100,\n        startTime: startTime,\n      });\n      const filteredEvents = response.events ?? [];\n\n      for (const event of filteredEvents) {\n        if (event.message) {\n          events.push({\n            message: event.message,\n            logGroupName,\n            timestamp: event.timestamp ? new Date(event.timestamp) : new Date(),\n          });\n\n          if (event.timestamp && endTime < event.timestamp) {\n            endTime = event.timestamp;\n          }\n        }\n      }\n      // As long as there are _any_ events in the log group `filterLogEvents` will return a nextToken.\n      // This is true even if these events are before `startTime`. So if we have 100 events and a nextToken\n      // then assume that we have hit the limit and let the user know some messages have been suppressed.\n      // We are essentially showing them a sampling (10000 events printed out is not very useful)\n      if (filteredEvents.length === 100 && response.nextToken) {\n        events.push({\n          message: '>>> `watch` shows only the first 100 log messages - the rest have been truncated...',\n          logGroupName,\n          timestamp: new Date(endTime),\n        });\n      }\n    } catch (e: any) {\n      // with Lambda functions the CloudWatch is not created\n      // until something is logged, so just keep polling until\n      // there is somthing to find\n      if (e.name === 'ResourceNotFoundException') {\n        return [];\n      }\n      throw e;\n    }\n    logGroupsAccessSettings.logGroupsStartTimes[logGroupName] = endTime + 1;\n    return events;\n  }\n}\n"]}
@@ -1,12 +1,11 @@
1
1
  import { ResourceDifference } from '@aws-cdk/cloudformation-diff';
2
2
  import * as cxapi from '@aws-cdk/cx-api';
3
- import { IIoHost, ToolkitAction } from '../../toolkit/cli-io-host';
3
+ import { IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';
4
4
  import { type Deployments, DeploymentMethod, ResourceIdentifierProperties, ResourcesToImport } from '../deployments';
5
5
  import type { Tag } from '../tags';
6
6
  export interface ResourceImporterProps {
7
7
  deployments: Deployments;
8
- ioHost: IIoHost;
9
- action: ToolkitAction;
8
+ ioHelper: IoHelper;
10
9
  }
11
10
  export interface ImportDeploymentOptions {
12
11
  /**
@@ -89,8 +88,7 @@ export declare class ResourceImporter {
89
88
  private _currentTemplate;
90
89
  private readonly stack;
91
90
  private readonly cfn;
92
- private readonly ioHost;
93
- private readonly action;
91
+ private readonly ioHelper;
94
92
  constructor(stack: cxapi.CloudFormationStackArtifact, props: ResourceImporterProps);
95
93
  /**
96
94
  * Ask the user for resources to import