aws-cdk 2.1004.0 → 2.1005.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 (148) hide show
  1. package/build-info.json +2 -2
  2. package/lib/api/aws-auth/account-cache.d.ts +1 -1
  3. package/lib/api/aws-auth/account-cache.js +1 -1
  4. package/lib/api/aws-auth/awscli-compatible.d.ts +1 -1
  5. package/lib/api/aws-auth/awscli-compatible.js +1 -1
  6. package/lib/api/aws-auth/credential-plugins.d.ts +1 -1
  7. package/lib/api/aws-auth/credential-plugins.js +1 -1
  8. package/lib/api/aws-auth/provider-caching.d.ts +1 -1
  9. package/lib/api/aws-auth/provider-caching.js +1 -1
  10. package/lib/api/aws-auth/sdk-logger.d.ts +2 -2
  11. package/lib/api/aws-auth/sdk-logger.js +1 -1
  12. package/lib/api/aws-auth/sdk-provider.d.ts +4 -4
  13. package/lib/api/aws-auth/sdk-provider.js +1 -1
  14. package/lib/api/aws-auth/sdk.d.ts +12 -12
  15. package/lib/api/aws-auth/sdk.js +1 -1
  16. package/lib/api/bootstrap/bootstrap-environment.d.ts +2 -2
  17. package/lib/api/bootstrap/bootstrap-environment.js +1 -1
  18. package/lib/api/bootstrap/bootstrap-props.d.ts +3 -3
  19. package/lib/api/bootstrap/bootstrap-props.js +1 -1
  20. package/lib/api/bootstrap/deploy-bootstrap.d.ts +4 -4
  21. package/lib/api/bootstrap/deploy-bootstrap.js +1 -1
  22. package/lib/api/bootstrap/legacy-template.d.ts +1 -1
  23. package/lib/api/bootstrap/legacy-template.js +1 -1
  24. package/lib/api/cxapp/cloud-executable.d.ts +3 -3
  25. package/lib/api/cxapp/cloud-executable.js +1 -1
  26. package/lib/api/cxapp/environments.d.ts +3 -3
  27. package/lib/api/cxapp/environments.js +1 -1
  28. package/lib/api/cxapp/exec.d.ts +4 -4
  29. package/lib/api/cxapp/exec.js +1 -1
  30. package/lib/api/deployments/asset-publishing.d.ts +2 -2
  31. package/lib/api/deployments/asset-publishing.js +1 -1
  32. package/lib/api/deployments/assets.d.ts +3 -3
  33. package/lib/api/deployments/assets.js +1 -1
  34. package/lib/api/deployments/checks.d.ts +2 -2
  35. package/lib/api/deployments/checks.js +1 -1
  36. package/lib/api/deployments/cloudformation.d.ts +2 -2
  37. package/lib/api/deployments/cloudformation.js +1 -1
  38. package/lib/api/deployments/deploy-stack.d.ts +6 -6
  39. package/lib/api/deployments/deploy-stack.js +36 -36
  40. package/lib/api/deployments/deployments.d.ts +6 -6
  41. package/lib/api/deployments/deployments.js +1 -1
  42. package/lib/api/deployments/hotswap-deployments.d.ts +6 -4
  43. package/lib/api/deployments/hotswap-deployments.js +51 -26
  44. package/lib/api/environment/environment-access.d.ts +6 -6
  45. package/lib/api/environment/environment-access.js +1 -1
  46. package/lib/api/environment/environment-resources.d.ts +1 -1
  47. package/lib/api/environment/environment-resources.js +1 -1
  48. package/lib/api/evaluate-cloudformation-template.d.ts +7 -2
  49. package/lib/api/evaluate-cloudformation-template.js +11 -5
  50. package/lib/api/garbage-collection/garbage-collector.d.ts +4 -4
  51. package/lib/api/garbage-collection/garbage-collector.js +2 -2
  52. package/lib/api/garbage-collection/progress-printer.d.ts +2 -2
  53. package/lib/api/garbage-collection/progress-printer.js +1 -1
  54. package/lib/api/garbage-collection/stack-refresh.d.ts +2 -2
  55. package/lib/api/garbage-collection/stack-refresh.js +1 -1
  56. package/lib/api/hotswap/appsync-mapping-templates.d.ts +3 -2
  57. package/lib/api/hotswap/appsync-mapping-templates.js +8 -6
  58. package/lib/api/hotswap/code-build-projects.d.ts +3 -2
  59. package/lib/api/hotswap/code-build-projects.js +8 -6
  60. package/lib/api/hotswap/common.d.ts +45 -34
  61. package/lib/api/hotswap/common.js +2 -14
  62. package/lib/api/hotswap/ecs-services.d.ts +3 -2
  63. package/lib/api/hotswap/ecs-services.js +7 -5
  64. package/lib/api/hotswap/lambda-functions.d.ts +3 -2
  65. package/lib/api/hotswap/lambda-functions.js +47 -65
  66. package/lib/api/hotswap/s3-bucket-deployments.d.ts +4 -8
  67. package/lib/api/hotswap/s3-bucket-deployments.js +18 -17
  68. package/lib/api/hotswap/stepfunctions-state-machines.d.ts +3 -2
  69. package/lib/api/hotswap/stepfunctions-state-machines.js +8 -6
  70. package/lib/api/logs/find-cloudwatch-logs.d.ts +1 -1
  71. package/lib/api/logs/find-cloudwatch-logs.js +2 -3
  72. package/lib/api/logs/logs-monitor.d.ts +2 -2
  73. package/lib/api/logs/logs-monitor.js +1 -1
  74. package/lib/api/resource-import/importer.d.ts +4 -4
  75. package/lib/api/resource-import/importer.js +1 -1
  76. package/lib/api/resource-import/migrator.d.ts +3 -3
  77. package/lib/api/resource-import/migrator.js +1 -1
  78. package/lib/api/stack-events/stack-activity-monitor.d.ts +1 -2
  79. package/lib/api/stack-events/stack-activity-monitor.js +3 -23
  80. package/lib/api/stack-events/stack-progress-monitor.d.ts +1 -1
  81. package/lib/api/stack-events/stack-progress-monitor.js +1 -1
  82. package/lib/api/toolkit-info.d.ts +2 -2
  83. package/lib/api/toolkit-info.js +1 -1
  84. package/lib/api/util/template-body-parameter.d.ts +1 -1
  85. package/lib/api/util/template-body-parameter.js +1 -1
  86. package/lib/api/work-graph/work-graph-builder.d.ts +2 -2
  87. package/lib/api/work-graph/work-graph-builder.js +1 -1
  88. package/lib/api/work-graph/work-graph-types.d.ts +2 -2
  89. package/lib/api/work-graph/work-graph-types.js +1 -1
  90. package/lib/api/work-graph/work-graph.d.ts +2 -2
  91. package/lib/api/work-graph/work-graph.js +1 -1
  92. package/lib/cli/activity-printer/base.d.ts +2 -2
  93. package/lib/cli/activity-printer/base.js +1 -1
  94. package/lib/cli/activity-printer/current.d.ts +2 -1
  95. package/lib/cli/activity-printer/current.js +1 -1
  96. package/lib/cli/activity-printer/history.d.ts +2 -1
  97. package/lib/cli/activity-printer/history.js +1 -1
  98. package/lib/cli/cdk-toolkit.d.ts +7 -7
  99. package/lib/cli/cdk-toolkit.js +10 -2
  100. package/lib/cli/ci-systems.d.ts +29 -0
  101. package/lib/cli/ci-systems.js +62 -0
  102. package/lib/cli/cli.d.ts +1 -1
  103. package/lib/cli/cli.js +22 -3
  104. package/lib/cli/convert-to-user-input.js +1 -1
  105. package/lib/cli/messages.d.ts +1 -1
  106. package/lib/cli/messages.js +1 -1
  107. package/lib/cli/parse-command-line-arguments.js +1 -1
  108. package/lib/cli/user-configuration.js +1 -1
  109. package/lib/cli/user-input.js +1 -1
  110. package/lib/commands/context.d.ts +1 -1
  111. package/lib/commands/context.js +1 -1
  112. package/lib/commands/migrate.d.ts +1 -1
  113. package/lib/commands/migrate.js +1 -1
  114. package/lib/context-providers/ami.d.ts +1 -1
  115. package/lib/context-providers/ami.js +1 -1
  116. package/lib/context-providers/availability-zones.d.ts +1 -1
  117. package/lib/context-providers/availability-zones.js +1 -1
  118. package/lib/context-providers/cc-api-provider.d.ts +1 -1
  119. package/lib/context-providers/cc-api-provider.js +1 -1
  120. package/lib/context-providers/endpoint-service-availability-zones.d.ts +1 -1
  121. package/lib/context-providers/endpoint-service-availability-zones.js +1 -1
  122. package/lib/context-providers/hosted-zones.d.ts +2 -2
  123. package/lib/context-providers/hosted-zones.js +1 -1
  124. package/lib/context-providers/index.d.ts +3 -3
  125. package/lib/context-providers/index.js +1 -1
  126. package/lib/context-providers/keys.d.ts +1 -1
  127. package/lib/context-providers/keys.js +1 -1
  128. package/lib/context-providers/load-balancers.d.ts +3 -3
  129. package/lib/context-providers/load-balancers.js +1 -1
  130. package/lib/context-providers/ssm-parameters.d.ts +1 -1
  131. package/lib/context-providers/ssm-parameters.js +1 -1
  132. package/lib/context-providers/vpcs.d.ts +1 -1
  133. package/lib/context-providers/vpcs.js +1 -1
  134. package/lib/diff.d.ts +1 -1
  135. package/lib/diff.js +1 -1
  136. package/lib/index.js +4028 -3727
  137. package/lib/legacy-logging-source.js +1 -1
  138. package/lib/list-stacks.d.ts +1 -1
  139. package/lib/list-stacks.js +1 -1
  140. package/lib/logging.d.ts +3 -3
  141. package/lib/logging.js +37 -35
  142. package/lib/notices.d.ts +22 -18
  143. package/lib/notices.js +46 -46
  144. package/lib/toolkit/cli-io-host.d.ts +17 -2
  145. package/lib/toolkit/cli-io-host.js +34 -5
  146. package/lib/tree.d.ts +1 -1
  147. package/lib/tree.js +1 -1
  148. package/package.json +3 -5
