@jjrawlins/cdk-diff-pr-github-action 1.3.12 → 1.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +18 -18
- package/lib/CdkDiffIamTemplate.js +6 -3
- package/lib/CdkDiffIamTemplateStackSet.js +4 -3
- package/lib/CdkDiffStackWorkflow.js +1 -1
- package/lib/CdkDriftDetectionWorkflow.js +1 -1
- package/lib/CdkDriftIamTemplate.js +2 -2
- package/lib/bin/cdk-changeset-script.js +153 -20
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/package.json +1 -1
- package/node_modules/@aws-sdk/client-cloudformation/package.json +2 -2
- package/node_modules/@smithy/core/dist-cjs/submodules/cbor/index.js +13 -10
- package/node_modules/@smithy/core/dist-cjs/submodules/protocols/index.js +5 -0
- package/node_modules/@smithy/core/dist-cjs/submodules/schema/index.js +24 -5
- package/node_modules/@smithy/core/dist-es/submodules/cbor/SmithyRpcV2CborProtocol.js +15 -11
- package/node_modules/@smithy/core/dist-es/submodules/protocols/HttpProtocol.js +6 -1
- package/node_modules/@smithy/core/dist-es/submodules/schema/TypeRegistry.js +24 -5
- package/node_modules/@smithy/core/dist-types/submodules/cbor/SmithyRpcV2CborProtocol.d.ts +7 -1
- package/node_modules/@smithy/core/dist-types/submodules/protocols/HttpBindingProtocol.d.ts +5 -1
- package/node_modules/@smithy/core/dist-types/submodules/protocols/HttpProtocol.d.ts +14 -1
- package/node_modules/@smithy/core/dist-types/submodules/protocols/RpcProtocol.d.ts +5 -0
- package/node_modules/@smithy/core/dist-types/submodules/schema/TypeRegistry.d.ts +8 -1
- package/node_modules/@smithy/core/dist-types/ts3.4/submodules/cbor/SmithyRpcV2CborProtocol.d.ts +7 -1
- package/node_modules/@smithy/core/dist-types/ts3.4/submodules/protocols/HttpBindingProtocol.d.ts +5 -1
- package/node_modules/@smithy/core/dist-types/ts3.4/submodules/protocols/HttpProtocol.d.ts +14 -1
- package/node_modules/@smithy/core/dist-types/ts3.4/submodules/protocols/RpcProtocol.d.ts +5 -0
- package/node_modules/@smithy/core/dist-types/ts3.4/submodules/schema/TypeRegistry.d.ts +8 -1
- package/node_modules/@smithy/core/package.json +2 -2
- package/node_modules/@smithy/middleware-endpoint/package.json +2 -2
- package/node_modules/@smithy/middleware-retry/package.json +2 -2
- package/node_modules/@smithy/node-http-handler/dist-cjs/index.js +9 -2
- package/node_modules/@smithy/node-http-handler/dist-es/write-request-body.js +9 -2
- package/node_modules/@smithy/node-http-handler/package.json +1 -1
- package/node_modules/@smithy/smithy-client/package.json +4 -4
- package/node_modules/@smithy/util-defaults-mode-browser/package.json +2 -2
- package/node_modules/@smithy/util-defaults-mode-node/package.json +2 -2
- package/node_modules/@smithy/util-stream/package.json +2 -2
- package/package.json +3 -3
|
@@ -79,7 +79,7 @@ class CdkDriftIamTemplateGenerator {
|
|
|
79
79
|
}
|
|
80
80
|
exports.CdkDriftIamTemplateGenerator = CdkDriftIamTemplateGenerator;
|
|
81
81
|
_a = JSII_RTTI_SYMBOL_1;
|
|
82
|
-
CdkDriftIamTemplateGenerator[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplateGenerator", version: "1.
|
|
82
|
+
CdkDriftIamTemplateGenerator[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplateGenerator", version: "1.4.0" };
|
|
83
83
|
/**
|
|
84
84
|
* Projen construct that emits a CloudFormation template with minimal IAM permissions
|
|
85
85
|
* for the CDK Drift Detection Workflow.
|
|
@@ -102,5 +102,5 @@ class CdkDriftIamTemplate {
|
|
|
102
102
|
}
|
|
103
103
|
exports.CdkDriftIamTemplate = CdkDriftIamTemplate;
|
|
104
104
|
_b = JSII_RTTI_SYMBOL_1;
|
|
105
|
-
CdkDriftIamTemplate[_b] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplate", version: "1.
|
|
105
|
+
CdkDriftIamTemplate[_b] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplate", version: "1.4.0" };
|
|
106
106
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDriftIamTemplate.js","sourceRoot":"","sources":["../src/CdkDriftIamTemplate.ts"],"names":[],"mappings":";;;;;AAAA,mCAAkC;AAclC;;;GAGG;AACH,MAAa,4BAA4B;IACvC;;OAEG;IACH,MAAM,CAAC,gBAAgB,CAAC,KAAwC;QAC9D,MAAM,KAAK,GAAG;YACZ,wCAAwC;YACxC,0DAA0D;YAC1D,EAAE;YACF,aAAa;YACb,sBAAsB;YACtB,kBAAkB;YAClB,yFAAyF;YACzF,iBAAiB,KAAK,CAAC,WAAW,GAAG;YACrC,EAAE;YACF,YAAY;YACZ,8FAA8F;YAC9F,iBAAiB;YACjB,0BAA0B;YAC1B,iBAAiB;YACjB,mBAAmB,GAAG,KAAK,CAAC,QAAQ,GAAG,GAAG;YAC1C,iCAAiC;YACjC,+BAA+B;YAC/B,oBAAoB;YACpB,2BAA2B;YAC3B,wBAAwB;YACxB,2CAA2C;YAC3C,oCAAoC;YACpC,wBAAwB;YACxB,6BAA6B;YAC7B,wCAAwC,GAAG,KAAK,CAAC,UAAU,GAAG,GAAG;YACjE,iBAAiB;YACjB,iDAAiD;YACjD,2BAA2B;YAC3B,mCAAmC;YACnC,wBAAwB;YACxB,2DAA2D;YAC3D,+BAA+B;YAC/B,yBAAyB;YACzB,qDAAqD;YACrD,sEAAsE;YACtE,gEAAgE;YAChE,mDAAmD;YACnD,uDAAuD;YACvD,6DAA6D;YAC7D,+BAA+B;YAC/B,EAAE;YACF,UAAU;YACV,oBAAoB;YACpB,wDAAwD;YACxD,qCAAqC;YACrC,aAAa;YACb,sDAAsD;YACtD,EAAE;YACF,qBAAqB;YACrB,yDAAyD;YACzD,8BAA8B;YAC9B,aAAa;YACb,uDAAuD;SACxD,CAAC;QAEF,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,qBAAqB,CAAC,eAAuB,sCAAsC;QACxF,OAAO,6CAA6C,YAAY,+EAA+E,CAAC;IAClJ,CAAC;;AArEH,oEAsEC;;;AAYD;;;;;GAKG;AACH,MAAa,mBAAmB;IAC9B,YAAY,KAA+B;QACzC,MAAM,UAAU,GAAG,KAAK,CAAC,UAAU,IAAI,sCAAsC,CAAC;QAE9E,wCAAwC;QACxC,MAAM,QAAQ,GAAG,4BAA4B,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;QACtE,IAAI,iBAAQ,CAAC,KAAK,CAAC,OAAO,EAAE,UAAU,EAAE,EAAE,KAAK,EAAE,QAAQ,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEzE,kBAAkB;QAClB,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,8BAA8B,EAAE;YACpD,WAAW,EACT,uIAAuI;YACzI,WAAW,EAAE,IAAI;YACjB,IAAI,EAAE,4BAA4B,CAAC,qBAAqB,CAAC,UAAU,CAAC;SACrE,CAAC,CAAC;IACL,CAAC;;AAfH,kDAgBC","sourcesContent":["import { TextFile } from 'projen';\n\n/**\n * Props for generating CDK Drift IAM templates (no Projen dependency)\n */\nexport interface CdkDriftIamTemplateGeneratorProps {\n  /** Name for the IAM role */\n  readonly roleName: string;\n  /** ARN of the existing GitHub OIDC role that can assume this drift role */\n  readonly oidcRoleArn: string;\n  /** Region for the OIDC trust condition */\n  readonly oidcRegion: string;\n}\n\n/**\n * Pure generator class for CDK Drift IAM templates.\n * No Projen dependency - can be used in any project.\n */\nexport class CdkDriftIamTemplateGenerator {\n  /**\n   * Generate the CloudFormation IAM template as a YAML string.\n   */\n  static generateTemplate(props: CdkDriftIamTemplateGeneratorProps): string {\n    const lines = [\n      \"AWSTemplateFormatVersion: '2010-09-09'\",\n      \"Description: 'IAM role for CDK Drift Detection Workflow'\",\n      '',\n      'Parameters:',\n      '  GitHubOIDCRoleArn:',\n      '    Type: String',\n      \"    Description: 'ARN of the existing GitHub OIDC role that can assume this drift role'\",\n      `    Default: '${props.oidcRoleArn}'`,\n      '',\n      'Resources:',\n      '  # CloudFormation Drift Detection Role - minimal permissions for drift detection operations',\n      '  CdkDriftRole:',\n      '    Type: AWS::IAM::Role',\n      '    Properties:',\n      \"      RoleName: '\" + props.roleName + \"'\",\n      '      AssumeRolePolicyDocument:',\n      \"        Version: '2012-10-17'\",\n      '        Statement:',\n      '          - Effect: Allow',\n      '            Principal:',\n      '              AWS: !Ref GitHubOIDCRoleArn',\n      '            Action: sts:AssumeRole',\n      '            Condition:',\n      '              StringEquals:',\n      \"                aws:RequestedRegion: '\" + props.oidcRegion + \"'\",\n      '      Policies:',\n      '        - PolicyName: CloudFormationDriftAccess',\n      '          PolicyDocument:',\n      \"            Version: '2012-10-17'\",\n      '            Statement:',\n      '              # CloudFormation drift detection operations',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - cloudformation:DetectStackDrift',\n      '                  - cloudformation:DescribeStackDriftDetectionStatus',\n      '                  - cloudformation:DescribeStackResourceDrifts',\n      '                  - cloudformation:DescribeStacks',\n      '                  - cloudformation:ListStackResources',\n      '                  - cloudformation:DetectStackResourceDrift',\n      \"                Resource: '*'\",\n      '',\n      'Outputs:',\n      '  CdkDriftRoleArn:',\n      \"    Description: 'ARN of the CDK drift detection role'\",\n      '    Value: !GetAtt CdkDriftRole.Arn',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-CdkDriftRoleArn'\",\n      '',\n      '  CdkDriftRoleName:',\n      \"    Description: 'Name of the CDK drift detection role'\",\n      '    Value: !Ref CdkDriftRole',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-CdkDriftRoleName'\",\n    ];\n\n    return lines.join('\\n');\n  }\n\n  /**\n   * Generate the AWS CLI deploy command for the IAM template.\n   */\n  static generateDeployCommand(templatePath: string = 'cdk-drift-workflow-iam-template.yaml'): string {\n    return `aws cloudformation deploy --template-file ${templatePath} --stack-name cdk-drift-workflow-iam-role --capabilities CAPABILITY_NAMED_IAM`;\n  }\n}\n\n/**\n * Props for the Projen-integrated CDK Drift IAM template construct\n */\nexport interface CdkDriftIamTemplateProps extends CdkDriftIamTemplateGeneratorProps {\n  /** Projen project instance */\n  readonly project: any;\n  /** Output path for the template file (default: 'cdk-drift-workflow-iam-template.yaml') */\n  readonly outputPath?: string;\n}\n\n/**\n * Projen construct that emits a CloudFormation template with minimal IAM permissions\n * for the CDK Drift Detection Workflow.\n *\n * For non-Projen projects, use `CdkDriftIamTemplateGenerator` directly.\n */\nexport class CdkDriftIamTemplate {\n  constructor(props: CdkDriftIamTemplateProps) {\n    const outputPath = props.outputPath ?? 'cdk-drift-workflow-iam-template.yaml';\n\n    // Generate template using the generator\n    const template = CdkDriftIamTemplateGenerator.generateTemplate(props);\n    new TextFile(props.project, outputPath, { lines: template.split('\\n') });\n\n    // Add deploy task\n    props.project.addTask('deploy-cdkdrift-iam-template', {\n      description:\n        'Deploy the CDK Drift Detection IAM template via CloudFormation (accepts extra AWS CLI args, e.g., --parameter-overrides Key=Value...)',\n      receiveArgs: true,\n      exec: CdkDriftIamTemplateGenerator.generateDeployCommand(outputPath),\n    });\n  }\n}\n"]}
|
|
@@ -8,7 +8,7 @@ class CdkChangesetScript {
|
|
|
8
8
|
new projen_1.TextFile(props.project, outputPath, {
|
|
9
9
|
lines: [
|
|
10
10
|
"import { appendFile } from 'fs/promises';",
|
|
11
|
-
"import { CloudFormationClient, DescribeChangeSetCommand } from '@aws-sdk/client-cloudformation';",
|
|
11
|
+
"import { CloudFormationClient, DescribeChangeSetCommand, DescribeStacksCommand, DescribeStackResourceDriftsCommand } from '@aws-sdk/client-cloudformation';",
|
|
12
12
|
'',
|
|
13
13
|
'/**',
|
|
14
14
|
' * Small sleep helper.',
|
|
@@ -175,20 +175,81 @@ class CdkChangesetScript {
|
|
|
175
175
|
" throw new Error('Timed out waiting for change set to reach a terminal status.');",
|
|
176
176
|
'}',
|
|
177
177
|
'',
|
|
178
|
-
'
|
|
179
|
-
'
|
|
180
|
-
|
|
181
|
-
'
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
"
|
|
185
|
-
'
|
|
186
|
-
'
|
|
187
|
-
' }
|
|
178
|
+
'/**',
|
|
179
|
+
' * Parse the Link header from GitHub API responses for pagination.',
|
|
180
|
+
' */',
|
|
181
|
+
'function parseLinkHeader(header: string | null): Record<string, string> {',
|
|
182
|
+
' const links: Record<string, string> = {};',
|
|
183
|
+
' if (!header) return links;',
|
|
184
|
+
" for (const part of header.split(',')) {",
|
|
185
|
+
' const match = part.match(/<([^>]+)>;\\s*rel="([^"]+)"/);',
|
|
186
|
+
' if (match) links[match[2]] = match[1];',
|
|
187
|
+
' }',
|
|
188
|
+
' return links;',
|
|
189
|
+
'}',
|
|
190
|
+
'',
|
|
191
|
+
'/**',
|
|
192
|
+
' * Search existing PR comments for one containing the given marker.',
|
|
193
|
+
' * Returns the comment ID if found, null otherwise.',
|
|
194
|
+
' */',
|
|
195
|
+
'async function findExistingComment(commentsUrl: string, token: string, marker: string): Promise<number | null> {',
|
|
196
|
+
' let url: string | undefined = `${commentsUrl}?per_page=100`;',
|
|
197
|
+
' while (url) {',
|
|
198
|
+
' const res = await fetch(url, {',
|
|
199
|
+
' headers: {',
|
|
200
|
+
" 'Authorization': `token ${token}`,",
|
|
201
|
+
" 'Accept': 'application/vnd.github+json',",
|
|
202
|
+
' },',
|
|
203
|
+
' });',
|
|
204
|
+
' if (!res.ok) return null;',
|
|
205
|
+
' const comments: any[] = await res.json();',
|
|
206
|
+
' for (const c of comments) {',
|
|
207
|
+
" if (typeof c.body === 'string' && c.body.includes(marker)) {",
|
|
208
|
+
' return c.id;',
|
|
209
|
+
' }',
|
|
210
|
+
' }',
|
|
211
|
+
" const links = parseLinkHeader(res.headers.get('link'));",
|
|
212
|
+
' url = links.next;',
|
|
213
|
+
' }',
|
|
214
|
+
' return null;',
|
|
215
|
+
'}',
|
|
216
|
+
'',
|
|
217
|
+
'/**',
|
|
218
|
+
' * Create or update a PR comment. Uses an HTML marker to find existing comments.',
|
|
219
|
+
' * If found, PATCHes the existing comment; otherwise POSTs a new one.',
|
|
220
|
+
' */',
|
|
221
|
+
'async function upsertGithubComment(commentsUrl: string, token: string, body: string, marker: string): Promise<void> {',
|
|
222
|
+
' const markedBody = `${marker}\\n${body}`;',
|
|
223
|
+
' const existingId = await findExistingComment(commentsUrl, token, marker);',
|
|
224
|
+
'',
|
|
225
|
+
' let res: Response;',
|
|
226
|
+
' if (existingId) {',
|
|
227
|
+
" const baseUrl = commentsUrl.substring(0, commentsUrl.indexOf('/issues/'));",
|
|
228
|
+
' const patchUrl = `${baseUrl}/issues/comments/${existingId}`;',
|
|
229
|
+
' res = await fetch(patchUrl, {',
|
|
230
|
+
" method: 'PATCH',",
|
|
231
|
+
' headers: {',
|
|
232
|
+
" 'Authorization': `token ${token}`,",
|
|
233
|
+
" 'Content-Type': 'application/json',",
|
|
234
|
+
" 'Accept': 'application/vnd.github+json',",
|
|
235
|
+
' },',
|
|
236
|
+
' body: JSON.stringify({ body: markedBody }),',
|
|
237
|
+
' });',
|
|
238
|
+
' } else {',
|
|
239
|
+
' res = await fetch(commentsUrl, {',
|
|
240
|
+
" method: 'POST',",
|
|
241
|
+
' headers: {',
|
|
242
|
+
" 'Authorization': `token ${token}`,",
|
|
243
|
+
" 'Content-Type': 'application/json',",
|
|
244
|
+
" 'Accept': 'application/vnd.github+json',",
|
|
245
|
+
' },',
|
|
246
|
+
' body: JSON.stringify({ body: markedBody }),',
|
|
247
|
+
' });',
|
|
248
|
+
' }',
|
|
188
249
|
'',
|
|
189
250
|
' if (!res.ok) {',
|
|
190
251
|
" const text = await res.text().catch(() => '');",
|
|
191
|
-
|
|
252
|
+
" throw new Error(`Failed to ${existingId ? 'update' : 'post'} GitHub comment: ${res.status} ${res.statusText} ${text}`);",
|
|
192
253
|
' }',
|
|
193
254
|
'}',
|
|
194
255
|
'',
|
|
@@ -196,6 +257,69 @@ class CdkChangesetScript {
|
|
|
196
257
|
" await appendFile(summaryPath, `${content}\\n`, { encoding: 'utf8' });",
|
|
197
258
|
'}',
|
|
198
259
|
'',
|
|
260
|
+
'/**',
|
|
261
|
+
' * Check cached drift status for the stack and return an HTML banner.',
|
|
262
|
+
' * Non-fatal: returns empty string on any error.',
|
|
263
|
+
' */',
|
|
264
|
+
'async function getDriftBannerHtml(client: CloudFormationClient, stackName: string): Promise<string> {',
|
|
265
|
+
' const stackResp = await client.send(new DescribeStacksCommand({ StackName: stackName }));',
|
|
266
|
+
' const stack = stackResp.Stacks?.[0];',
|
|
267
|
+
" if (!stack) return '';",
|
|
268
|
+
'',
|
|
269
|
+
' const driftInfo = stack.DriftInformation;',
|
|
270
|
+
' const driftStatus = driftInfo?.StackDriftStatus;',
|
|
271
|
+
' const lastChecked = driftInfo?.LastCheckTimestamp;',
|
|
272
|
+
'',
|
|
273
|
+
" if (driftStatus === 'IN_SYNC') return '';",
|
|
274
|
+
'',
|
|
275
|
+
" if (driftStatus === 'NOT_CHECKED') {",
|
|
276
|
+
" return '<blockquote><p>ℹ️ Drift detection has not been run for this stack.</p></blockquote>';",
|
|
277
|
+
' }',
|
|
278
|
+
'',
|
|
279
|
+
" if (driftStatus === 'DETECTION_IN_PROGRESS') {",
|
|
280
|
+
" return '<blockquote><p>⏳ Drift detection is currently in progress.</p></blockquote>';",
|
|
281
|
+
' }',
|
|
282
|
+
'',
|
|
283
|
+
" if (driftStatus === 'DRIFTED') {",
|
|
284
|
+
" let banner = '<blockquote>';",
|
|
285
|
+
'',
|
|
286
|
+
' try {',
|
|
287
|
+
' const driftsResp = await client.send(new DescribeStackResourceDriftsCommand({',
|
|
288
|
+
' StackName: stackName,',
|
|
289
|
+
" StackResourceDriftStatusFilters: ['MODIFIED', 'DELETED'],",
|
|
290
|
+
' }));',
|
|
291
|
+
' const drifts = driftsResp.StackResourceDrifts ?? [];',
|
|
292
|
+
' const count = drifts.length;',
|
|
293
|
+
" const ts = lastChecked ? lastChecked.toISOString() : 'unknown';",
|
|
294
|
+
'',
|
|
295
|
+
" banner += `<h3>⚠️ Stack has drifted (${count} resource${count !== 1 ? 's' : ''} out of sync)</h3>`;",
|
|
296
|
+
' banner += `<p>Last drift check: <em>${ts}</em></p>`;',
|
|
297
|
+
'',
|
|
298
|
+
' if (count > 0) {',
|
|
299
|
+
" banner += '<details><summary>View drifted resources</summary>';",
|
|
300
|
+
" banner += '<table><tr><th>Resource</th><th>Type</th><th>Drift Status</th></tr>';",
|
|
301
|
+
' for (const d of drifts) {',
|
|
302
|
+
" const logicalId = d.LogicalResourceId ?? '-';",
|
|
303
|
+
" const resourceType = d.ResourceType ?? '-';",
|
|
304
|
+
" const status = d.StackResourceDriftStatus ?? '-';",
|
|
305
|
+
' banner += `<tr><td>${logicalId}</td><td>${resourceType}</td><td>${status}</td></tr>`;',
|
|
306
|
+
' }',
|
|
307
|
+
" banner += '</table></details>';",
|
|
308
|
+
' }',
|
|
309
|
+
' } catch (e: any) {',
|
|
310
|
+
" const ts = lastChecked ? lastChecked.toISOString() : 'unknown';",
|
|
311
|
+
" banner += '<h3>⚠️ Stack has drifted</h3>';",
|
|
312
|
+
' banner += `<p>Last drift check: <em>${ts}</em></p>`;',
|
|
313
|
+
" banner += `<p><em>Could not retrieve drift details: ${e?.message ?? 'unknown error'}</em></p>`;",
|
|
314
|
+
' }',
|
|
315
|
+
'',
|
|
316
|
+
" banner += '</blockquote>';",
|
|
317
|
+
' return banner;',
|
|
318
|
+
' }',
|
|
319
|
+
'',
|
|
320
|
+
" return '';",
|
|
321
|
+
'}',
|
|
322
|
+
'',
|
|
199
323
|
'async function main() {',
|
|
200
324
|
' const {',
|
|
201
325
|
' STACK_NAME,',
|
|
@@ -217,6 +341,7 @@ class CdkChangesetScript {
|
|
|
217
341
|
' }',
|
|
218
342
|
'',
|
|
219
343
|
' const changeSetName = CHANGE_SET_NAME || STACK_NAME;',
|
|
344
|
+
' const marker = `<!-- cdk-diff:stack:${STACK_NAME} -->`;',
|
|
220
345
|
'',
|
|
221
346
|
' const client = new CloudFormationClient({ region });',
|
|
222
347
|
'',
|
|
@@ -254,27 +379,35 @@ class CdkChangesetScript {
|
|
|
254
379
|
' // Build HTML exactly like pretty_format.py logic (table when there are changes; "no change." otherwise).',
|
|
255
380
|
' const html = buildHtml(STACK_NAME, filteredChanges);',
|
|
256
381
|
'',
|
|
382
|
+
' // Check cached drift status (non-fatal)',
|
|
383
|
+
" let driftBanner = '';",
|
|
384
|
+
' try {',
|
|
385
|
+
' driftBanner = await getDriftBannerHtml(client, STACK_NAME);',
|
|
386
|
+
' } catch (e: any) {',
|
|
387
|
+
" console.error('Drift check failed:', e?.message || e);",
|
|
388
|
+
' }',
|
|
389
|
+
' const fullHtml = driftBanner + html;',
|
|
390
|
+
'',
|
|
257
391
|
' // Print to stdout',
|
|
258
|
-
'
|
|
259
|
-
' console.log(html);',
|
|
392
|
+
' console.log(fullHtml);',
|
|
260
393
|
'',
|
|
261
394
|
' // Optionally append to GitHub Step Summary',
|
|
262
395
|
' if (GITHUB_STEP_SUMMARY) {',
|
|
263
396
|
' try {',
|
|
264
|
-
' await appendStepSummary(GITHUB_STEP_SUMMARY,
|
|
397
|
+
' await appendStepSummary(GITHUB_STEP_SUMMARY, fullHtml);',
|
|
265
398
|
' console.error(`Appended HTML to GITHUB_STEP_SUMMARY: ${GITHUB_STEP_SUMMARY}`);',
|
|
266
399
|
' } catch (e: any) {',
|
|
267
400
|
" console.error('Failed to append to GITHUB_STEP_SUMMARY:', e?.message || e);",
|
|
268
401
|
' }',
|
|
269
402
|
' }',
|
|
270
403
|
'',
|
|
271
|
-
' //
|
|
404
|
+
' // Upsert PR comment (find existing by marker, update or create)',
|
|
272
405
|
' if (GITHUB_TOKEN && GITHUB_COMMENT_URL) {',
|
|
273
406
|
' try {',
|
|
274
|
-
' await
|
|
275
|
-
" console.error('
|
|
407
|
+
' await upsertGithubComment(GITHUB_COMMENT_URL, GITHUB_TOKEN, fullHtml, marker);',
|
|
408
|
+
" console.error('Upserted GitHub PR comment.');",
|
|
276
409
|
' } catch (e: any) {',
|
|
277
|
-
" console.error('Failed to
|
|
410
|
+
" console.error('Failed to upsert GitHub PR comment:', e?.message || e);",
|
|
278
411
|
' // Do not fail the whole script just for comment posting',
|
|
279
412
|
' }',
|
|
280
413
|
' }',
|
|
@@ -294,4 +427,4 @@ class CdkChangesetScript {
|
|
|
294
427
|
}
|
|
295
428
|
}
|
|
296
429
|
exports.CdkChangesetScript = CdkChangesetScript;
|
|
297
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdk-changeset-script.js","sourceRoot":"","sources":["../../src/bin/cdk-changeset-script.ts"],"names":[],"mappings":"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","sourcesContent":["import { TextFile } from 'projen';\nimport { AwsCdkTypeScriptApp } from 'projen/lib/awscdk';\n\ninterface CdkChangesetScriptProps {\n  project: AwsCdkTypeScriptApp;\n  outputPath?: string;\n}\n\nexport class CdkChangesetScript {\n  constructor(props: CdkChangesetScriptProps) {\n    const outputPath = props.outputPath ?? 'projenrc/describe-cfn-changeset.ts';\n\n    new TextFile(props.project, outputPath, {\n      lines: [\n        \"import { appendFile } from 'fs/promises';\",\n        \"import { CloudFormationClient, DescribeChangeSetCommand } from '@aws-sdk/client-cloudformation';\",\n        '',\n        '/**',\n        ' * Small sleep helper.',\n        ' */',\n        'const sleep = (ms: number) => new Promise((res) => setTimeout(res, ms));',\n        '',\n        '/**',\n        ' * Build the HTML color chip for an action.',\n        ' * Modify=blue, Add=green, Remove=red (match pretty_format.py).',\n        ' */',\n        'function actionChip(action?: string): string {',\n        '  // Use emoji instead of external images for reliability',\n        '  const emojiMap: Record<string, string> = {',\n        \"    Modify: '🔵',\",\n        \"    Add: '🟢',\",\n        \"    Remove: '🔴',\",\n        '  };',\n        \"  const mark = action ? (emojiMap[action] ?? '⚪') : '⚪';\",\n        \"  return `${mark} ${action ?? '-'}`;\",\n        '}',\n        '',\n        '/**',\n        ' * Escape HTML special characters.',\n        ' */',\n        'function escapeHtml(str: string): string {',\n        \"  return str.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;');\",\n        '}',\n        '',\n        '/**',\n        ' * Format a before→after change.',\n        ' */',\n        'function formatChange(name: string, changeType: string | undefined, before: string | undefined, after: string | undefined): string {',\n        \"  const changeEmoji = changeType === 'Add' ? '🟢' : changeType === 'Remove' ? '🔴' : '🔵';\",\n        '  ',\n        '  // If both values exist and are large, use a more compact format',\n        '  const beforeLen = before?.length ?? 0;',\n        '  const afterLen = after?.length ?? 0;',\n        '  const isLarge = beforeLen > 80 || afterLen > 80;',\n        '  ',\n        '  if (before && after) {',\n        '    if (isLarge) {',\n        '      return `<details><summary>${changeEmoji} <strong>${escapeHtml(name)}</strong> (modified)</summary><strong>Before:</strong><pre>${escapeHtml(before)}</pre><strong>After:</strong><pre>${escapeHtml(after)}</pre></details>`;',\n        '    }',\n        '    return `${changeEmoji} <strong>${escapeHtml(name)}</strong>: <code>${escapeHtml(before)}</code> → <code>${escapeHtml(after)}</code>`;',\n        '  } else if (after) {',\n        '    if (isLarge) {',\n        '      return `<details><summary>${changeEmoji} <strong>${escapeHtml(name)}</strong> (added)</summary><pre>${escapeHtml(after)}</pre></details>`;',\n        '    }',\n        '    return `${changeEmoji} <strong>${escapeHtml(name)}</strong>: <code>${escapeHtml(after)}</code>`;',\n        '  } else if (before) {',\n        '    if (isLarge) {',\n        '      return `<details><summary>${changeEmoji} <strong>${escapeHtml(name)}</strong> (removed)</summary><pre>${escapeHtml(before)}</pre></details>`;',\n        '    }',\n        '    return `${changeEmoji} <strong>${escapeHtml(name)}</strong>: <s><code>${escapeHtml(before)}</code></s>`;',\n        '  }',\n        '  // Fallback: no values available (API did not return them)',\n        '  return `${changeEmoji} <strong>${escapeHtml(name)}</strong>`;',\n        '}',\n        '',\n        '/**',\n        ' * Extract changed properties/tags/attributes with before/after values.',\n        ' */',\n        'function changedPropertiesHTML(change: any): string {',\n        '  const details = change?.ResourceChange?.Details ?? [];',\n        '  const props: string[] = [];',\n        '  for (const d of details) {',\n        '    const attr = d?.Target?.Attribute;',\n        '    // Use property name if available, otherwise use attribute type',\n        \"    const name = d?.Target?.Name ?? attr ?? 'unknown';\",\n        '    const changeType = d?.Target?.AttributeChangeType;',\n        '    const before = d?.Target?.BeforeValue;',\n        '    const after = d?.Target?.AfterValue;',\n        '    props.push(formatChange(name, changeType, before, after));',\n        '  }',\n        \"  return props.join('<br>');\",\n        '}',\n        '',\n        '/**',\n        ' * Determine if a change should be ignored based on logical IDs and/or resource types.',\n        \" * - IGNORE_LOGICAL_IDS: comma-separated list of logical IDs to ignore (default includes 'CDKMetadata')\",\n        \" * - IGNORE_RESOURCE_TYPES: comma-separated list of resource types to ignore (e.g., 'AWS::CDK::Metadata')\",\n        ' */',\n        'function shouldIgnoreChange(change: any, ignoreIds: Set<string>, ignoreTypes: Set<string>): boolean {',\n        '  const rc = change?.ResourceChange ?? {};',\n        '  const logicalId = rc?.LogicalResourceId as string | undefined;',\n        '  const resourceType = rc?.ResourceType as string | undefined;',\n        '  if (logicalId && ignoreIds.has(logicalId)) return true;',\n        '  if (resourceType && ignoreTypes.has(resourceType)) return true;',\n        '  return false;',\n        '}',\n        '',\n        '/**',\n        ' * Generate the HTML body similar to pretty_format.py',\n        ' */',\n        'function buildHtml(stackName: string, changes: any[]): string {',\n        '  let body = `<h1>Change set</h1><h2>Stack Name: ${stackName}</h2><br>`;',\n        '  if ((changes?.length ?? 0) > 0) {',\n        \"    body += '<table><tr><th>Action&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</th><th>ID</th><th>Type</th><th>Replacement</th><th>Details</th></tr>';\",\n        '    for (const c of changes) {',\n        '      const rc = c?.ResourceChange ?? {};',\n        '      const action = actionChip(rc?.Action);',\n        \"      const logicalId = rc?.LogicalResourceId ?? '-';\",\n        \"      const type = rc?.ResourceType ?? '-';\",\n        \"      const replacement = rc?.Replacement ?? '-';\",\n        '      const details = changedPropertiesHTML(c);',\n        '',\n        \"      body += '<tr>';\",\n        '      body += `<td>${action}</td>`;',\n        '      body += `<td>${logicalId}</td>`;',\n        '      body += `<td>${type}</td>`;',\n        '      body += `<td>${replacement}</td>`;',\n        '      body += `<td>${details}</td>`;',\n        \"      body += '</tr>';\",\n        '    }',\n        \"    body += '</table>';\",\n        '  } else {',\n        \"    body += 'no change.';\",\n        '  }',\n        '  return body;',\n        '}',\n        '',\n        '/**',\n        ' * Poll DescribeChangeSet until a terminal status, then paginate to retrieve all Changes.',\n        ' */',\n        'async function getTerminalChangeSet(',\n        '  client: CloudFormationClient,',\n        '  stackName: string,',\n        '  changeSetName: string,',\n        '  maxAttempts = 60,',\n        '  delayMs = 3000,',\n        '): Promise<{ status?: string; statusReason?: string; changes: any[] }> {',\n        '  for (let attempt = 1; attempt <= maxAttempts; attempt++) {',\n        '    const resp = await client.send(new DescribeChangeSetCommand({',\n        '      StackName: stackName,',\n        '      ChangeSetName: changeSetName,',\n        '      IncludePropertyValues: true,',\n        '    }));',\n        '',\n        '    const status = resp.Status;',\n        '    const statusReason = resp.StatusReason;',\n        '',\n        \"    if (status === 'CREATE_COMPLETE' || status === 'FAILED') {\",\n        '      // Gather all pages',\n        '      const changes: any[] = [];',\n        '      if (resp.Changes) changes.push(...resp.Changes);',\n        '      let next = resp.NextToken;',\n        '      while (next) {',\n        '        const page = await client.send(new DescribeChangeSetCommand({',\n        '          StackName: stackName,',\n        '          ChangeSetName: changeSetName,',\n        '          IncludePropertyValues: true,',\n        '          NextToken: next,',\n        '        }));',\n        '        if (page.Changes) changes.push(...page.Changes);',\n        '        next = page.NextToken;',\n        '      }',\n        '      return { status, statusReason, changes };',\n        '    }',\n        '',\n        '    // Not terminal yet; wait and retry',\n        '    await sleep(delayMs);',\n        '  }',\n        '',\n        \"  throw new Error('Timed out waiting for change set to reach a terminal status.');\",\n        '}',\n        '',\n        'async function postGithubComment(url: string, token: string, body: string): Promise<void> {',\n        '  const res = await fetch(url, {',\n        \"    method: 'POST',\",\n        '    headers: {',\n        \"      'Authorization': `token ${token}`,\",\n        \"      'Content-Type': 'application/json',\",\n        \"      'Accept': 'application/vnd.github+json',\",\n        '    },',\n        '    body: JSON.stringify({ body }),',\n        '  });',\n        '',\n        '  if (!res.ok) {',\n        \"    const text = await res.text().catch(() => '');\",\n        '    throw new Error(`Failed to post GitHub comment: ${res.status} ${res.statusText} ${text}`);',\n        '  }',\n        '}',\n        '',\n        'async function appendStepSummary(summaryPath: string, content: string): Promise<void> {',\n        \"  await appendFile(summaryPath, `${content}\\\\n`, { encoding: 'utf8' });\",\n        '}',\n        '',\n        'async function main() {',\n        '  const {',\n        '    STACK_NAME,',\n        '    CHANGE_SET_NAME,',\n        '    AWS_REGION,',\n        '    GITHUB_TOKEN,',\n        '    GITHUB_COMMENT_URL,',\n        '    GITHUB_STEP_SUMMARY,',\n        '    IGNORE_LOGICAL_IDS,',\n        '    IGNORE_RESOURCE_TYPES,',\n        '  } = process.env;',\n        '',\n        '  if (!STACK_NAME) {',\n        \"    throw new Error('STACK_NAME is required');\",\n        '  }',\n        '  const region = AWS_REGION || process.env.AWS_DEFAULT_REGION;',\n        '  if (!region) {',\n        \"    throw new Error('AWS_REGION is required');\",\n        '  }',\n        '',\n        '  const changeSetName = CHANGE_SET_NAME || STACK_NAME;',\n        '',\n        '  const client = new CloudFormationClient({ region });',\n        '',\n        '  let status: string | undefined;',\n        '  let statusReason: string | undefined;',\n        '  let changes: any[] = [];',\n        '',\n        '  try {',\n        '    const result = await getTerminalChangeSet(client, STACK_NAME, changeSetName);',\n        '    status = result.status;',\n        '    statusReason = result.statusReason;',\n        '    changes = result.changes ?? [];',\n        '  } catch (err: any) {',\n        '    // If DescribeChangeSet fails entirely, surface the error',\n        \"    console.error('Error describing change set:', err?.message || err);\",\n        '    process.exitCode = 1;',\n        '    return;',\n        '  }',\n        '',\n        \"  // Apply ignores from env vars (IDs and types). Default ignore IDs include 'CDKMetadata'.\",\n        '  const ignoreIdSet = new Set(',\n        \"    (IGNORE_LOGICAL_IDS ?? 'CDKMetadata')\",\n        \"      .split(',')\",\n        '      .map(s => s.trim())',\n        '      .filter(Boolean),',\n        '  );',\n        '  const ignoreTypeSet = new Set(',\n        \"    (IGNORE_RESOURCE_TYPES ?? '')\",\n        \"      .split(',')\",\n        '      .map(s => s.trim())',\n        '      .filter(Boolean),',\n        '  );',\n        '  const filteredChanges = changes.filter(c => !shouldIgnoreChange(c, ignoreIdSet, ignoreTypeSet));',\n        '',\n        '  // Build HTML exactly like pretty_format.py logic (table when there are changes; \"no change.\" otherwise).',\n        '  const html = buildHtml(STACK_NAME, filteredChanges);',\n        '',\n        '  // Print to stdout',\n        '  // This allows capturing output or redirecting to a file if needed.',\n        '  console.log(html);',\n        '',\n        '  // Optionally append to GitHub Step Summary',\n        '  if (GITHUB_STEP_SUMMARY) {',\n        '    try {',\n        '      await appendStepSummary(GITHUB_STEP_SUMMARY, html);',\n        '      console.error(`Appended HTML to GITHUB_STEP_SUMMARY: ${GITHUB_STEP_SUMMARY}`);',\n        '    } catch (e: any) {',\n        \"      console.error('Failed to append to GITHUB_STEP_SUMMARY:', e?.message || e);\",\n        '    }',\n        '  }',\n        '',\n        '  // Optionally post a PR comment',\n        '  if (GITHUB_TOKEN && GITHUB_COMMENT_URL) {',\n        '    try {',\n        '      await postGithubComment(GITHUB_COMMENT_URL, GITHUB_TOKEN, html);',\n        \"      console.error('Posted HTML as a GitHub PR comment.');\",\n        '    } catch (e: any) {',\n        \"      console.error('Failed to post GitHub PR comment:', e?.message || e);\",\n        '      // Do not fail the whole script just for comment posting',\n        '    }',\n        '  }',\n        '',\n        \"  // Note: When status is FAILED due to \\\"didn't contain changes\\\", the HTML naturally says \\\"no change.\\\"\",\n        \"  if (status === 'FAILED' && statusReason) {\",\n        '    console.error(`Change set status: FAILED. Reason: ${statusReason}`);',\n        '  }',\n        '}',\n        '',\n        'main().catch((err) => {',\n        '  console.error(err);',\n        '  process.exitCode = 1;',\n        '});',\n      ],\n    });\n  }\n}\n"]}
|
|
430
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdk-changeset-script.js","sourceRoot":"","sources":["../../src/bin/cdk-changeset-script.ts"],"names":[],"mappings":"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","sourcesContent":["import { TextFile } from 'projen';\nimport { AwsCdkTypeScriptApp } from 'projen/lib/awscdk';\n\ninterface CdkChangesetScriptProps {\n  project: AwsCdkTypeScriptApp;\n  outputPath?: string;\n}\n\nexport class CdkChangesetScript {\n  constructor(props: CdkChangesetScriptProps) {\n    const outputPath = props.outputPath ?? 'projenrc/describe-cfn-changeset.ts';\n\n    new TextFile(props.project, outputPath, {\n      lines: [\n        \"import { appendFile } from 'fs/promises';\",\n        \"import { CloudFormationClient, DescribeChangeSetCommand, DescribeStacksCommand, DescribeStackResourceDriftsCommand } from '@aws-sdk/client-cloudformation';\",\n        '',\n        '/**',\n        ' * Small sleep helper.',\n        ' */',\n        'const sleep = (ms: number) => new Promise((res) => setTimeout(res, ms));',\n        '',\n        '/**',\n        ' * Build the HTML color chip for an action.',\n        ' * Modify=blue, Add=green, Remove=red (match pretty_format.py).',\n        ' */',\n        'function actionChip(action?: string): string {',\n        '  // Use emoji instead of external images for reliability',\n        '  const emojiMap: Record<string, string> = {',\n        \"    Modify: '🔵',\",\n        \"    Add: '🟢',\",\n        \"    Remove: '🔴',\",\n        '  };',\n        \"  const mark = action ? (emojiMap[action] ?? '⚪') : '⚪';\",\n        \"  return `${mark} ${action ?? '-'}`;\",\n        '}',\n        '',\n        '/**',\n        ' * Escape HTML special characters.',\n        ' */',\n        'function escapeHtml(str: string): string {',\n        \"  return str.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;');\",\n        '}',\n        '',\n        '/**',\n        ' * Format a before→after change.',\n        ' */',\n        'function formatChange(name: string, changeType: string | undefined, before: string | undefined, after: string | undefined): string {',\n        \"  const changeEmoji = changeType === 'Add' ? '🟢' : changeType === 'Remove' ? '🔴' : '🔵';\",\n        '  ',\n        '  // If both values exist and are large, use a more compact format',\n        '  const beforeLen = before?.length ?? 0;',\n        '  const afterLen = after?.length ?? 0;',\n        '  const isLarge = beforeLen > 80 || afterLen > 80;',\n        '  ',\n        '  if (before && after) {',\n        '    if (isLarge) {',\n        '      return `<details><summary>${changeEmoji} <strong>${escapeHtml(name)}</strong> (modified)</summary><strong>Before:</strong><pre>${escapeHtml(before)}</pre><strong>After:</strong><pre>${escapeHtml(after)}</pre></details>`;',\n        '    }',\n        '    return `${changeEmoji} <strong>${escapeHtml(name)}</strong>: <code>${escapeHtml(before)}</code> → <code>${escapeHtml(after)}</code>`;',\n        '  } else if (after) {',\n        '    if (isLarge) {',\n        '      return `<details><summary>${changeEmoji} <strong>${escapeHtml(name)}</strong> (added)</summary><pre>${escapeHtml(after)}</pre></details>`;',\n        '    }',\n        '    return `${changeEmoji} <strong>${escapeHtml(name)}</strong>: <code>${escapeHtml(after)}</code>`;',\n        '  } else if (before) {',\n        '    if (isLarge) {',\n        '      return `<details><summary>${changeEmoji} <strong>${escapeHtml(name)}</strong> (removed)</summary><pre>${escapeHtml(before)}</pre></details>`;',\n        '    }',\n        '    return `${changeEmoji} <strong>${escapeHtml(name)}</strong>: <s><code>${escapeHtml(before)}</code></s>`;',\n        '  }',\n        '  // Fallback: no values available (API did not return them)',\n        '  return `${changeEmoji} <strong>${escapeHtml(name)}</strong>`;',\n        '}',\n        '',\n        '/**',\n        ' * Extract changed properties/tags/attributes with before/after values.',\n        ' */',\n        'function changedPropertiesHTML(change: any): string {',\n        '  const details = change?.ResourceChange?.Details ?? [];',\n        '  const props: string[] = [];',\n        '  for (const d of details) {',\n        '    const attr = d?.Target?.Attribute;',\n        '    // Use property name if available, otherwise use attribute type',\n        \"    const name = d?.Target?.Name ?? attr ?? 'unknown';\",\n        '    const changeType = d?.Target?.AttributeChangeType;',\n        '    const before = d?.Target?.BeforeValue;',\n        '    const after = d?.Target?.AfterValue;',\n        '    props.push(formatChange(name, changeType, before, after));',\n        '  }',\n        \"  return props.join('<br>');\",\n        '}',\n        '',\n        '/**',\n        ' * Determine if a change should be ignored based on logical IDs and/or resource types.',\n        \" * - IGNORE_LOGICAL_IDS: comma-separated list of logical IDs to ignore (default includes 'CDKMetadata')\",\n        \" * - IGNORE_RESOURCE_TYPES: comma-separated list of resource types to ignore (e.g., 'AWS::CDK::Metadata')\",\n        ' */',\n        'function shouldIgnoreChange(change: any, ignoreIds: Set<string>, ignoreTypes: Set<string>): boolean {',\n        '  const rc = change?.ResourceChange ?? {};',\n        '  const logicalId = rc?.LogicalResourceId as string | undefined;',\n        '  const resourceType = rc?.ResourceType as string | undefined;',\n        '  if (logicalId && ignoreIds.has(logicalId)) return true;',\n        '  if (resourceType && ignoreTypes.has(resourceType)) return true;',\n        '  return false;',\n        '}',\n        '',\n        '/**',\n        ' * Generate the HTML body similar to pretty_format.py',\n        ' */',\n        'function buildHtml(stackName: string, changes: any[]): string {',\n        '  let body = `<h1>Change set</h1><h2>Stack Name: ${stackName}</h2><br>`;',\n        '  if ((changes?.length ?? 0) > 0) {',\n        \"    body += '<table><tr><th>Action&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</th><th>ID</th><th>Type</th><th>Replacement</th><th>Details</th></tr>';\",\n        '    for (const c of changes) {',\n        '      const rc = c?.ResourceChange ?? {};',\n        '      const action = actionChip(rc?.Action);',\n        \"      const logicalId = rc?.LogicalResourceId ?? '-';\",\n        \"      const type = rc?.ResourceType ?? '-';\",\n        \"      const replacement = rc?.Replacement ?? '-';\",\n        '      const details = changedPropertiesHTML(c);',\n        '',\n        \"      body += '<tr>';\",\n        '      body += `<td>${action}</td>`;',\n        '      body += `<td>${logicalId}</td>`;',\n        '      body += `<td>${type}</td>`;',\n        '      body += `<td>${replacement}</td>`;',\n        '      body += `<td>${details}</td>`;',\n        \"      body += '</tr>';\",\n        '    }',\n        \"    body += '</table>';\",\n        '  } else {',\n        \"    body += 'no change.';\",\n        '  }',\n        '  return body;',\n        '}',\n        '',\n        '/**',\n        ' * Poll DescribeChangeSet until a terminal status, then paginate to retrieve all Changes.',\n        ' */',\n        'async function getTerminalChangeSet(',\n        '  client: CloudFormationClient,',\n        '  stackName: string,',\n        '  changeSetName: string,',\n        '  maxAttempts = 60,',\n        '  delayMs = 3000,',\n        '): Promise<{ status?: string; statusReason?: string; changes: any[] }> {',\n        '  for (let attempt = 1; attempt <= maxAttempts; attempt++) {',\n        '    const resp = await client.send(new DescribeChangeSetCommand({',\n        '      StackName: stackName,',\n        '      ChangeSetName: changeSetName,',\n        '      IncludePropertyValues: true,',\n        '    }));',\n        '',\n        '    const status = resp.Status;',\n        '    const statusReason = resp.StatusReason;',\n        '',\n        \"    if (status === 'CREATE_COMPLETE' || status === 'FAILED') {\",\n        '      // Gather all pages',\n        '      const changes: any[] = [];',\n        '      if (resp.Changes) changes.push(...resp.Changes);',\n        '      let next = resp.NextToken;',\n        '      while (next) {',\n        '        const page = await client.send(new DescribeChangeSetCommand({',\n        '          StackName: stackName,',\n        '          ChangeSetName: changeSetName,',\n        '          IncludePropertyValues: true,',\n        '          NextToken: next,',\n        '        }));',\n        '        if (page.Changes) changes.push(...page.Changes);',\n        '        next = page.NextToken;',\n        '      }',\n        '      return { status, statusReason, changes };',\n        '    }',\n        '',\n        '    // Not terminal yet; wait and retry',\n        '    await sleep(delayMs);',\n        '  }',\n        '',\n        \"  throw new Error('Timed out waiting for change set to reach a terminal status.');\",\n        '}',\n        '',\n        '/**',\n        ' * Parse the Link header from GitHub API responses for pagination.',\n        ' */',\n        'function parseLinkHeader(header: string | null): Record<string, string> {',\n        '  const links: Record<string, string> = {};',\n        '  if (!header) return links;',\n        \"  for (const part of header.split(',')) {\",\n        '    const match = part.match(/<([^>]+)>;\\\\s*rel=\"([^\"]+)\"/);',\n        '    if (match) links[match[2]] = match[1];',\n        '  }',\n        '  return links;',\n        '}',\n        '',\n        '/**',\n        ' * Search existing PR comments for one containing the given marker.',\n        ' * Returns the comment ID if found, null otherwise.',\n        ' */',\n        'async function findExistingComment(commentsUrl: string, token: string, marker: string): Promise<number | null> {',\n        '  let url: string | undefined = `${commentsUrl}?per_page=100`;',\n        '  while (url) {',\n        '    const res = await fetch(url, {',\n        '      headers: {',\n        \"        'Authorization': `token ${token}`,\",\n        \"        'Accept': 'application/vnd.github+json',\",\n        '      },',\n        '    });',\n        '    if (!res.ok) return null;',\n        '    const comments: any[] = await res.json();',\n        '    for (const c of comments) {',\n        \"      if (typeof c.body === 'string' && c.body.includes(marker)) {\",\n        '        return c.id;',\n        '      }',\n        '    }',\n        \"    const links = parseLinkHeader(res.headers.get('link'));\",\n        '    url = links.next;',\n        '  }',\n        '  return null;',\n        '}',\n        '',\n        '/**',\n        ' * Create or update a PR comment. Uses an HTML marker to find existing comments.',\n        ' * If found, PATCHes the existing comment; otherwise POSTs a new one.',\n        ' */',\n        'async function upsertGithubComment(commentsUrl: string, token: string, body: string, marker: string): Promise<void> {',\n        '  const markedBody = `${marker}\\\\n${body}`;',\n        '  const existingId = await findExistingComment(commentsUrl, token, marker);',\n        '',\n        '  let res: Response;',\n        '  if (existingId) {',\n        \"    const baseUrl = commentsUrl.substring(0, commentsUrl.indexOf('/issues/'));\",\n        '    const patchUrl = `${baseUrl}/issues/comments/${existingId}`;',\n        '    res = await fetch(patchUrl, {',\n        \"      method: 'PATCH',\",\n        '      headers: {',\n        \"        'Authorization': `token ${token}`,\",\n        \"        'Content-Type': 'application/json',\",\n        \"        'Accept': 'application/vnd.github+json',\",\n        '      },',\n        '      body: JSON.stringify({ body: markedBody }),',\n        '    });',\n        '  } else {',\n        '    res = await fetch(commentsUrl, {',\n        \"      method: 'POST',\",\n        '      headers: {',\n        \"        'Authorization': `token ${token}`,\",\n        \"        'Content-Type': 'application/json',\",\n        \"        'Accept': 'application/vnd.github+json',\",\n        '      },',\n        '      body: JSON.stringify({ body: markedBody }),',\n        '    });',\n        '  }',\n        '',\n        '  if (!res.ok) {',\n        \"    const text = await res.text().catch(() => '');\",\n        \"    throw new Error(`Failed to ${existingId ? 'update' : 'post'} GitHub comment: ${res.status} ${res.statusText} ${text}`);\",\n        '  }',\n        '}',\n        '',\n        'async function appendStepSummary(summaryPath: string, content: string): Promise<void> {',\n        \"  await appendFile(summaryPath, `${content}\\\\n`, { encoding: 'utf8' });\",\n        '}',\n        '',\n        '/**',\n        ' * Check cached drift status for the stack and return an HTML banner.',\n        ' * Non-fatal: returns empty string on any error.',\n        ' */',\n        'async function getDriftBannerHtml(client: CloudFormationClient, stackName: string): Promise<string> {',\n        '  const stackResp = await client.send(new DescribeStacksCommand({ StackName: stackName }));',\n        '  const stack = stackResp.Stacks?.[0];',\n        \"  if (!stack) return '';\",\n        '',\n        '  const driftInfo = stack.DriftInformation;',\n        '  const driftStatus = driftInfo?.StackDriftStatus;',\n        '  const lastChecked = driftInfo?.LastCheckTimestamp;',\n        '',\n        \"  if (driftStatus === 'IN_SYNC') return '';\",\n        '',\n        \"  if (driftStatus === 'NOT_CHECKED') {\",\n        \"    return '<blockquote><p>ℹ️ Drift detection has not been run for this stack.</p></blockquote>';\",\n        '  }',\n        '',\n        \"  if (driftStatus === 'DETECTION_IN_PROGRESS') {\",\n        \"    return '<blockquote><p>⏳ Drift detection is currently in progress.</p></blockquote>';\",\n        '  }',\n        '',\n        \"  if (driftStatus === 'DRIFTED') {\",\n        \"    let banner = '<blockquote>';\",\n        '',\n        '    try {',\n        '      const driftsResp = await client.send(new DescribeStackResourceDriftsCommand({',\n        '        StackName: stackName,',\n        \"        StackResourceDriftStatusFilters: ['MODIFIED', 'DELETED'],\",\n        '      }));',\n        '      const drifts = driftsResp.StackResourceDrifts ?? [];',\n        '      const count = drifts.length;',\n        \"      const ts = lastChecked ? lastChecked.toISOString() : 'unknown';\",\n        '',\n        \"      banner += `<h3>⚠️ Stack has drifted (${count} resource${count !== 1 ? 's' : ''} out of sync)</h3>`;\",\n        '      banner += `<p>Last drift check: <em>${ts}</em></p>`;',\n        '',\n        '      if (count > 0) {',\n        \"        banner += '<details><summary>View drifted resources</summary>';\",\n        \"        banner += '<table><tr><th>Resource</th><th>Type</th><th>Drift Status</th></tr>';\",\n        '        for (const d of drifts) {',\n        \"          const logicalId = d.LogicalResourceId ?? '-';\",\n        \"          const resourceType = d.ResourceType ?? '-';\",\n        \"          const status = d.StackResourceDriftStatus ?? '-';\",\n        '          banner += `<tr><td>${logicalId}</td><td>${resourceType}</td><td>${status}</td></tr>`;',\n        '        }',\n        \"        banner += '</table></details>';\",\n        '      }',\n        '    } catch (e: any) {',\n        \"      const ts = lastChecked ? lastChecked.toISOString() : 'unknown';\",\n        \"      banner += '<h3>⚠️ Stack has drifted</h3>';\",\n        '      banner += `<p>Last drift check: <em>${ts}</em></p>`;',\n        \"      banner += `<p><em>Could not retrieve drift details: ${e?.message ?? 'unknown error'}</em></p>`;\",\n        '    }',\n        '',\n        \"    banner += '</blockquote>';\",\n        '    return banner;',\n        '  }',\n        '',\n        \"  return '';\",\n        '}',\n        '',\n        'async function main() {',\n        '  const {',\n        '    STACK_NAME,',\n        '    CHANGE_SET_NAME,',\n        '    AWS_REGION,',\n        '    GITHUB_TOKEN,',\n        '    GITHUB_COMMENT_URL,',\n        '    GITHUB_STEP_SUMMARY,',\n        '    IGNORE_LOGICAL_IDS,',\n        '    IGNORE_RESOURCE_TYPES,',\n        '  } = process.env;',\n        '',\n        '  if (!STACK_NAME) {',\n        \"    throw new Error('STACK_NAME is required');\",\n        '  }',\n        '  const region = AWS_REGION || process.env.AWS_DEFAULT_REGION;',\n        '  if (!region) {',\n        \"    throw new Error('AWS_REGION is required');\",\n        '  }',\n        '',\n        '  const changeSetName = CHANGE_SET_NAME || STACK_NAME;',\n        '  const marker = `<!-- cdk-diff:stack:${STACK_NAME} -->`;',\n        '',\n        '  const client = new CloudFormationClient({ region });',\n        '',\n        '  let status: string | undefined;',\n        '  let statusReason: string | undefined;',\n        '  let changes: any[] = [];',\n        '',\n        '  try {',\n        '    const result = await getTerminalChangeSet(client, STACK_NAME, changeSetName);',\n        '    status = result.status;',\n        '    statusReason = result.statusReason;',\n        '    changes = result.changes ?? [];',\n        '  } catch (err: any) {',\n        '    // If DescribeChangeSet fails entirely, surface the error',\n        \"    console.error('Error describing change set:', err?.message || err);\",\n        '    process.exitCode = 1;',\n        '    return;',\n        '  }',\n        '',\n        \"  // Apply ignores from env vars (IDs and types). Default ignore IDs include 'CDKMetadata'.\",\n        '  const ignoreIdSet = new Set(',\n        \"    (IGNORE_LOGICAL_IDS ?? 'CDKMetadata')\",\n        \"      .split(',')\",\n        '      .map(s => s.trim())',\n        '      .filter(Boolean),',\n        '  );',\n        '  const ignoreTypeSet = new Set(',\n        \"    (IGNORE_RESOURCE_TYPES ?? '')\",\n        \"      .split(',')\",\n        '      .map(s => s.trim())',\n        '      .filter(Boolean),',\n        '  );',\n        '  const filteredChanges = changes.filter(c => !shouldIgnoreChange(c, ignoreIdSet, ignoreTypeSet));',\n        '',\n        '  // Build HTML exactly like pretty_format.py logic (table when there are changes; \"no change.\" otherwise).',\n        '  const html = buildHtml(STACK_NAME, filteredChanges);',\n        '',\n        '  // Check cached drift status (non-fatal)',\n        \"  let driftBanner = '';\",\n        '  try {',\n        '    driftBanner = await getDriftBannerHtml(client, STACK_NAME);',\n        '  } catch (e: any) {',\n        \"    console.error('Drift check failed:', e?.message || e);\",\n        '  }',\n        '  const fullHtml = driftBanner + html;',\n        '',\n        '  // Print to stdout',\n        '  console.log(fullHtml);',\n        '',\n        '  // Optionally append to GitHub Step Summary',\n        '  if (GITHUB_STEP_SUMMARY) {',\n        '    try {',\n        '      await appendStepSummary(GITHUB_STEP_SUMMARY, fullHtml);',\n        '      console.error(`Appended HTML to GITHUB_STEP_SUMMARY: ${GITHUB_STEP_SUMMARY}`);',\n        '    } catch (e: any) {',\n        \"      console.error('Failed to append to GITHUB_STEP_SUMMARY:', e?.message || e);\",\n        '    }',\n        '  }',\n        '',\n        '  // Upsert PR comment (find existing by marker, update or create)',\n        '  if (GITHUB_TOKEN && GITHUB_COMMENT_URL) {',\n        '    try {',\n        '      await upsertGithubComment(GITHUB_COMMENT_URL, GITHUB_TOKEN, fullHtml, marker);',\n        \"      console.error('Upserted GitHub PR comment.');\",\n        '    } catch (e: any) {',\n        \"      console.error('Failed to upsert GitHub PR comment:', e?.message || e);\",\n        '      // Do not fail the whole script just for comment posting',\n        '    }',\n        '  }',\n        '',\n        \"  // Note: When status is FAILED due to \\\"didn't contain changes\\\", the HTML naturally says \\\"no change.\\\"\",\n        \"  if (status === 'FAILED' && statusReason) {\",\n        '    console.error(`Change set status: FAILED. Reason: ${statusReason}`);',\n        '  }',\n        '}',\n        '',\n        'main().catch((err) => {',\n        '  console.error(err);',\n        '  process.exitCode = 1;',\n        '});',\n      ],\n    });\n  }\n}\n"]}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@aws-sdk/client-cloudformation",
|
|
3
3
|
"description": "AWS SDK for JavaScript Cloudformation Client for Node.js, Browser and React Native",
|
|
4
|
-
"version": "3.
|
|
4
|
+
"version": "3.987.0",
|
|
5
5
|
"scripts": {
|
|
6
6
|
"build": "concurrently 'yarn:build:types' 'yarn:build:es' && yarn build:cjs",
|
|
7
7
|
"build:cjs": "node ../../scripts/compilation/inline client-cloudformation",
|
|
@@ -31,7 +31,7 @@
|
|
|
31
31
|
"@aws-sdk/middleware-user-agent": "^3.972.7",
|
|
32
32
|
"@aws-sdk/region-config-resolver": "^3.972.3",
|
|
33
33
|
"@aws-sdk/types": "^3.973.1",
|
|
34
|
-
"@aws-sdk/util-endpoints": "3.
|
|
34
|
+
"@aws-sdk/util-endpoints": "3.987.0",
|
|
35
35
|
"@aws-sdk/util-user-agent-browser": "^3.972.3",
|
|
36
36
|
"@aws-sdk/util-user-agent-node": "^3.972.5",
|
|
37
37
|
"@smithy/config-resolver": "^4.4.6",
|
|
@@ -990,8 +990,8 @@ class SmithyRpcV2CborProtocol extends protocols.RpcProtocol {
|
|
|
990
990
|
codec = new CborCodec();
|
|
991
991
|
serializer = this.codec.createSerializer();
|
|
992
992
|
deserializer = this.codec.createDeserializer();
|
|
993
|
-
constructor({ defaultNamespace }) {
|
|
994
|
-
super({ defaultNamespace });
|
|
993
|
+
constructor({ defaultNamespace, errorTypeRegistries, }) {
|
|
994
|
+
super({ defaultNamespace, errorTypeRegistries });
|
|
995
995
|
}
|
|
996
996
|
getShapeId() {
|
|
997
997
|
return "smithy.protocols#rpcv2Cbor";
|
|
@@ -1035,15 +1035,17 @@ class SmithyRpcV2CborProtocol extends protocols.RpcProtocol {
|
|
|
1035
1035
|
}
|
|
1036
1036
|
async handleError(operationSchema, context, response, dataObject, metadata) {
|
|
1037
1037
|
const errorName = loadSmithyRpcV2CborErrorCode(response, dataObject) ?? "Unknown";
|
|
1038
|
-
let namespace = this.options.defaultNamespace;
|
|
1039
|
-
if (errorName.includes("#")) {
|
|
1040
|
-
[namespace] = errorName.split("#");
|
|
1041
|
-
}
|
|
1042
1038
|
const errorMetadata = {
|
|
1043
1039
|
$metadata: metadata,
|
|
1044
1040
|
$fault: response.statusCode <= 500 ? "client" : "server",
|
|
1045
1041
|
};
|
|
1046
|
-
|
|
1042
|
+
let namespace = this.options.defaultNamespace;
|
|
1043
|
+
if (errorName.includes("#")) {
|
|
1044
|
+
[namespace] = errorName.split("#");
|
|
1045
|
+
}
|
|
1046
|
+
const registry = this.compositeErrorRegistry;
|
|
1047
|
+
const nsRegistry = schema.TypeRegistry.for(namespace);
|
|
1048
|
+
registry.copyFrom(nsRegistry);
|
|
1047
1049
|
let errorSchema;
|
|
1048
1050
|
try {
|
|
1049
1051
|
errorSchema = registry.getSchema(errorName);
|
|
@@ -1052,10 +1054,11 @@ class SmithyRpcV2CborProtocol extends protocols.RpcProtocol {
|
|
|
1052
1054
|
if (dataObject.Message) {
|
|
1053
1055
|
dataObject.message = dataObject.Message;
|
|
1054
1056
|
}
|
|
1055
|
-
const
|
|
1056
|
-
|
|
1057
|
+
const syntheticRegistry = schema.TypeRegistry.for("smithy.ts.sdk.synthetic." + namespace);
|
|
1058
|
+
registry.copyFrom(syntheticRegistry);
|
|
1059
|
+
const baseExceptionSchema = registry.getBaseException();
|
|
1057
1060
|
if (baseExceptionSchema) {
|
|
1058
|
-
const ErrorCtor =
|
|
1061
|
+
const ErrorCtor = registry.getErrorCtor(baseExceptionSchema);
|
|
1059
1062
|
throw Object.assign(new ErrorCtor({ name: errorName }), errorMetadata, dataObject);
|
|
1060
1063
|
}
|
|
1061
1064
|
throw Object.assign(new Error(errorName), errorMetadata, dataObject);
|
|
@@ -33,9 +33,14 @@ class SerdeContext {
|
|
|
33
33
|
|
|
34
34
|
class HttpProtocol extends SerdeContext {
|
|
35
35
|
options;
|
|
36
|
+
compositeErrorRegistry;
|
|
36
37
|
constructor(options) {
|
|
37
38
|
super();
|
|
38
39
|
this.options = options;
|
|
40
|
+
this.compositeErrorRegistry = schema.TypeRegistry.for(options.defaultNamespace);
|
|
41
|
+
for (const etr of options.errorTypeRegistries ?? []) {
|
|
42
|
+
this.compositeErrorRegistry.copyFrom(etr);
|
|
43
|
+
}
|
|
39
44
|
}
|
|
40
45
|
getRequestType() {
|
|
41
46
|
return protocolHttp.HttpRequest;
|
|
@@ -566,10 +566,24 @@ class TypeRegistry {
|
|
|
566
566
|
}
|
|
567
567
|
return TypeRegistry.registries.get(namespace);
|
|
568
568
|
}
|
|
569
|
+
copyFrom(other) {
|
|
570
|
+
const { schemas, exceptions } = this;
|
|
571
|
+
for (const [k, v] of other.schemas) {
|
|
572
|
+
if (!schemas.has(k)) {
|
|
573
|
+
schemas.set(k, v);
|
|
574
|
+
}
|
|
575
|
+
}
|
|
576
|
+
for (const [k, v] of other.exceptions) {
|
|
577
|
+
if (!exceptions.has(k)) {
|
|
578
|
+
exceptions.set(k, v);
|
|
579
|
+
}
|
|
580
|
+
}
|
|
581
|
+
}
|
|
569
582
|
register(shapeId, schema) {
|
|
570
583
|
const qualifiedName = this.normalizeShapeId(shapeId);
|
|
571
|
-
const
|
|
572
|
-
|
|
584
|
+
for (const r of [this, TypeRegistry.for(qualifiedName.split("#")[0])]) {
|
|
585
|
+
r.schemas.set(qualifiedName, schema);
|
|
586
|
+
}
|
|
573
587
|
}
|
|
574
588
|
getSchema(shapeId) {
|
|
575
589
|
const id = this.normalizeShapeId(shapeId);
|
|
@@ -580,12 +594,17 @@ class TypeRegistry {
|
|
|
580
594
|
}
|
|
581
595
|
registerError(es, ctor) {
|
|
582
596
|
const $error = es;
|
|
583
|
-
const
|
|
584
|
-
|
|
585
|
-
|
|
597
|
+
const ns = $error[1];
|
|
598
|
+
for (const r of [this, TypeRegistry.for(ns)]) {
|
|
599
|
+
r.schemas.set(ns + "#" + $error[2], $error);
|
|
600
|
+
r.exceptions.set($error, ctor);
|
|
601
|
+
}
|
|
586
602
|
}
|
|
587
603
|
getErrorCtor(es) {
|
|
588
604
|
const $error = es;
|
|
605
|
+
if (this.exceptions.has($error)) {
|
|
606
|
+
return this.exceptions.get($error);
|
|
607
|
+
}
|
|
589
608
|
const registry = TypeRegistry.for($error[1]);
|
|
590
609
|
return registry.exceptions.get($error);
|
|
591
610
|
}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { RpcProtocol } from "@smithy/core/protocols";
|
|
2
|
-
import {
|
|
2
|
+
import { TypeRegistry } from "@smithy/core/schema";
|
|
3
|
+
import { deref, NormalizedSchema } from "@smithy/core/schema";
|
|
3
4
|
import { getSmithyContext } from "@smithy/util-middleware";
|
|
4
5
|
import { CborCodec } from "./CborCodec";
|
|
5
6
|
import { loadSmithyRpcV2CborErrorCode } from "./parseCborBody";
|
|
@@ -7,8 +8,8 @@ export class SmithyRpcV2CborProtocol extends RpcProtocol {
|
|
|
7
8
|
codec = new CborCodec();
|
|
8
9
|
serializer = this.codec.createSerializer();
|
|
9
10
|
deserializer = this.codec.createDeserializer();
|
|
10
|
-
constructor({ defaultNamespace }) {
|
|
11
|
-
super({ defaultNamespace });
|
|
11
|
+
constructor({ defaultNamespace, errorTypeRegistries, }) {
|
|
12
|
+
super({ defaultNamespace, errorTypeRegistries });
|
|
12
13
|
}
|
|
13
14
|
getShapeId() {
|
|
14
15
|
return "smithy.protocols#rpcv2Cbor";
|
|
@@ -52,15 +53,17 @@ export class SmithyRpcV2CborProtocol extends RpcProtocol {
|
|
|
52
53
|
}
|
|
53
54
|
async handleError(operationSchema, context, response, dataObject, metadata) {
|
|
54
55
|
const errorName = loadSmithyRpcV2CborErrorCode(response, dataObject) ?? "Unknown";
|
|
55
|
-
let namespace = this.options.defaultNamespace;
|
|
56
|
-
if (errorName.includes("#")) {
|
|
57
|
-
[namespace] = errorName.split("#");
|
|
58
|
-
}
|
|
59
56
|
const errorMetadata = {
|
|
60
57
|
$metadata: metadata,
|
|
61
58
|
$fault: response.statusCode <= 500 ? "client" : "server",
|
|
62
59
|
};
|
|
63
|
-
|
|
60
|
+
let namespace = this.options.defaultNamespace;
|
|
61
|
+
if (errorName.includes("#")) {
|
|
62
|
+
[namespace] = errorName.split("#");
|
|
63
|
+
}
|
|
64
|
+
const registry = this.compositeErrorRegistry;
|
|
65
|
+
const nsRegistry = TypeRegistry.for(namespace);
|
|
66
|
+
registry.copyFrom(nsRegistry);
|
|
64
67
|
let errorSchema;
|
|
65
68
|
try {
|
|
66
69
|
errorSchema = registry.getSchema(errorName);
|
|
@@ -69,10 +72,11 @@ export class SmithyRpcV2CborProtocol extends RpcProtocol {
|
|
|
69
72
|
if (dataObject.Message) {
|
|
70
73
|
dataObject.message = dataObject.Message;
|
|
71
74
|
}
|
|
72
|
-
const
|
|
73
|
-
|
|
75
|
+
const syntheticRegistry = TypeRegistry.for("smithy.ts.sdk.synthetic." + namespace);
|
|
76
|
+
registry.copyFrom(syntheticRegistry);
|
|
77
|
+
const baseExceptionSchema = registry.getBaseException();
|
|
74
78
|
if (baseExceptionSchema) {
|
|
75
|
-
const ErrorCtor =
|
|
79
|
+
const ErrorCtor = registry.getErrorCtor(baseExceptionSchema);
|
|
76
80
|
throw Object.assign(new ErrorCtor({ name: errorName }), errorMetadata, dataObject);
|
|
77
81
|
}
|
|
78
82
|
throw Object.assign(new Error(errorName), errorMetadata, dataObject);
|
|
@@ -1,11 +1,16 @@
|
|
|
1
|
-
import { NormalizedSchema, translateTraits } from "@smithy/core/schema";
|
|
1
|
+
import { NormalizedSchema, translateTraits, TypeRegistry } from "@smithy/core/schema";
|
|
2
2
|
import { HttpRequest, HttpResponse } from "@smithy/protocol-http";
|
|
3
3
|
import { SerdeContext } from "./SerdeContext";
|
|
4
4
|
export class HttpProtocol extends SerdeContext {
|
|
5
5
|
options;
|
|
6
|
+
compositeErrorRegistry;
|
|
6
7
|
constructor(options) {
|
|
7
8
|
super();
|
|
8
9
|
this.options = options;
|
|
10
|
+
this.compositeErrorRegistry = TypeRegistry.for(options.defaultNamespace);
|
|
11
|
+
for (const etr of options.errorTypeRegistries ?? []) {
|
|
12
|
+
this.compositeErrorRegistry.copyFrom(etr);
|
|
13
|
+
}
|
|
9
14
|
}
|
|
10
15
|
getRequestType() {
|
|
11
16
|
return HttpRequest;
|