@@ -10,25 +10,15 @@ const evaluate_cloudformation_template_1 = require("../evaluate-cloudformation-t
10
10
  // eslint-disable-next-line @typescript-eslint/no-require-imports
11
11
  const archiver = require('archiver');
12
12
  async function isHotswappableLambdaFunctionChange(logicalId, change, evaluateCfnTemplate) {
13
- var _a;
14
- // if the change is for a Lambda Version,
15
- // ignore it by returning an empty hotswap operation -
16
- // we will publish a new version when we get to hotswapping the actual Function this Version points to, below
13
+ var _a, _b;
14
+ // if the change is for a Lambda Version, we just ignore it
15
+ // we will publish a new version when we get to hotswapping the actual Function this Version points to
17
16
  // (Versions can't be changed in CloudFormation anyway, they're immutable)
18
17
  if (change.newValue.Type === 'AWS::Lambda::Version') {
19
- return [
20
- {
21
- hotswappable: true,
22
- resourceType: 'AWS::Lambda::Version',
23
- resourceNames: [],
24
- propsChanged: [],
25
- service: 'lambda',
26
- apply: async (_sdk) => {
27
- },
28
- },
29
- ];
18
+ return [];
30
19
  }
31
20
  // we handle Aliases specially too
21
+ // the actual alias update will happen if we change the function
32
22
  if (change.newValue.Type === 'AWS::Lambda::Alias') {
33
23
  return classifyAliasChanges(change);
34
24
  }
@@ -40,29 +30,24 @@ async function isHotswappableLambdaFunctionChange(logicalId, change, evaluateCfn
40
30
  classifiedChanges.reportNonHotswappablePropertyChanges(ret);
41
31
  const functionName = await evaluateCfnTemplate.establishResourcePhysicalName(logicalId, (_a = change.newValue.Properties) === null || _a === void 0 ? void 0 : _a.FunctionName);
42
32
  const namesOfHotswappableChanges = Object.keys(classifiedChanges.hotswappableProps);
43
- if (namesOfHotswappableChanges.length > 0) {
33
+ if (functionName && namesOfHotswappableChanges.length > 0) {
34
+ const lambdaCodeChange = await evaluateLambdaFunctionProps(classifiedChanges.hotswappableProps, (_b = change.newValue.Properties) === null || _b === void 0 ? void 0 : _b.Runtime, evaluateCfnTemplate);
35
+ // nothing to do here
36
+ if (lambdaCodeChange === undefined) {
37
+ return ret;
38
+ }
39
+ const dependencies = await dependantResources(logicalId, functionName, evaluateCfnTemplate);
44
40
  ret.push({
41
+ change: {
42
+ cause: change,
43
+ },
45
44
  hotswappable: true,
46
- resourceType: change.newValue.Type,
47
- propsChanged: namesOfHotswappableChanges,
48
45
  service: 'lambda',
49
46
  resourceNames: [
50
47
  `Lambda Function '${functionName}'`,
51
- // add Version here if we're publishing a new one
52
- ...(await renderVersions(logicalId, evaluateCfnTemplate, [`Lambda Version for Function '${functionName}'`])),
53
- // add any Aliases that we are hotswapping here
54
- ...(await renderAliases(logicalId, evaluateCfnTemplate, async (alias) => `Lambda Alias '${alias}' for Function '${functionName}'`)),
48
+ ...dependencies.map(d => { var _a; return (_a = d.description) !== null && _a !== void 0 ? _a : `${d.resourceType} '${d.physicalName}'`; }),
55
49
  ],
56
50
  apply: async (sdk) => {
57
- var _a;
58
- const lambdaCodeChange = await evaluateLambdaFunctionProps(classifiedChanges.hotswappableProps, (_a = change.newValue.Properties) === null || _a === void 0 ? void 0 : _a.Runtime, evaluateCfnTemplate);
59
- if (lambdaCodeChange === undefined) {
60
- return;
61
- }
62
- if (!functionName) {
63
- return;
64
- }
65
- const { versionsReferencingFunction, aliasesNames } = await versionsAndAliases(logicalId, evaluateCfnTemplate);
66
51
  const lambda = sdk.lambda();
67
52
  const operations = [];
68
53
  if (lambdaCodeChange.code !== undefined || lambdaCodeChange.configurations !== undefined) {
@@ -91,17 +76,19 @@ async function isHotswappableLambdaFunctionChange(logicalId, change, evaluateCfn
91
76
  await waitForLambdasPropertiesUpdateToFinish(updateFunctionCodeResponse, lambda, functionName);
92
77
  }
93
78
  // only if the code changed is there any point in publishing a new Version
94
- if (versionsReferencingFunction.length > 0) {
79
+ const versions = dependencies.filter((d) => d.resourceType === 'AWS::Lambda::Version');
80
+ if (versions.length) {
95
81
  const publishVersionPromise = lambda.publishVersion({
96
82
  FunctionName: functionName,
97
83
  });
98
- if (aliasesNames.length > 0) {
84
+ const aliases = dependencies.filter((d) => d.resourceType === 'AWS::Lambda::Alias');
85
+ if (aliases.length) {
99
86
  // we need to wait for the Version to finish publishing
100
87
  const versionUpdate = await publishVersionPromise;
101
- for (const alias of aliasesNames) {
88
+ for (const alias of aliases) {
102
89
  operations.push(lambda.updateAlias({
103
90
  FunctionName: functionName,
104
- Name: alias,
91
+ Name: alias.physicalName,
105
92
  FunctionVersion: versionUpdate.Version,
106
93
  }));
107
94
  }
@@ -127,18 +114,8 @@ function classifyAliasChanges(change) {
127
114
  const ret = [];
128
115
  const classifiedChanges = (0, common_1.classifyChanges)(change, ['FunctionVersion']);
129
116
  classifiedChanges.reportNonHotswappablePropertyChanges(ret);
130
- const namesOfHotswappableChanges = Object.keys(classifiedChanges.hotswappableProps);
131
- if (namesOfHotswappableChanges.length > 0) {
132
- ret.push({
133
- hotswappable: true,
134
- resourceType: change.newValue.Type,
135
- propsChanged: [],
136
- service: 'lambda',
137
- resourceNames: [],
138
- apply: async (_sdk) => {
139
- },
140
- });
141
- }
117
+ // we only want to report not hotswappable changes to aliases
118
+ // the actual alias update will happen if we change the function
142
119
  return ret;
143
120
  }
144
121
  /**
@@ -287,25 +264,30 @@ async function versionsAndAliases(logicalId, evaluateCfnTemplate) {
287
264
  .filter((r) => r.Type === 'AWS::Lambda::Version');
288
265
  // find all Lambda Aliases that reference the above Versions
289
266
  const aliasesReferencingVersions = (0, util_1.flatMap)(versionsReferencingFunction, v => evaluateCfnTemplate.findReferencesTo(v.LogicalId));
290
- // Limited set of updates per function
291
- // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism
292
- const aliasesNames = await Promise.all(aliasesReferencingVersions.map(a => { var _a; return evaluateCfnTemplate.evaluateCfnExpression((_a = a.Properties) === null || _a === void 0 ? void 0 : _a.Name); }));
293
- return { versionsReferencingFunction, aliasesNames };
267
+ return { versionsReferencingFunction, aliasesReferencingVersions };
294
268
  }
295
- /**
296
- * Renders the string used in displaying Alias resource names that reference the specified Lambda Function
297
- */
298
- async function renderAliases(logicalId, evaluateCfnTemplate, callbackfn) {
299
- const aliasesNames = (await versionsAndAliases(logicalId, evaluateCfnTemplate)).aliasesNames;
269
+ async function dependantResources(logicalId, functionName, evaluateCfnTemplate) {
270
+ const candidates = await versionsAndAliases(logicalId, evaluateCfnTemplate);
300
271
  // Limited set of updates per function
301
272
  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism
302
- return Promise.all(aliasesNames.map(callbackfn));
303
- }
304
- /**
305
- * Renders the string used in displaying Version resource names that reference the specified Lambda Function
306
- */
307
- async function renderVersions(logicalId, evaluateCfnTemplate, versionString) {
308
- const versions = (await versionsAndAliases(logicalId, evaluateCfnTemplate)).versionsReferencingFunction;
309
- return versions.length > 0 ? versionString : [];
273
+ const aliases = await Promise.all(candidates.aliasesReferencingVersions.map(async (a) => {
274
+ var _a;
275
+ const name = await evaluateCfnTemplate.evaluateCfnExpression((_a = a.Properties) === null || _a === void 0 ? void 0 : _a.Name);
276
+ return {
277
+ logicalId: a.LogicalId,
278
+ physicalName: name,
279
+ resourceType: 'AWS::Lambda::Alias',
280
+ description: `Lambda Alias '${name}' for Function '${functionName}'`,
281
+ };
282
+ }));
283
+ const versions = candidates.versionsReferencingFunction.map((v) => ({
284
+ logicalId: v.LogicalId,
285
+ resourceType: v.Type,
286
+ description: `Lambda Version for Function '${functionName}'`,
287
+ }));
288
+ return [
289
+ ...versions,
290
+ ...aliases,
291
+ ];
310
292
  }
311
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-functions.js","sourceRoot":"","sources":["lambda-functions.ts"],"names":[],"mappings":"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","sourcesContent":["import { Writable } from 'stream';\nimport { type FunctionConfiguration, type UpdateFunctionConfigurationCommandInput } from '@aws-sdk/client-lambda';\nimport { type ChangeHotswapResult, classifyChanges, type HotswappableChangeCandidate, PropDiffs } from './common';\nimport { ToolkitError } from '../../toolkit/error';\nimport { flatMap } from '../../util';\nimport type { ILambdaClient, SDK } from '../aws-auth';\nimport { CfnEvaluationException, type EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';\n\n// namespace object imports won't work in the bundle for function exports\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nconst archiver = require('archiver');\n\nexport async function isHotswappableLambdaFunctionChange(\n  logicalId: string,\n  change: HotswappableChangeCandidate,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<ChangeHotswapResult> {\n  // if the change is for a Lambda Version,\n  // ignore it by returning an empty hotswap operation -\n  // we will publish a new version when we get to hotswapping the actual Function this Version points to, below\n  // (Versions can't be changed in CloudFormation anyway, they're immutable)\n  if (change.newValue.Type === 'AWS::Lambda::Version') {\n    return [\n      {\n        hotswappable: true,\n        resourceType: 'AWS::Lambda::Version',\n        resourceNames: [],\n        propsChanged: [],\n        service: 'lambda',\n        apply: async (_sdk: SDK) => {\n        },\n      },\n    ];\n  }\n\n  // we handle Aliases specially too\n  if (change.newValue.Type === 'AWS::Lambda::Alias') {\n    return classifyAliasChanges(change);\n  }\n\n  if (change.newValue.Type !== 'AWS::Lambda::Function') {\n    return [];\n  }\n\n  const ret: ChangeHotswapResult = [];\n  const classifiedChanges = classifyChanges(change, ['Code', 'Environment', 'Description']);\n  classifiedChanges.reportNonHotswappablePropertyChanges(ret);\n\n  const functionName = await evaluateCfnTemplate.establishResourcePhysicalName(\n    logicalId,\n    change.newValue.Properties?.FunctionName,\n  );\n  const namesOfHotswappableChanges = Object.keys(classifiedChanges.hotswappableProps);\n  if (namesOfHotswappableChanges.length > 0) {\n    ret.push({\n      hotswappable: true,\n      resourceType: change.newValue.Type,\n      propsChanged: namesOfHotswappableChanges,\n      service: 'lambda',\n      resourceNames: [\n        `Lambda Function '${functionName}'`,\n        // add Version here if we're publishing a new one\n        ...(await renderVersions(logicalId, evaluateCfnTemplate, [`Lambda Version for Function '${functionName}'`])),\n        // add any Aliases that we are hotswapping here\n        ...(await renderAliases(\n          logicalId,\n          evaluateCfnTemplate,\n          async (alias) => `Lambda Alias '${alias}' for Function '${functionName}'`,\n        )),\n      ],\n      apply: async (sdk: SDK) => {\n        const lambdaCodeChange = await evaluateLambdaFunctionProps(\n          classifiedChanges.hotswappableProps,\n          change.newValue.Properties?.Runtime,\n          evaluateCfnTemplate,\n        );\n        if (lambdaCodeChange === undefined) {\n          return;\n        }\n\n        if (!functionName) {\n          return;\n        }\n\n        const { versionsReferencingFunction, aliasesNames } = await versionsAndAliases(logicalId, evaluateCfnTemplate);\n        const lambda = sdk.lambda();\n        const operations: Promise<any>[] = [];\n\n        if (lambdaCodeChange.code !== undefined || lambdaCodeChange.configurations !== undefined) {\n          if (lambdaCodeChange.code !== undefined) {\n            const updateFunctionCodeResponse = await lambda.updateFunctionCode({\n              FunctionName: functionName,\n              S3Bucket: lambdaCodeChange.code.s3Bucket,\n              S3Key: lambdaCodeChange.code.s3Key,\n              ImageUri: lambdaCodeChange.code.imageUri,\n              ZipFile: lambdaCodeChange.code.functionCodeZip,\n              S3ObjectVersion: lambdaCodeChange.code.s3ObjectVersion,\n            });\n\n            await waitForLambdasPropertiesUpdateToFinish(updateFunctionCodeResponse, lambda, functionName);\n          }\n\n          if (lambdaCodeChange.configurations !== undefined) {\n            const updateRequest: UpdateFunctionConfigurationCommandInput = {\n              FunctionName: functionName,\n            };\n            if (lambdaCodeChange.configurations.description !== undefined) {\n              updateRequest.Description = lambdaCodeChange.configurations.description;\n            }\n            if (lambdaCodeChange.configurations.environment !== undefined) {\n              updateRequest.Environment = lambdaCodeChange.configurations.environment;\n            }\n            const updateFunctionCodeResponse = await lambda.updateFunctionConfiguration(updateRequest);\n            await waitForLambdasPropertiesUpdateToFinish(updateFunctionCodeResponse, lambda, functionName);\n          }\n\n          // only if the code changed is there any point in publishing a new Version\n          if (versionsReferencingFunction.length > 0) {\n            const publishVersionPromise = lambda.publishVersion({\n              FunctionName: functionName,\n            });\n\n            if (aliasesNames.length > 0) {\n              // we need to wait for the Version to finish publishing\n              const versionUpdate = await publishVersionPromise;\n              for (const alias of aliasesNames) {\n                operations.push(\n                  lambda.updateAlias({\n                    FunctionName: functionName,\n                    Name: alias,\n                    FunctionVersion: versionUpdate.Version,\n                  }),\n                );\n              }\n            } else {\n              operations.push(publishVersionPromise);\n            }\n          }\n        }\n\n        // run all of our updates in parallel\n        // Limited set of updates per function\n        // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n        await Promise.all(operations);\n      },\n    });\n  }\n\n  return ret;\n}\n\n/**\n * Determines which changes to this Alias are hotswappable or not\n */\nfunction classifyAliasChanges(change: HotswappableChangeCandidate): ChangeHotswapResult {\n  const ret: ChangeHotswapResult = [];\n  const classifiedChanges = classifyChanges(change, ['FunctionVersion']);\n  classifiedChanges.reportNonHotswappablePropertyChanges(ret);\n\n  const namesOfHotswappableChanges = Object.keys(classifiedChanges.hotswappableProps);\n  if (namesOfHotswappableChanges.length > 0) {\n    ret.push({\n      hotswappable: true,\n      resourceType: change.newValue.Type,\n      propsChanged: [],\n      service: 'lambda',\n      resourceNames: [],\n      apply: async (_sdk: SDK) => {\n      },\n    });\n  }\n\n  return ret;\n}\n\n/**\n * Evaluates the hotswappable properties of an AWS::Lambda::Function and\n * Returns a `LambdaFunctionChange` if the change is hotswappable.\n * Returns `undefined` if the change is not hotswappable.\n */\nasync function evaluateLambdaFunctionProps(\n  hotswappablePropChanges: PropDiffs,\n  runtime: string,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<LambdaFunctionChange | undefined> {\n  /*\n   * At first glance, we would want to initialize these using the \"previous\" values (change.oldValue),\n   * in case only one of them changed, like the key, and the Bucket stayed the same.\n   * However, that actually fails for old-style synthesis, which uses CFN Parameters!\n   * Because the names of the Parameters depend on the hash of the Asset,\n   * the Parameters used for the \"old\" values no longer exist in `assetParams` at this point,\n   * which means we don't have the correct values available to evaluate the CFN expression with.\n   * Fortunately, the diff will always include both the s3Bucket and s3Key parts of the Lambda's Code property,\n   * even if only one of them was actually changed,\n   * which means we don't need the \"old\" values at all, and we can safely initialize these with just `''`.\n   */\n  let code: LambdaFunctionCode | undefined = undefined;\n  let description: string | undefined = undefined;\n  let environment: { [key: string]: string } | undefined = undefined;\n\n  for (const updatedPropName in hotswappablePropChanges) {\n    const updatedProp = hotswappablePropChanges[updatedPropName];\n\n    switch (updatedPropName) {\n      case 'Code':\n        let s3Bucket, s3Key, s3ObjectVersion, imageUri, functionCodeZip;\n\n        for (const newPropName in updatedProp.newValue) {\n          switch (newPropName) {\n            case 'S3Bucket':\n              s3Bucket = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n              break;\n            case 'S3Key':\n              s3Key = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n              break;\n            case 'S3ObjectVersion':\n              s3ObjectVersion = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n              break;\n            case 'ImageUri':\n              imageUri = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n              break;\n            case 'ZipFile':\n              // We must create a zip package containing a file with the inline code\n              const functionCode = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n              const functionRuntime = await evaluateCfnTemplate.evaluateCfnExpression(runtime);\n              if (!functionRuntime) {\n                return undefined;\n              }\n              // file extension must be chosen depending on the runtime\n              const codeFileExt = determineCodeFileExtFromRuntime(functionRuntime);\n              functionCodeZip = await zipString(`index.${codeFileExt}`, functionCode);\n              break;\n          }\n        }\n        code = {\n          s3Bucket,\n          s3Key,\n          s3ObjectVersion,\n          imageUri,\n          functionCodeZip,\n        };\n        break;\n      case 'Description':\n        description = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue);\n        break;\n      case 'Environment':\n        environment = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue);\n        break;\n      default:\n        // we will never get here, but just in case we do throw an error\n        throw new ToolkitError(\n          'while apply()ing, found a property that cannot be hotswapped. Please report this at github.com/aws/aws-cdk/issues/new/choose',\n        );\n    }\n  }\n\n  const configurations = description || environment ? { description, environment } : undefined;\n  return code || configurations ? { code, configurations } : undefined;\n}\n\ninterface LambdaFunctionCode {\n  readonly s3Bucket?: string;\n  readonly s3Key?: string;\n  readonly s3ObjectVersion?: string;\n  readonly imageUri?: string;\n  readonly functionCodeZip?: Buffer;\n}\n\ninterface LambdaFunctionConfigurations {\n  readonly description?: string;\n  readonly environment?: { [key: string]: string };\n}\n\ninterface LambdaFunctionChange {\n  readonly code?: LambdaFunctionCode;\n  readonly configurations?: LambdaFunctionConfigurations;\n}\n\n/**\n * Compress a string as a file, returning a promise for the zip buffer\n * https://github.com/archiverjs/node-archiver/issues/342\n */\nfunction zipString(fileName: string, rawString: string): Promise<Buffer> {\n  return new Promise((resolve, reject) => {\n    const buffers: Buffer[] = [];\n\n    const converter = new Writable();\n\n    converter._write = (chunk: Buffer, _: string, callback: () => void) => {\n      buffers.push(chunk);\n      process.nextTick(callback);\n    };\n\n    converter.on('finish', () => {\n      resolve(Buffer.concat(buffers));\n    });\n\n    const archive = archiver('zip');\n\n    archive.on('error', (err: any) => {\n      reject(err);\n    });\n\n    archive.pipe(converter);\n\n    archive.append(rawString, {\n      name: fileName,\n      date: new Date('1980-01-01T00:00:00.000Z'), // Add date to make resulting zip file deterministic\n    });\n\n    void archive.finalize();\n  });\n}\n\n/**\n * After a Lambda Function is updated, it cannot be updated again until the\n * `State=Active` and the `LastUpdateStatus=Successful`.\n *\n * Depending on the configuration of the Lambda Function this could happen relatively quickly\n * or very slowly. For example, Zip based functions _not_ in a VPC can take ~1 second whereas VPC\n * or Container functions can take ~25 seconds (and 'idle' VPC functions can take minutes).\n */\nasync function waitForLambdasPropertiesUpdateToFinish(\n  currentFunctionConfiguration: FunctionConfiguration,\n  lambda: ILambdaClient,\n  functionName: string,\n): Promise<void> {\n  const functionIsInVpcOrUsesDockerForCode =\n    currentFunctionConfiguration.VpcConfig?.VpcId || currentFunctionConfiguration.PackageType === 'Image';\n\n  // if the function is deployed in a VPC or if it is a container image function\n  // then the update will take much longer and we can wait longer between checks\n  // otherwise, the update will be quick, so a 1-second delay is fine\n  const delaySeconds = functionIsInVpcOrUsesDockerForCode ? 5 : 1;\n\n  await lambda.waitUntilFunctionUpdated(delaySeconds, {\n    FunctionName: functionName,\n  });\n}\n\n/**\n * Get file extension from Lambda runtime string.\n * We use this extension to create a deployment package from Lambda inline code.\n */\nfunction determineCodeFileExtFromRuntime(runtime: string): string {\n  if (runtime.startsWith('node')) {\n    return 'js';\n  }\n  if (runtime.startsWith('python')) {\n    return 'py';\n  }\n  // Currently inline code only supports Node.js and Python, ignoring other runtimes.\n  // https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-properties-lambda-function-code.html#aws-properties-lambda-function-code-properties\n  throw new CfnEvaluationException(\n    `runtime ${runtime} is unsupported, only node.js and python runtimes are currently supported.`,\n  );\n}\n\n/**\n * Finds all Versions that reference an AWS::Lambda::Function with logical ID `logicalId`\n * and Aliases that reference those Versions.\n */\nasync function versionsAndAliases(logicalId: string, evaluateCfnTemplate: EvaluateCloudFormationTemplate) {\n  // find all Lambda Versions that reference this Function\n  const versionsReferencingFunction = evaluateCfnTemplate\n    .findReferencesTo(logicalId)\n    .filter((r) => r.Type === 'AWS::Lambda::Version');\n  // find all Lambda Aliases that reference the above Versions\n  const aliasesReferencingVersions = flatMap(versionsReferencingFunction, v =>\n    evaluateCfnTemplate.findReferencesTo(v.LogicalId));\n  // Limited set of updates per function\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  const aliasesNames = await Promise.all(aliasesReferencingVersions.map(a =>\n    evaluateCfnTemplate.evaluateCfnExpression(a.Properties?.Name)));\n\n  return { versionsReferencingFunction, aliasesNames };\n}\n\n/**\n * Renders the string used in displaying Alias resource names that reference the specified Lambda Function\n */\nasync function renderAliases(\n  logicalId: string,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  callbackfn: (value: any, index: number, array: any[]) => Promise<string>,\n): Promise<string[]> {\n  const aliasesNames = (await versionsAndAliases(logicalId, evaluateCfnTemplate)).aliasesNames;\n\n  // Limited set of updates per function\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  return Promise.all(aliasesNames.map(callbackfn));\n}\n\n/**\n * Renders the string used in displaying Version resource names that reference the specified Lambda Function\n */\nasync function renderVersions(\n  logicalId: string,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  versionString: string[],\n): Promise<string[]> {\n  const versions = (await versionsAndAliases(logicalId, evaluateCfnTemplate)).versionsReferencingFunction;\n\n  return versions.length > 0 ? versionString : [];\n}\n"]}
293
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-functions.js","sourceRoot":"","sources":["lambda-functions.ts"],"names":[],"mappings":"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","sourcesContent":["import { Writable } from 'stream';\nimport type { PropertyDifference } from '@aws-cdk/cloudformation-diff';\nimport type { FunctionConfiguration, UpdateFunctionConfigurationCommandInput } from '@aws-sdk/client-lambda';\nimport type { ChangeHotswapResult } from './common';\nimport { classifyChanges } from './common';\nimport type { AffectedResource, ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';\nimport { ToolkitError } from '../../toolkit/error';\nimport { flatMap } from '../../util';\nimport type { ILambdaClient, SDK } from '../aws-auth';\nimport { CfnEvaluationException, type EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';\n\n// namespace object imports won't work in the bundle for function exports\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nconst archiver = require('archiver');\n\nexport async function isHotswappableLambdaFunctionChange(\n  logicalId: string,\n  change: ResourceChange,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<ChangeHotswapResult> {\n  // if the change is for a Lambda Version, we just ignore it\n  // we will publish a new version when we get to hotswapping the actual Function this Version points to\n  // (Versions can't be changed in CloudFormation anyway, they're immutable)\n  if (change.newValue.Type === 'AWS::Lambda::Version') {\n    return [];\n  }\n\n  // we handle Aliases specially too\n  // the actual alias update will happen if we change the function\n  if (change.newValue.Type === 'AWS::Lambda::Alias') {\n    return classifyAliasChanges(change);\n  }\n\n  if (change.newValue.Type !== 'AWS::Lambda::Function') {\n    return [];\n  }\n\n  const ret: ChangeHotswapResult = [];\n  const classifiedChanges = classifyChanges(change, ['Code', 'Environment', 'Description']);\n  classifiedChanges.reportNonHotswappablePropertyChanges(ret);\n\n  const functionName = await evaluateCfnTemplate.establishResourcePhysicalName(\n    logicalId,\n    change.newValue.Properties?.FunctionName,\n  );\n  const namesOfHotswappableChanges = Object.keys(classifiedChanges.hotswappableProps);\n  if (functionName && namesOfHotswappableChanges.length > 0) {\n    const lambdaCodeChange = await evaluateLambdaFunctionProps(\n      classifiedChanges.hotswappableProps,\n      change.newValue.Properties?.Runtime,\n      evaluateCfnTemplate,\n    );\n\n    // nothing to do here\n    if (lambdaCodeChange === undefined) {\n      return ret;\n    }\n\n    const dependencies = await dependantResources(logicalId, functionName, evaluateCfnTemplate);\n\n    ret.push({\n      change: {\n        cause: change,\n      },\n      hotswappable: true,\n      service: 'lambda',\n      resourceNames: [\n        `Lambda Function '${functionName}'`,\n        ...dependencies.map(d => d.description ?? `${d.resourceType} '${d.physicalName}'`),\n      ],\n      apply: async (sdk: SDK) => {\n        const lambda = sdk.lambda();\n        const operations: Promise<any>[] = [];\n\n        if (lambdaCodeChange.code !== undefined || lambdaCodeChange.configurations !== undefined) {\n          if (lambdaCodeChange.code !== undefined) {\n            const updateFunctionCodeResponse = await lambda.updateFunctionCode({\n              FunctionName: functionName,\n              S3Bucket: lambdaCodeChange.code.s3Bucket,\n              S3Key: lambdaCodeChange.code.s3Key,\n              ImageUri: lambdaCodeChange.code.imageUri,\n              ZipFile: lambdaCodeChange.code.functionCodeZip,\n              S3ObjectVersion: lambdaCodeChange.code.s3ObjectVersion,\n            });\n\n            await waitForLambdasPropertiesUpdateToFinish(updateFunctionCodeResponse, lambda, functionName);\n          }\n\n          if (lambdaCodeChange.configurations !== undefined) {\n            const updateRequest: UpdateFunctionConfigurationCommandInput = {\n              FunctionName: functionName,\n            };\n            if (lambdaCodeChange.configurations.description !== undefined) {\n              updateRequest.Description = lambdaCodeChange.configurations.description;\n            }\n            if (lambdaCodeChange.configurations.environment !== undefined) {\n              updateRequest.Environment = lambdaCodeChange.configurations.environment;\n            }\n            const updateFunctionCodeResponse = await lambda.updateFunctionConfiguration(updateRequest);\n            await waitForLambdasPropertiesUpdateToFinish(updateFunctionCodeResponse, lambda, functionName);\n          }\n\n          // only if the code changed is there any point in publishing a new Version\n          const versions = dependencies.filter((d) => d.resourceType === 'AWS::Lambda::Version');\n          if (versions.length) {\n            const publishVersionPromise = lambda.publishVersion({\n              FunctionName: functionName,\n            });\n\n            const aliases = dependencies.filter((d) => d.resourceType === 'AWS::Lambda::Alias');\n            if (aliases.length) {\n              // we need to wait for the Version to finish publishing\n              const versionUpdate = await publishVersionPromise;\n              for (const alias of aliases) {\n                operations.push(\n                  lambda.updateAlias({\n                    FunctionName: functionName,\n                    Name: alias.physicalName,\n                    FunctionVersion: versionUpdate.Version,\n                  }),\n                );\n              }\n            } else {\n              operations.push(publishVersionPromise);\n            }\n          }\n        }\n\n        // run all of our updates in parallel\n        // Limited set of updates per function\n        // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n        await Promise.all(operations);\n      },\n    });\n  }\n\n  return ret;\n}\n\n/**\n * Determines which changes to this Alias are hotswappable or not\n */\nfunction classifyAliasChanges(change: ResourceChange): ChangeHotswapResult {\n  const ret: ChangeHotswapResult = [];\n  const classifiedChanges = classifyChanges(change, ['FunctionVersion']);\n  classifiedChanges.reportNonHotswappablePropertyChanges(ret);\n\n  // we only want to report not hotswappable changes to aliases\n  // the actual alias update will happen if we change the function\n\n  return ret;\n}\n\n/**\n * Evaluates the hotswappable properties of an AWS::Lambda::Function and\n * Returns a `LambdaFunctionChange` if the change is hotswappable.\n * Returns `undefined` if the change is not hotswappable.\n */\nasync function evaluateLambdaFunctionProps(\n  hotswappablePropChanges: Record<string, PropertyDifference<any>>,\n  runtime: string,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<LambdaFunctionChange | undefined> {\n  /*\n   * At first glance, we would want to initialize these using the \"previous\" values (change.oldValue),\n   * in case only one of them changed, like the key, and the Bucket stayed the same.\n   * However, that actually fails for old-style synthesis, which uses CFN Parameters!\n   * Because the names of the Parameters depend on the hash of the Asset,\n   * the Parameters used for the \"old\" values no longer exist in `assetParams` at this point,\n   * which means we don't have the correct values available to evaluate the CFN expression with.\n   * Fortunately, the diff will always include both the s3Bucket and s3Key parts of the Lambda's Code property,\n   * even if only one of them was actually changed,\n   * which means we don't need the \"old\" values at all, and we can safely initialize these with just `''`.\n   */\n  let code: LambdaFunctionCode | undefined = undefined;\n  let description: string | undefined = undefined;\n  let environment: { [key: string]: string } | undefined = undefined;\n\n  for (const updatedPropName in hotswappablePropChanges) {\n    const updatedProp = hotswappablePropChanges[updatedPropName];\n\n    switch (updatedPropName) {\n      case 'Code':\n        let s3Bucket, s3Key, s3ObjectVersion, imageUri, functionCodeZip;\n\n        for (const newPropName in updatedProp.newValue) {\n          switch (newPropName) {\n            case 'S3Bucket':\n              s3Bucket = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n              break;\n            case 'S3Key':\n              s3Key = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n              break;\n            case 'S3ObjectVersion':\n              s3ObjectVersion = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n              break;\n            case 'ImageUri':\n              imageUri = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n              break;\n            case 'ZipFile':\n              // We must create a zip package containing a file with the inline code\n              const functionCode = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n              const functionRuntime = await evaluateCfnTemplate.evaluateCfnExpression(runtime);\n              if (!functionRuntime) {\n                return undefined;\n              }\n              // file extension must be chosen depending on the runtime\n              const codeFileExt = determineCodeFileExtFromRuntime(functionRuntime);\n              functionCodeZip = await zipString(`index.${codeFileExt}`, functionCode);\n              break;\n          }\n        }\n        code = {\n          s3Bucket,\n          s3Key,\n          s3ObjectVersion,\n          imageUri,\n          functionCodeZip,\n        };\n        break;\n      case 'Description':\n        description = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue);\n        break;\n      case 'Environment':\n        environment = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue);\n        break;\n      default:\n        // we will never get here, but just in case we do throw an error\n        throw new ToolkitError(\n          'while apply()ing, found a property that cannot be hotswapped. Please report this at github.com/aws/aws-cdk/issues/new/choose',\n        );\n    }\n  }\n\n  const configurations = description || environment ? { description, environment } : undefined;\n  return code || configurations ? { code, configurations } : undefined;\n}\n\ninterface LambdaFunctionCode {\n  readonly s3Bucket?: string;\n  readonly s3Key?: string;\n  readonly s3ObjectVersion?: string;\n  readonly imageUri?: string;\n  readonly functionCodeZip?: Buffer;\n}\n\ninterface LambdaFunctionConfigurations {\n  readonly description?: string;\n  readonly environment?: { [key: string]: string };\n}\n\ninterface LambdaFunctionChange {\n  readonly code?: LambdaFunctionCode;\n  readonly configurations?: LambdaFunctionConfigurations;\n}\n\n/**\n * Compress a string as a file, returning a promise for the zip buffer\n * https://github.com/archiverjs/node-archiver/issues/342\n */\nfunction zipString(fileName: string, rawString: string): Promise<Buffer> {\n  return new Promise((resolve, reject) => {\n    const buffers: Buffer[] = [];\n\n    const converter = new Writable();\n\n    converter._write = (chunk: Buffer, _: string, callback: () => void) => {\n      buffers.push(chunk);\n      process.nextTick(callback);\n    };\n\n    converter.on('finish', () => {\n      resolve(Buffer.concat(buffers));\n    });\n\n    const archive = archiver('zip');\n\n    archive.on('error', (err: any) => {\n      reject(err);\n    });\n\n    archive.pipe(converter);\n\n    archive.append(rawString, {\n      name: fileName,\n      date: new Date('1980-01-01T00:00:00.000Z'), // Add date to make resulting zip file deterministic\n    });\n\n    void archive.finalize();\n  });\n}\n\n/**\n * After a Lambda Function is updated, it cannot be updated again until the\n * `State=Active` and the `LastUpdateStatus=Successful`.\n *\n * Depending on the configuration of the Lambda Function this could happen relatively quickly\n * or very slowly. For example, Zip based functions _not_ in a VPC can take ~1 second whereas VPC\n * or Container functions can take ~25 seconds (and 'idle' VPC functions can take minutes).\n */\nasync function waitForLambdasPropertiesUpdateToFinish(\n  currentFunctionConfiguration: FunctionConfiguration,\n  lambda: ILambdaClient,\n  functionName: string,\n): Promise<void> {\n  const functionIsInVpcOrUsesDockerForCode =\n    currentFunctionConfiguration.VpcConfig?.VpcId || currentFunctionConfiguration.PackageType === 'Image';\n\n  // if the function is deployed in a VPC or if it is a container image function\n  // then the update will take much longer and we can wait longer between checks\n  // otherwise, the update will be quick, so a 1-second delay is fine\n  const delaySeconds = functionIsInVpcOrUsesDockerForCode ? 5 : 1;\n\n  await lambda.waitUntilFunctionUpdated(delaySeconds, {\n    FunctionName: functionName,\n  });\n}\n\n/**\n * Get file extension from Lambda runtime string.\n * We use this extension to create a deployment package from Lambda inline code.\n */\nfunction determineCodeFileExtFromRuntime(runtime: string): string {\n  if (runtime.startsWith('node')) {\n    return 'js';\n  }\n  if (runtime.startsWith('python')) {\n    return 'py';\n  }\n  // Currently inline code only supports Node.js and Python, ignoring other runtimes.\n  // https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-properties-lambda-function-code.html#aws-properties-lambda-function-code-properties\n  throw new CfnEvaluationException(\n    `runtime ${runtime} is unsupported, only node.js and python runtimes are currently supported.`,\n  );\n}\n\n/**\n * Finds all Versions that reference an AWS::Lambda::Function with logical ID `logicalId`\n * and Aliases that reference those Versions.\n */\nasync function versionsAndAliases(logicalId: string, evaluateCfnTemplate: EvaluateCloudFormationTemplate) {\n  // find all Lambda Versions that reference this Function\n  const versionsReferencingFunction = evaluateCfnTemplate\n    .findReferencesTo(logicalId)\n    .filter((r) => r.Type === 'AWS::Lambda::Version');\n  // find all Lambda Aliases that reference the above Versions\n  const aliasesReferencingVersions = flatMap(versionsReferencingFunction, v =>\n    evaluateCfnTemplate.findReferencesTo(v.LogicalId));\n\n  return { versionsReferencingFunction, aliasesReferencingVersions };\n}\n\nasync function dependantResources(\n  logicalId: string,\n  functionName: string,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<Array<AffectedResource>> {\n  const candidates = await versionsAndAliases(logicalId, evaluateCfnTemplate);\n\n  // Limited set of updates per function\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  const aliases = await Promise.all(candidates.aliasesReferencingVersions.map(async (a) => {\n    const name = await evaluateCfnTemplate.evaluateCfnExpression(a.Properties?.Name);\n    return {\n      logicalId: a.LogicalId,\n      physicalName: name,\n      resourceType: 'AWS::Lambda::Alias',\n      description: `Lambda Alias '${name}' for Function '${functionName}'`,\n    };\n  }));\n\n  const versions = candidates.versionsReferencingFunction.map((v) => (\n    {\n      logicalId: v.LogicalId,\n      resourceType: v.Type,\n      description: `Lambda Version for Function '${functionName}'`,\n    }\n  ));\n\n  return [\n    ...versions,\n    ...aliases,\n  ];\n}\n"]}
@@ -1,9 +1,5 @@
1
- import type { ChangeHotswapResult, HotswappableChangeCandidate } from './common';
1
+ import type { ChangeHotswapResult } from './common';
2
+ import type { ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';
2
3
  import type { EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';
3
- /**
4
- * This means that the value is required to exist by CloudFormation's Custom Resource API (or our S3 Bucket Deployment Lambda's API)
5
- * but the actual value specified is irrelevant
6
- */
7
- export declare const REQUIRED_BY_CFN = "required-to-be-present-by-cfn";
8
- export declare function isHotswappableS3BucketDeploymentChange(_logicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate): Promise<ChangeHotswapResult>;
9
- export declare function skipChangeForS3DeployCustomResourcePolicy(iamPolicyLogicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate): Promise<boolean>;
4
+ export declare function isHotswappableS3BucketDeploymentChange(_logicalId: string, change: ResourceChange, evaluateCfnTemplate: EvaluateCloudFormationTemplate): Promise<ChangeHotswapResult>;
5
+ export declare function skipChangeForS3DeployCustomResourcePolicy(iamPolicyLogicalId: string, change: ResourceChange, evaluateCfnTemplate: EvaluateCloudFormationTemplate): Promise<boolean>;
@@ -1,18 +1,19 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.REQUIRED_BY_CFN = void 0;
4
3
  exports.isHotswappableS3BucketDeploymentChange = isHotswappableS3BucketDeploymentChange;
5
4
  exports.skipChangeForS3DeployCustomResourcePolicy = skipChangeForS3DeployCustomResourcePolicy;
6
5
  /**
7
6
  * This means that the value is required to exist by CloudFormation's Custom Resource API (or our S3 Bucket Deployment Lambda's API)
8
7
  * but the actual value specified is irrelevant
9
8
  */
10
- exports.REQUIRED_BY_CFN = 'required-to-be-present-by-cfn';
9
+ const REQUIRED_BY_CFN = 'required-to-be-present-by-cfn';
10
+ const CDK_BUCKET_DEPLOYMENT_CFN_TYPE = 'Custom::CDKBucketDeployment';
11
11
  async function isHotswappableS3BucketDeploymentChange(_logicalId, change, evaluateCfnTemplate) {
12
+ var _a;
12
13
  // In old-style synthesis, the policy used by the lambda to copy assets Ref's the assets directly,
13
14
  // meaning that the changes made to the Policy are artifacts that can be safely ignored
14
15
  const ret = [];
15
- if (change.newValue.Type !== 'Custom::CDKBucketDeployment') {
16
+ if (change.newValue.Type !== CDK_BUCKET_DEPLOYMENT_CFN_TYPE) {
16
17
  return [];
17
18
  }
18
19
  // no classification to be done here; all the properties of this custom resource thing are hotswappable
@@ -20,29 +21,29 @@ async function isHotswappableS3BucketDeploymentChange(_logicalId, change, evalua
20
21
  ...change.newValue.Properties,
21
22
  ServiceToken: undefined,
22
23
  });
24
+ // note that this gives the ARN of the lambda, not the name. This is fine though, the invoke() sdk call will take either
25
+ const functionName = await evaluateCfnTemplate.evaluateCfnExpression((_a = change.newValue.Properties) === null || _a === void 0 ? void 0 : _a.ServiceToken);
26
+ if (!functionName) {
27
+ return ret;
28
+ }
23
29
  ret.push({
30
+ change: {
31
+ cause: change,
32
+ },
24
33
  hotswappable: true,
25
- resourceType: change.newValue.Type,
26
- propsChanged: ['*'],
27
34
  service: 'custom-s3-deployment',
28
35
  resourceNames: [`Contents of S3 Bucket '${customResourceProperties.DestinationBucketName}'`],
29
36
  apply: async (sdk) => {
30
- var _a;
31
- // note that this gives the ARN of the lambda, not the name. This is fine though, the invoke() sdk call will take either
32
- const functionName = await evaluateCfnTemplate.evaluateCfnExpression((_a = change.newValue.Properties) === null || _a === void 0 ? void 0 : _a.ServiceToken);
33
- if (!functionName) {
34
- return;
35
- }
36
37
  await sdk.lambda().invokeCommand({
37
38
  FunctionName: functionName,
38
39
  // Lambda refuses to take a direct JSON object and requires it to be stringify()'d
39
40
  Payload: JSON.stringify({
40
41
  RequestType: 'Update',
41
- ResponseURL: exports.REQUIRED_BY_CFN,
42
- PhysicalResourceId: exports.REQUIRED_BY_CFN,
43
- StackId: exports.REQUIRED_BY_CFN,
44
- RequestId: exports.REQUIRED_BY_CFN,
45
- LogicalResourceId: exports.REQUIRED_BY_CFN,
42
+ ResponseURL: REQUIRED_BY_CFN,
43
+ PhysicalResourceId: REQUIRED_BY_CFN,
44
+ StackId: REQUIRED_BY_CFN,
45
+ RequestId: REQUIRED_BY_CFN,
46
+ LogicalResourceId: REQUIRED_BY_CFN,
46
47
  ResourceProperties: stringifyObject(customResourceProperties), // JSON.stringify() doesn't turn the actual objects to strings, but the lambda expects strings
47
48
  }),
48
49
  });
@@ -109,4 +110,4 @@ function stringifyObject(obj) {
109
110
  }
110
111
  return ret;
111
112
  }
112
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"s3-bucket-deployments.js","sourceRoot":"","sources":["s3-bucket-deployments.ts"],"names":[],"mappings":"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","sourcesContent":["import type { ChangeHotswapResult, HotswappableChangeCandidate } from './common';\nimport type { SDK } from '../aws-auth';\nimport type { EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';\n\n/**\n * This means that the value is required to exist by CloudFormation's Custom Resource API (or our S3 Bucket Deployment Lambda's API)\n * but the actual value specified is irrelevant\n */\nexport const REQUIRED_BY_CFN = 'required-to-be-present-by-cfn';\n\nexport async function isHotswappableS3BucketDeploymentChange(\n  _logicalId: string,\n  change: HotswappableChangeCandidate,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<ChangeHotswapResult> {\n  // In old-style synthesis, the policy used by the lambda to copy assets Ref's the assets directly,\n  // meaning that the changes made to the Policy are artifacts that can be safely ignored\n  const ret: ChangeHotswapResult = [];\n\n  if (change.newValue.Type !== 'Custom::CDKBucketDeployment') {\n    return [];\n  }\n\n  // no classification to be done here; all the properties of this custom resource thing are hotswappable\n  const customResourceProperties = await evaluateCfnTemplate.evaluateCfnExpression({\n    ...change.newValue.Properties,\n    ServiceToken: undefined,\n  });\n\n  ret.push({\n    hotswappable: true,\n    resourceType: change.newValue.Type,\n    propsChanged: ['*'],\n    service: 'custom-s3-deployment',\n    resourceNames: [`Contents of S3 Bucket '${customResourceProperties.DestinationBucketName}'`],\n    apply: async (sdk: SDK) => {\n      // note that this gives the ARN of the lambda, not the name. This is fine though, the invoke() sdk call will take either\n      const functionName = await evaluateCfnTemplate.evaluateCfnExpression(change.newValue.Properties?.ServiceToken);\n      if (!functionName) {\n        return;\n      }\n\n      await sdk.lambda().invokeCommand({\n        FunctionName: functionName,\n        // Lambda refuses to take a direct JSON object and requires it to be stringify()'d\n        Payload: JSON.stringify({\n          RequestType: 'Update',\n          ResponseURL: REQUIRED_BY_CFN,\n          PhysicalResourceId: REQUIRED_BY_CFN,\n          StackId: REQUIRED_BY_CFN,\n          RequestId: REQUIRED_BY_CFN,\n          LogicalResourceId: REQUIRED_BY_CFN,\n          ResourceProperties: stringifyObject(customResourceProperties), // JSON.stringify() doesn't turn the actual objects to strings, but the lambda expects strings\n        }),\n      });\n    },\n  });\n\n  return ret;\n}\n\nexport async function skipChangeForS3DeployCustomResourcePolicy(\n  iamPolicyLogicalId: string,\n  change: HotswappableChangeCandidate,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<boolean> {\n  if (change.newValue.Type !== 'AWS::IAM::Policy') {\n    return false;\n  }\n  const roles: string[] = change.newValue.Properties?.Roles;\n\n  // If no roles are referenced, the policy is definitely not used for a S3Deployment\n  if (!roles || !roles.length) {\n    return false;\n  }\n\n  // Check if every role this policy is referenced by is only used for a S3Deployment\n  for (const role of roles) {\n    const roleArn = await evaluateCfnTemplate.evaluateCfnExpression(role);\n    const roleLogicalId = await evaluateCfnTemplate.findLogicalIdForPhysicalName(roleArn);\n\n    // We must assume this role is used for something else, because we can't check it\n    if (!roleLogicalId) {\n      return false;\n    }\n\n    // Find all interesting reference to the role\n    const roleRefs = evaluateCfnTemplate\n      .findReferencesTo(roleLogicalId)\n      // we are not interested in the reference from the original policy - it always exists\n      .filter((roleRef) => !(roleRef.Type == 'AWS::IAM::Policy' && roleRef.LogicalId === iamPolicyLogicalId));\n\n    // Check if the role is only used for S3Deployment\n    // We know this is the case, if S3Deployment -> Lambda -> Role is satisfied for every reference\n    // And we have at least one reference.\n    const isRoleOnlyForS3Deployment =\n      roleRefs.length >= 1 &&\n      roleRefs.every((roleRef) => {\n        if (roleRef.Type === 'AWS::Lambda::Function') {\n          const lambdaRefs = evaluateCfnTemplate.findReferencesTo(roleRef.LogicalId);\n          // Every reference must be to the custom resource and at least one reference must be present\n          return (\n            lambdaRefs.length >= 1 && lambdaRefs.every((lambdaRef) => lambdaRef.Type === 'Custom::CDKBucketDeployment')\n          );\n        }\n        return false;\n      });\n\n    // We have determined this role is used for something else, so we can't skip the change\n    if (!isRoleOnlyForS3Deployment) {\n      return false;\n    }\n  }\n\n  // We have checked that any use of this policy is only for S3Deployment and we can safely skip it\n  return true;\n}\n\nfunction stringifyObject(obj: any): any {\n  if (obj == null) {\n    return obj;\n  }\n  if (Array.isArray(obj)) {\n    return obj.map(stringifyObject);\n  }\n  if (typeof obj !== 'object') {\n    return obj.toString();\n  }\n\n  const ret: { [k: string]: any } = {};\n  for (const [k, v] of Object.entries(obj)) {\n    ret[k] = stringifyObject(v);\n  }\n  return ret;\n}\n"]}
113
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"s3-bucket-deployments.js","sourceRoot":"","sources":["s3-bucket-deployments.ts"],"names":[],"mappings":"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","sourcesContent":["import type { ChangeHotswapResult } from './common';\nimport type { ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';\nimport type { SDK } from '../aws-auth';\nimport type { EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';\n\n/**\n * This means that the value is required to exist by CloudFormation's Custom Resource API (or our S3 Bucket Deployment Lambda's API)\n * but the actual value specified is irrelevant\n */\nconst REQUIRED_BY_CFN = 'required-to-be-present-by-cfn';\n\nconst CDK_BUCKET_DEPLOYMENT_CFN_TYPE = 'Custom::CDKBucketDeployment';\n\nexport async function isHotswappableS3BucketDeploymentChange(\n  _logicalId: string,\n  change: ResourceChange,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<ChangeHotswapResult> {\n  // In old-style synthesis, the policy used by the lambda to copy assets Ref's the assets directly,\n  // meaning that the changes made to the Policy are artifacts that can be safely ignored\n  const ret: ChangeHotswapResult = [];\n\n  if (change.newValue.Type !== CDK_BUCKET_DEPLOYMENT_CFN_TYPE) {\n    return [];\n  }\n\n  // no classification to be done here; all the properties of this custom resource thing are hotswappable\n  const customResourceProperties = await evaluateCfnTemplate.evaluateCfnExpression({\n    ...change.newValue.Properties,\n    ServiceToken: undefined,\n  });\n\n  // note that this gives the ARN of the lambda, not the name. This is fine though, the invoke() sdk call will take either\n  const functionName = await evaluateCfnTemplate.evaluateCfnExpression(change.newValue.Properties?.ServiceToken);\n  if (!functionName) {\n    return ret;\n  }\n\n  ret.push({\n    change: {\n      cause: change,\n    },\n    hotswappable: true,\n    service: 'custom-s3-deployment',\n    resourceNames: [`Contents of S3 Bucket '${customResourceProperties.DestinationBucketName}'`],\n    apply: async (sdk: SDK) => {\n      await sdk.lambda().invokeCommand({\n        FunctionName: functionName,\n        // Lambda refuses to take a direct JSON object and requires it to be stringify()'d\n        Payload: JSON.stringify({\n          RequestType: 'Update',\n          ResponseURL: REQUIRED_BY_CFN,\n          PhysicalResourceId: REQUIRED_BY_CFN,\n          StackId: REQUIRED_BY_CFN,\n          RequestId: REQUIRED_BY_CFN,\n          LogicalResourceId: REQUIRED_BY_CFN,\n          ResourceProperties: stringifyObject(customResourceProperties), // JSON.stringify() doesn't turn the actual objects to strings, but the lambda expects strings\n        }),\n      });\n    },\n  });\n\n  return ret;\n}\n\nexport async function skipChangeForS3DeployCustomResourcePolicy(\n  iamPolicyLogicalId: string,\n  change: ResourceChange,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<boolean> {\n  if (change.newValue.Type !== 'AWS::IAM::Policy') {\n    return false;\n  }\n  const roles: string[] = change.newValue.Properties?.Roles;\n\n  // If no roles are referenced, the policy is definitely not used for a S3Deployment\n  if (!roles || !roles.length) {\n    return false;\n  }\n\n  // Check if every role this policy is referenced by is only used for a S3Deployment\n  for (const role of roles) {\n    const roleArn = await evaluateCfnTemplate.evaluateCfnExpression(role);\n    const roleLogicalId = await evaluateCfnTemplate.findLogicalIdForPhysicalName(roleArn);\n\n    // We must assume this role is used for something else, because we can't check it\n    if (!roleLogicalId) {\n      return false;\n    }\n\n    // Find all interesting reference to the role\n    const roleRefs = evaluateCfnTemplate\n      .findReferencesTo(roleLogicalId)\n      // we are not interested in the reference from the original policy - it always exists\n      .filter((roleRef) => !(roleRef.Type == 'AWS::IAM::Policy' && roleRef.LogicalId === iamPolicyLogicalId));\n\n    // Check if the role is only used for S3Deployment\n    // We know this is the case, if S3Deployment -> Lambda -> Role is satisfied for every reference\n    // And we have at least one reference.\n    const isRoleOnlyForS3Deployment =\n      roleRefs.length >= 1 &&\n      roleRefs.every((roleRef) => {\n        if (roleRef.Type === 'AWS::Lambda::Function') {\n          const lambdaRefs = evaluateCfnTemplate.findReferencesTo(roleRef.LogicalId);\n          // Every reference must be to the custom resource and at least one reference must be present\n          return (\n            lambdaRefs.length >= 1 && lambdaRefs.every((lambdaRef) => lambdaRef.Type === 'Custom::CDKBucketDeployment')\n          );\n        }\n        return false;\n      });\n\n    // We have determined this role is used for something else, so we can't skip the change\n    if (!isRoleOnlyForS3Deployment) {\n      return false;\n    }\n  }\n\n  // We have checked that any use of this policy is only for S3Deployment and we can safely skip it\n  return true;\n}\n\nfunction stringifyObject(obj: any): any {\n  if (obj == null) {\n    return obj;\n  }\n  if (Array.isArray(obj)) {\n    return obj.map(stringifyObject);\n  }\n  if (typeof obj !== 'object') {\n    return obj.toString();\n  }\n\n  const ret: { [k: string]: any } = {};\n  for (const [k, v] of Object.entries(obj)) {\n    ret[k] = stringifyObject(v);\n  }\n  return ret;\n}\n"]}
@@ -1,3 +1,4 @@
1
- import { type ChangeHotswapResult, type HotswappableChangeCandidate } from './common';
1
+ import { type ChangeHotswapResult } from './common';
2
+ import type { ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';
2
3
  import type { EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';
3
- export declare function isHotswappableStateMachineChange(logicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate): Promise<ChangeHotswapResult>;
4
+ export declare function isHotswappableStateMachineChange(logicalId: string, change: ResourceChange, evaluateCfnTemplate: EvaluateCloudFormationTemplate): Promise<ChangeHotswapResult>;
@@ -19,16 +19,18 @@ async function isHotswappableStateMachineChange(logicalId, change, evaluateCfnTe
19
19
  stateMachineNameInCfnTemplate,
20
20
  })
21
21
  : await evaluateCfnTemplate.findPhysicalNameFor(logicalId);
22
+ // nothing to do
23
+ if (!stateMachineArn) {
24
+ return ret;
25
+ }
22
26
  ret.push({
27
+ change: {
28
+ cause: change,
29
+ },
23
30
  hotswappable: true,
24
- resourceType: change.newValue.Type,
25
- propsChanged: namesOfHotswappableChanges,
26
31
  service: 'stepfunctions-service',
27
32
  resourceNames: [`${change.newValue.Type} '${stateMachineArn === null || stateMachineArn === void 0 ? void 0 : stateMachineArn.split(':')[6]}'`],
28
33
  apply: async (sdk) => {
29
- if (!stateMachineArn) {
30
- return;
31
- }
32
34
  // not passing the optional properties leaves them unchanged
33
35
  await sdk.stepFunctions().updateStateMachine({
34
36
  stateMachineArn,
@@ -39,4 +41,4 @@ async function isHotswappableStateMachineChange(logicalId, change, evaluateCfnTe
39
41
  }
40
42
  return ret;
41
43
  }
42
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RlcGZ1bmN0aW9ucy1zdGF0ZS1tYWNoaW5lcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInN0ZXBmdW5jdGlvbnMtc3RhdGUtbWFjaGluZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFJQSw0RUEyQ0M7QUEvQ0QscUNBQXVHO0FBSWhHLEtBQUssVUFBVSxnQ0FBZ0MsQ0FDcEQsU0FBaUIsRUFDakIsTUFBbUMsRUFDbkMsbUJBQW1EOztJQUVuRCxJQUFJLE1BQU0sQ0FBQyxRQUFRLENBQUMsSUFBSSxLQUFLLGtDQUFrQyxFQUFFLENBQUM7UUFDaEUsT0FBTyxFQUFFLENBQUM7SUFDWixDQUFDO0lBQ0QsTUFBTSxHQUFHLEdBQXdCLEVBQUUsQ0FBQztJQUNwQyxNQUFNLGlCQUFpQixHQUFHLElBQUEsd0JBQWUsRUFBQyxNQUFNLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDLENBQUM7SUFDeEUsaUJBQWlCLENBQUMsb0NBQW9DLENBQUMsR0FBRyxDQUFDLENBQUM7SUFFNUQsTUFBTSwwQkFBMEIsR0FBRyxNQUFNLENBQUMsSUFBSSxDQUFDLGlCQUFpQixDQUFDLGlCQUFpQixDQUFDLENBQUM7SUFDcEYsSUFBSSwwQkFBMEIsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxFQUFFLENBQUM7UUFDMUMsTUFBTSw2QkFBNkIsR0FBRyxNQUFBLE1BQUEsTUFBTSxDQUFDLFFBQVEsMENBQUUsVUFBVSwwQ0FBRSxnQkFBZ0IsQ0FBQztRQUNwRixNQUFNLGVBQWUsR0FBRyw2QkFBNkI7WUFDbkQsQ0FBQyxDQUFDLE1BQU0sbUJBQW1CLENBQUMscUJBQXFCLENBQUM7Z0JBQ2hELFNBQVMsRUFDTCw2RUFBNkU7b0JBQzdFLDZCQUE2QjthQUNsQyxDQUFDO1lBQ0YsQ0FBQyxDQUFDLE1BQU0sbUJBQW1CLENBQUMsbUJBQW1CLENBQUMsU0FBUyxDQUFDLENBQUM7UUFDN0QsR0FBRyxDQUFDLElBQUksQ0FBQztZQUNQLFlBQVksRUFBRSxJQUFJO1lBQ2xCLFlBQVksRUFBRSxNQUFNLENBQUMsUUFBUSxDQUFDLElBQUk7WUFDbEMsWUFBWSxFQUFFLDBCQUEwQjtZQUN4QyxPQUFPLEVBQUUsdUJBQXVCO1lBQ2hDLGFBQWEsRUFBRSxDQUFDLEdBQUcsTUFBTSxDQUFDLFFBQVEsQ0FBQyxJQUFJLEtBQUssZUFBZSxhQUFmLGVBQWUsdUJBQWYsZUFBZSxDQUFFLEtBQUssQ0FBQyxHQUFHLEVBQUUsQ0FBQyxDQUFDLEdBQUcsQ0FBQztZQUM5RSxLQUFLLEVBQUUsS0FBSyxFQUFFLEdBQVEsRUFBRSxFQUFFO2dCQUN4QixJQUFJLENBQUMsZUFBZSxFQUFFLENBQUM7b0JBQ3JCLE9BQU87Z0JBQ1QsQ0FBQztnQkFFRCw0REFBNEQ7Z0JBQzVELE1BQU0sR0FBRyxDQUFDLGFBQWEsRUFBRSxDQUFDLGtCQUFrQixDQUFDO29CQUMzQyxlQUFlO29CQUNmLFVBQVUsRUFBRSxNQUFNLG1CQUFtQixDQUFDLHFCQUFxQixDQUFDLE1BQU0sQ0FBQyxlQUFlLENBQUMsZ0JBQWdCLENBQUMsUUFBUSxDQUFDO2lCQUM5RyxDQUFDLENBQUM7WUFDTCxDQUFDO1NBQ0YsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUVELE9BQU8sR0FBRyxDQUFDO0FBQ2IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHR5cGUgQ2hhbmdlSG90c3dhcFJlc3VsdCwgY2xhc3NpZnlDaGFuZ2VzLCB0eXBlIEhvdHN3YXBwYWJsZUNoYW5nZUNhbmRpZGF0ZSB9IGZyb20gJy4vY29tbW9uJztcbmltcG9ydCB0eXBlIHsgU0RLIH0gZnJvbSAnLi4vYXdzLWF1dGgnO1xuaW1wb3J0IHR5cGUgeyBFdmFsdWF0ZUNsb3VkRm9ybWF0aW9uVGVtcGxhdGUgfSBmcm9tICcuLi9ldmFsdWF0ZS1jbG91ZGZvcm1hdGlvbi10ZW1wbGF0ZSc7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBpc0hvdHN3YXBwYWJsZVN0YXRlTWFjaGluZUNoYW5nZShcbiAgbG9naWNhbElkOiBzdHJpbmcsXG4gIGNoYW5nZTogSG90c3dhcHBhYmxlQ2hhbmdlQ2FuZGlkYXRlLFxuICBldmFsdWF0ZUNmblRlbXBsYXRlOiBFdmFsdWF0ZUNsb3VkRm9ybWF0aW9uVGVtcGxhdGUsXG4pOiBQcm9taXNlPENoYW5nZUhvdHN3YXBSZXN1bHQ+IHtcbiAgaWYgKGNoYW5nZS5uZXdWYWx1ZS5UeXBlICE9PSAnQVdTOjpTdGVwRnVuY3Rpb25zOjpTdGF0ZU1hY2hpbmUnKSB7XG4gICAgcmV0dXJuIFtdO1xuICB9XG4gIGNvbnN0IHJldDogQ2hhbmdlSG90c3dhcFJlc3VsdCA9IFtdO1xuICBjb25zdCBjbGFzc2lmaWVkQ2hhbmdlcyA9IGNsYXNzaWZ5Q2hhbmdlcyhjaGFuZ2UsIFsnRGVmaW5pdGlvblN0cmluZyddKTtcbiAgY2xhc3NpZmllZENoYW5nZXMucmVwb3J0Tm9uSG90c3dhcHBhYmxlUHJvcGVydHlDaGFuZ2VzKHJldCk7XG5cbiAgY29uc3QgbmFtZXNPZkhvdHN3YXBwYWJsZUNoYW5nZXMgPSBPYmplY3Qua2V5cyhjbGFzc2lmaWVkQ2hhbmdlcy5ob3Rzd2FwcGFibGVQcm9wcyk7XG4gIGlmIChuYW1lc09mSG90c3dhcHBhYmxlQ2hhbmdlcy5sZW5ndGggPiAwKSB7XG4gICAgY29uc3Qgc3RhdGVNYWNoaW5lTmFtZUluQ2ZuVGVtcGxhdGUgPSBjaGFuZ2UubmV3VmFsdWU/LlByb3BlcnRpZXM/LlN0YXRlTWFjaGluZU5hbWU7XG4gICAgY29uc3Qgc3RhdGVNYWNoaW5lQXJuID0gc3RhdGVNYWNoaW5lTmFtZUluQ2ZuVGVtcGxhdGVcbiAgICAgID8gYXdhaXQgZXZhbHVhdGVDZm5UZW1wbGF0ZS5ldmFsdWF0ZUNmbkV4cHJlc3Npb24oe1xuICAgICAgICAnRm46OlN1Yic6XG4gICAgICAgICAgICAnYXJuOiR7QVdTOjpQYXJ0aXRpb259OnN0YXRlczoke0FXUzo6UmVnaW9ufToke0FXUzo6QWNjb3VudElkfTpzdGF0ZU1hY2hpbmU6JyArXG4gICAgICAgICAgICBzdGF0ZU1hY2hpbmVOYW1lSW5DZm5UZW1wbGF0ZSxcbiAgICAgIH0pXG4gICAgICA6IGF3YWl0IGV2YWx1YXRlQ2ZuVGVtcGxhdGUuZmluZFBoeXNpY2FsTmFtZUZvcihsb2dpY2FsSWQpO1xuICAgIHJldC5wdXNoKHtcbiAgICAgIGhvdHN3YXBwYWJsZTogdHJ1ZSxcbiAgICAgIHJlc291cmNlVHlwZTogY2hhbmdlLm5ld1ZhbHVlLlR5cGUsXG4gICAgICBwcm9wc0NoYW5nZWQ6IG5hbWVzT2ZIb3Rzd2FwcGFibGVDaGFuZ2VzLFxuICAgICAgc2VydmljZTogJ3N0ZXBmdW5jdGlvbnMtc2VydmljZScsXG4gICAgICByZXNvdXJjZU5hbWVzOiBbYCR7Y2hhbmdlLm5ld1ZhbHVlLlR5cGV9ICcke3N0YXRlTWFjaGluZUFybj8uc3BsaXQoJzonKVs2XX0nYF0sXG4gICAgICBhcHBseTogYXN5bmMgKHNkazogU0RLKSA9PiB7XG4gICAgICAgIGlmICghc3RhdGVNYWNoaW5lQXJuKSB7XG4gICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gbm90IHBhc3NpbmcgdGhlIG9wdGlvbmFsIHByb3BlcnRpZXMgbGVhdmVzIHRoZW0gdW5jaGFuZ2VkXG4gICAgICAgIGF3YWl0IHNkay5zdGVwRnVuY3Rpb25zKCkudXBkYXRlU3RhdGVNYWNoaW5lKHtcbiAgICAgICAgICBzdGF0ZU1hY2hpbmVBcm4sXG4gICAgICAgICAgZGVmaW5pdGlvbjogYXdhaXQgZXZhbHVhdGVDZm5UZW1wbGF0ZS5ldmFsdWF0ZUNmbkV4cHJlc3Npb24oY2hhbmdlLnByb3BlcnR5VXBkYXRlcy5EZWZpbml0aW9uU3RyaW5nLm5ld1ZhbHVlKSxcbiAgICAgICAgfSk7XG4gICAgICB9LFxuICAgIH0pO1xuICB9XG5cbiAgcmV0dXJuIHJldDtcbn1cbiJdfQ==
44
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RlcGZ1bmN0aW9ucy1zdGF0ZS1tYWNoaW5lcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInN0ZXBmdW5jdGlvbnMtc3RhdGUtbWFjaGluZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFLQSw0RUE4Q0M7QUFuREQscUNBQXFFO0FBSzlELEtBQUssVUFBVSxnQ0FBZ0MsQ0FDcEQsU0FBaUIsRUFDakIsTUFBc0IsRUFDdEIsbUJBQW1EOztJQUVuRCxJQUFJLE1BQU0sQ0FBQyxRQUFRLENBQUMsSUFBSSxLQUFLLGtDQUFrQyxFQUFFLENBQUM7UUFDaEUsT0FBTyxFQUFFLENBQUM7SUFDWixDQUFDO0lBQ0QsTUFBTSxHQUFHLEdBQXdCLEVBQUUsQ0FBQztJQUNwQyxNQUFNLGlCQUFpQixHQUFHLElBQUEsd0JBQWUsRUFBQyxNQUFNLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDLENBQUM7SUFDeEUsaUJBQWlCLENBQUMsb0NBQW9DLENBQUMsR0FBRyxDQUFDLENBQUM7SUFFNUQsTUFBTSwwQkFBMEIsR0FBRyxNQUFNLENBQUMsSUFBSSxDQUFDLGlCQUFpQixDQUFDLGlCQUFpQixDQUFDLENBQUM7SUFDcEYsSUFBSSwwQkFBMEIsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxFQUFFLENBQUM7UUFDMUMsTUFBTSw2QkFBNkIsR0FBRyxNQUFBLE1BQUEsTUFBTSxDQUFDLFFBQVEsMENBQUUsVUFBVSwwQ0FBRSxnQkFBZ0IsQ0FBQztRQUNwRixNQUFNLGVBQWUsR0FBRyw2QkFBNkI7WUFDbkQsQ0FBQyxDQUFDLE1BQU0sbUJBQW1CLENBQUMscUJBQXFCLENBQUM7Z0JBQ2hELFNBQVMsRUFDTCw2RUFBNkU7b0JBQzdFLDZCQUE2QjthQUNsQyxDQUFDO1lBQ0YsQ0FBQyxDQUFDLE1BQU0sbUJBQW1CLENBQUMsbUJBQW1CLENBQUMsU0FBUyxDQUFDLENBQUM7UUFFN0QsZ0JBQWdCO1FBQ2hCLElBQUksQ0FBQyxlQUFlLEVBQUUsQ0FBQztZQUNyQixPQUFPLEdBQUcsQ0FBQztRQUNiLENBQUM7UUFFRCxHQUFHLENBQUMsSUFBSSxDQUFDO1lBQ1AsTUFBTSxFQUFFO2dCQUNOLEtBQUssRUFBRSxNQUFNO2FBQ2Q7WUFDRCxZQUFZLEVBQUUsSUFBSTtZQUNsQixPQUFPLEVBQUUsdUJBQXVCO1lBQ2hDLGFBQWEsRUFBRSxDQUFDLEdBQUcsTUFBTSxDQUFDLFFBQVEsQ0FBQyxJQUFJLEtBQUssZUFBZSxhQUFmLGVBQWUsdUJBQWYsZUFBZSxDQUFFLEtBQUssQ0FBQyxHQUFHLEVBQUUsQ0FBQyxDQUFDLEdBQUcsQ0FBQztZQUM5RSxLQUFLLEVBQUUsS0FBSyxFQUFFLEdBQVEsRUFBRSxFQUFFO2dCQUN4Qiw0REFBNEQ7Z0JBQzVELE1BQU0sR0FBRyxDQUFDLGFBQWEsRUFBRSxDQUFDLGtCQUFrQixDQUFDO29CQUMzQyxlQUFlO29CQUNmLFVBQVUsRUFBRSxNQUFNLG1CQUFtQixDQUFDLHFCQUFxQixDQUFDLE1BQU0sQ0FBQyxlQUFlLENBQUMsZ0JBQWdCLENBQUMsUUFBUSxDQUFDO2lCQUM5RyxDQUFDLENBQUM7WUFDTCxDQUFDO1NBQ0YsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUVELE9BQU8sR0FBRyxDQUFDO0FBQ2IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHR5cGUgQ2hhbmdlSG90c3dhcFJlc3VsdCwgY2xhc3NpZnlDaGFuZ2VzIH0gZnJvbSAnLi9jb21tb24nO1xuaW1wb3J0IHR5cGUgeyBSZXNvdXJjZUNoYW5nZSB9IGZyb20gJy4uLy4uLy4uLy4uL0Bhd3MtY2RrL3RtcC10b29sa2l0LWhlbHBlcnMvc3JjL2FwaS9pby9wYXlsb2Fkcy9ob3Rzd2FwJztcbmltcG9ydCB0eXBlIHsgU0RLIH0gZnJvbSAnLi4vYXdzLWF1dGgnO1xuaW1wb3J0IHR5cGUgeyBFdmFsdWF0ZUNsb3VkRm9ybWF0aW9uVGVtcGxhdGUgfSBmcm9tICcuLi9ldmFsdWF0ZS1jbG91ZGZvcm1hdGlvbi10ZW1wbGF0ZSc7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBpc0hvdHN3YXBwYWJsZVN0YXRlTWFjaGluZUNoYW5nZShcbiAgbG9naWNhbElkOiBzdHJpbmcsXG4gIGNoYW5nZTogUmVzb3VyY2VDaGFuZ2UsXG4gIGV2YWx1YXRlQ2ZuVGVtcGxhdGU6IEV2YWx1YXRlQ2xvdWRGb3JtYXRpb25UZW1wbGF0ZSxcbik6IFByb21pc2U8Q2hhbmdlSG90c3dhcFJlc3VsdD4ge1xuICBpZiAoY2hhbmdlLm5ld1ZhbHVlLlR5cGUgIT09ICdBV1M6OlN0ZXBGdW5jdGlvbnM6OlN0YXRlTWFjaGluZScpIHtcbiAgICByZXR1cm4gW107XG4gIH1cbiAgY29uc3QgcmV0OiBDaGFuZ2VIb3Rzd2FwUmVzdWx0ID0gW107XG4gIGNvbnN0IGNsYXNzaWZpZWRDaGFuZ2VzID0gY2xhc3NpZnlDaGFuZ2VzKGNoYW5nZSwgWydEZWZpbml0aW9uU3RyaW5nJ10pO1xuICBjbGFzc2lmaWVkQ2hhbmdlcy5yZXBvcnROb25Ib3Rzd2FwcGFibGVQcm9wZXJ0eUNoYW5nZXMocmV0KTtcblxuICBjb25zdCBuYW1lc09mSG90c3dhcHBhYmxlQ2hhbmdlcyA9IE9iamVjdC5rZXlzKGNsYXNzaWZpZWRDaGFuZ2VzLmhvdHN3YXBwYWJsZVByb3BzKTtcbiAgaWYgKG5hbWVzT2ZIb3Rzd2FwcGFibGVDaGFuZ2VzLmxlbmd0aCA+IDApIHtcbiAgICBjb25zdCBzdGF0ZU1hY2hpbmVOYW1lSW5DZm5UZW1wbGF0ZSA9IGNoYW5nZS5uZXdWYWx1ZT8uUHJvcGVydGllcz8uU3RhdGVNYWNoaW5lTmFtZTtcbiAgICBjb25zdCBzdGF0ZU1hY2hpbmVBcm4gPSBzdGF0ZU1hY2hpbmVOYW1lSW5DZm5UZW1wbGF0ZVxuICAgICAgPyBhd2FpdCBldmFsdWF0ZUNmblRlbXBsYXRlLmV2YWx1YXRlQ2ZuRXhwcmVzc2lvbih7XG4gICAgICAgICdGbjo6U3ViJzpcbiAgICAgICAgICAgICdhcm46JHtBV1M6OlBhcnRpdGlvbn06c3RhdGVzOiR7QVdTOjpSZWdpb259OiR7QVdTOjpBY2NvdW50SWR9OnN0YXRlTWFjaGluZTonICtcbiAgICAgICAgICAgIHN0YXRlTWFjaGluZU5hbWVJbkNmblRlbXBsYXRlLFxuICAgICAgfSlcbiAgICAgIDogYXdhaXQgZXZhbHVhdGVDZm5UZW1wbGF0ZS5maW5kUGh5c2ljYWxOYW1lRm9yKGxvZ2ljYWxJZCk7XG5cbiAgICAvLyBub3RoaW5nIHRvIGRvXG4gICAgaWYgKCFzdGF0ZU1hY2hpbmVBcm4pIHtcbiAgICAgIHJldHVybiByZXQ7XG4gICAgfVxuXG4gICAgcmV0LnB1c2goe1xuICAgICAgY2hhbmdlOiB7XG4gICAgICAgIGNhdXNlOiBjaGFuZ2UsXG4gICAgICB9LFxuICAgICAgaG90c3dhcHBhYmxlOiB0cnVlLFxuICAgICAgc2VydmljZTogJ3N0ZXBmdW5jdGlvbnMtc2VydmljZScsXG4gICAgICByZXNvdXJjZU5hbWVzOiBbYCR7Y2hhbmdlLm5ld1ZhbHVlLlR5cGV9ICcke3N0YXRlTWFjaGluZUFybj8uc3BsaXQoJzonKVs2XX0nYF0sXG4gICAgICBhcHBseTogYXN5bmMgKHNkazogU0RLKSA9PiB7XG4gICAgICAgIC8vIG5vdCBwYXNzaW5nIHRoZSBvcHRpb25hbCBwcm9wZXJ0aWVzIGxlYXZlcyB0aGVtIHVuY2hhbmdlZFxuICAgICAgICBhd2FpdCBzZGsuc3RlcEZ1bmN0aW9ucygpLnVwZGF0ZVN0YXRlTWFjaGluZSh7XG4gICAgICAgICAgc3RhdGVNYWNoaW5lQXJuLFxuICAgICAgICAgIGRlZmluaXRpb246IGF3YWl0IGV2YWx1YXRlQ2ZuVGVtcGxhdGUuZXZhbHVhdGVDZm5FeHByZXNzaW9uKGNoYW5nZS5wcm9wZXJ0eVVwZGF0ZXMuRGVmaW5pdGlvblN0cmluZy5uZXdWYWx1ZSksXG4gICAgICAgIH0pO1xuICAgICAgfSxcbiAgICB9KTtcbiAgfVxuXG4gIHJldHVybiByZXQ7XG59XG4iXX0=
@@ -1,5 +1,5 @@
1
1
  import type { CloudFormationStackArtifact, Environment } from '@aws-cdk/cx-api';
2
- import { IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';
2
+ import type { 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,8 +22,7 @@ async function findCloudWatchLogGroups(sdkProvider, ioHelper, stackArtifact) {
22
22
  }
23
23
  const listStackResources = new evaluate_cloudformation_template_1.LazyListStackResources(sdk, stackArtifact.stackName);
24
24
  const evaluateCfnTemplate = new evaluate_cloudformation_template_1.EvaluateCloudFormationTemplate({
25
- stackName: stackArtifact.stackName,
26
- template: stackArtifact.template,
25
+ stackArtifact,
27
26
  parameters: {},
28
27
  account: resolvedEnv.account,
29
28
  region: resolvedEnv.region,
@@ -94,4 +93,4 @@ function findAllLogGroupNames(stackResources, evaluateCfnTemplate) {
94
93
  }
95
94
  return logGroupNames;
96
95
  }
97
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"find-cloudwatch-logs.js","sourceRoot":"","sources":["find-cloudwatch-logs.ts"],"names":[],"mappings":";;AAqCA,0DAkCC;AArED,yFAA2F;AAC3F,qCAAgD;AAEhD,gDAAmD;AACnD,0FAA6G;AAC7G,yCAAsC;AACtC,kDAA6D;AAE7D,2FAA2F;AAC3F,MAAM,0BAA0B,GAAG,CAAC,mBAAmB,EAAE,wBAAwB,EAAE,yBAAyB,CAAC,CAAC;AA0BvG,KAAK,UAAU,uBAAuB,CAC3C,WAAwB,EACxB,QAAkB,EAClB,aAA0C;IAE1C,IAAI,GAAQ,CAAC;IACb,MAAM,WAAW,GAAG,MAAM,WAAW,CAAC,kBAAkB,CAAC,aAAa,CAAC,WAAW,CAAC,CAAC;IACpF,wEAAwE;IACxE,IAAI,CAAC;QACH,GAAG,GAAG,CAAC,MAAM,IAAI,+BAAiB,CAAC,WAAW,EAAE,yCAA0B,EAAE,QAAQ,CAAC,CAAC,oBAAoB,CAAC,aAAa,CAAC,CAAC,CAAC,GAAG,CAAC;IACjI,CAAC;IAAC,OAAO,CAAM,EAAE,CAAC;QAChB,MAAM,QAAQ,CAAC,MAAM,CAAC,YAAE,CAAC,qBAAqB,CAAC,GAAG,CAAC,qCAAqC,IAAA,yBAAkB,EAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QAClH,GAAG,GAAG,CAAC,MAAM,WAAW,CAAC,cAAc,CAAC,WAAW,EAAE,WAAI,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC;IAC7E,CAAC;IAED,MAAM,kBAAkB,GAAG,IAAI,yDAAsB,CAAC,GAAG,EAAE,aAAa,CAAC,SAAS,CAAC,CAAC;IACpF,MAAM,mBAAmB,GAAG,IAAI,iEAA8B,CAAC;QAC7D,SAAS,EAAE,aAAa,CAAC,SAAS;QAClC,QAAQ,EAAE,aAAa,CAAC,QAAQ;QAChC,UAAU,EAAE,EAAE;QACd,OAAO,EAAE,WAAW,CAAC,OAAO;QAC5B,MAAM,EAAE,WAAW,CAAC,MAAM;QAC1B,SAAS,EAAE,CAAC,MAAM,GAAG,CAAC,cAAc,EAAE,CAAC,CAAC,SAAS;QACjD,GAAG;KACJ,CAAC,CAAC;IAEH,MAAM,cAAc,GAAG,MAAM,kBAAkB,CAAC,kBAAkB,EAAE,CAAC;IACrE,MAAM,aAAa,GAAG,oBAAoB,CAAC,cAAc,EAAE,mBAAmB,CAAC,CAAC;IAEhF,OAAO;QACL,GAAG,EAAE,WAAW;QAChB,GAAG;QACH,aAAa;KACd,CAAC;AACJ,CAAC;AAED;;;GAGG;AACH,SAAS,+BAA+B,CACtC,gBAAsC,EACtC,mBAAmD;IAEnD,MAAM,4BAA4B,GAAG,mBAAmB,CAAC,gBAAgB,CAAC,gBAAgB,CAAC,iBAAkB,CAAC,CAAC;IAC/G,OAAO,4BAA4B,CAAC,IAAI,CAAC,CAAC,SAAS,EAAE,EAAE;QACrD,OAAO,0BAA0B,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;IAC7D,CAAC,CAAC,CAAC;AACL,CAAC;AAOD,MAAM,uBAAuB,GAA2C;IACtE,qBAAqB,EAAE,CAAC,QAAQ,EAAE,mBAAmB,EAAE,EAAE;;QACvD,IAAI,+BAA+B,CAAC,QAAQ,EAAE,mBAAmB,CAAC,EAAE,CAAC;YACnE,OAAO,SAAS,CAAC;QACnB,CAAC;QACD,OAAO,MAAA,QAAQ,CAAC,kBAAkB,0CAAE,QAAQ,EAAE,CAAC;IACjD,CAAC;IAED,sGAAsG;IACtG,8GAA8G;IAC9G,uFAAuF;IACvF,uBAAuB,EAAE,CAAC,QAAQ,EAAE,mBAAmB,EAAE,EAAE;QACzD,MAAM,aAAa,GAAG,mBAAmB,CAAC,mBAAmB,CAAC,QAAQ,CAAC,iBAAkB,EAAE,eAAe,CAAC,CAAC;QAC5G,IAAI,aAAa,aAAb,aAAa,uBAAb,aAAa,CAAE,QAAQ,EAAE,CAAC;YAC5B,+GAA+G;YAC/G,IAAI,OAAO,aAAa,CAAC,QAAQ,KAAK,QAAQ,EAAE,CAAC;gBAC/C,OAAO,aAAa,CAAC,QAAQ,CAAC;YAChC,CAAC;YAED,8GAA8G;YAC9G,IAAI,OAAO,aAAa,CAAC,QAAQ,KAAK,QAAQ,EAAE,CAAC;gBAC/C,IAAI,aAAa,CAAC,QAAQ,CAAC,GAAG,EAAE,CAAC;oBAC/B,OAAO,mBAAmB,CAAC,mBAAmB,CAAC,aAAa,CAAC,QAAQ,CAAC,GAAG,EAAE,cAAc,CAAC,CAAC;gBAC7F,CAAC;YACH,CAAC;QACH,CAAC;QAED,OAAO,eAAe,QAAQ,CAAC,kBAAkB,EAAE,CAAC;IACtD,CAAC;CACF,CAAC;AAEF;;;;GAIG;AACH,SAAS,oBAAoB,CAC3B,cAAsC,EACtC,mBAAmD;IAEnD,MAAM,aAAa,GAAa,EAAE,CAAC;IAEnC,KAAK,MAAM,QAAQ,IAAI,cAAc,EAAE,CAAC;QACtC,MAAM,gBAAgB,GAAG,uBAAuB,CAAC,QAAQ,CAAC,YAAa,CAAC,CAAC;QACzE,IAAI,gBAAgB,EAAE,CAAC;YACrB,MAAM,YAAY,GAAG,gBAAgB,CAAC,QAAQ,EAAE,mBAAmB,CAAC,CAAC;YACrE,IAAI,YAAY,EAAE,CAAC;gBACjB,aAAa,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YACnC,CAAC;QACH,CAAC;IACH,CAAC;IAED,OAAO,aAAa,CAAC;AACvB,CAAC","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"]}
96
+ //# 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 type { IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport { IO } 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    stackArtifact,\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,5 @@
1
- import * as cxapi from '@aws-cdk/cx-api';
2
- import { IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';
1
+ import type * as cxapi from '@aws-cdk/cx-api';
2
+ import type { IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';
3
3
  import type { SDK } from '../aws-auth';
4
4
  export interface CloudWatchLogEventMonitorProps {
5
5
  /**