Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 41a5f94

Browse files
committedFeb 6, 2025··
fix(sns): for SSE topics, add KMS permissions in grantPublish
1 parent bc82f57 commit 41a5f94

File tree

10 files changed

+241
-30
lines changed

10 files changed

+241
-30
lines changed
 

‎packages/@aws-cdk-testing/framework-integ/test/aws-sns/test/integ.sns.js.snapshot/SNSInteg.assets.json

+3-3
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎packages/@aws-cdk-testing/framework-integ/test/aws-sns/test/integ.sns.js.snapshot/SNSInteg.template.json

+43-7
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,17 @@
2727
}
2828
},
2929
"Resource": "*"
30+
},
31+
{
32+
"Action": [
33+
"kms:Decrypt",
34+
"kms:GenerateDataKey*"
35+
],
36+
"Effect": "Allow",
37+
"Principal": {
38+
"Service": "s3.amazonaws.com"
39+
},
40+
"Resource": "*"
3041
}
3142
],
3243
"Version": "2012-10-17"
@@ -132,12 +143,6 @@
132143
"Type": "AWS::SNS::Topic",
133144
"Properties": {
134145
"DisplayName": "fooDisplayName2",
135-
"KmsMasterKeyId": {
136-
"Fn::GetAtt": [
137-
"CustomKey1E6D0D07",
138-
"Arn"
139-
]
140-
},
141146
"TopicName": "fooTopic2"
142147
}
143148
},
@@ -166,8 +171,26 @@
166171
{
167172
"Action": "sns:Publish",
168173
"Effect": "Allow",
174+
"Resource": [
175+
{
176+
"Ref": "MyTopic288CE2107"
177+
},
178+
{
179+
"Ref": "MyTopic3134CFDFB"
180+
}
181+
]
182+
},
183+
{
184+
"Action": [
185+
"kms:Decrypt",
186+
"kms:GenerateDataKey*"
187+
],
188+
"Effect": "Allow",
169189
"Resource": {
170-
"Ref": "MyTopic288CE2107"
190+
"Fn::GetAtt": [
191+
"CustomKey1E6D0D07",
192+
"Arn"
193+
]
171194
}
172195
}
173196
],
@@ -180,6 +203,19 @@
180203
}
181204
]
182205
}
206+
},
207+
"MyTopic3134CFDFB": {
208+
"Type": "AWS::SNS::Topic",
209+
"Properties": {
210+
"DisplayName": "fooDisplayName3",
211+
"KmsMasterKeyId": {
212+
"Fn::GetAtt": [
213+
"CustomKey1E6D0D07",
214+
"Arn"
215+
]
216+
},
217+
"TopicName": "fooTopic3"
218+
}
183219
}
184220
},
185221
"Parameters": {

‎packages/@aws-cdk-testing/framework-integ/test/aws-sns/test/integ.sns.js.snapshot/cdk.out

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎packages/@aws-cdk-testing/framework-integ/test/aws-sns/test/integ.sns.js.snapshot/integ.json

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎packages/@aws-cdk-testing/framework-integ/test/aws-sns/test/integ.sns.js.snapshot/manifest.json

+8-2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎packages/@aws-cdk-testing/framework-integ/test/aws-sns/test/integ.sns.js.snapshot/tree.json

+85-11
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎packages/@aws-cdk-testing/framework-integ/test/aws-sns/test/integ.sns.ts

+21-3
Original file line numberDiff line numberDiff line change
@@ -62,14 +62,32 @@ class SNSInteg extends Stack {
6262
const topic2 = new Topic(this, 'MyTopic2', {
6363
topicName: 'fooTopic2',
6464
displayName: 'fooDisplayName2',
65-
masterKey: key,
6665
});
67-
const importedTopic = Topic.fromTopicArn(this, 'ImportedTopic', topic2.topicArn);
66+
const importedTopic2 = Topic.fromTopicArn(this, 'ImportedTopic2', topic2.topicArn);
6867

6968
const publishRole = new Role(this, 'PublishRole', {
7069
assumedBy: new ServicePrincipal('s3.amazonaws.com'),
7170
});
72-
importedTopic.grantPublish(publishRole);
71+
importedTopic2.grantPublish(publishRole);
72+
73+
// Can import encrypted topic by attributes
74+
const topic3 = new Topic(this, 'MyTopic3', {
75+
topicName: 'fooTopic3',
76+
displayName: 'fooDisplayName3',
77+
masterKey: key,
78+
});
79+
const importedTopic3 = Topic.fromTopicAttributes(this, 'ImportedTopic3', {
80+
topicArn: topic3.topicArn,
81+
keyArn: key.keyArn,
82+
});
83+
importedTopic3.grantPublish(publishRole);
84+
85+
// Can reference KMS key after creation
86+
topic3.masterKey!.addToResourcePolicy(new PolicyStatement({
87+
principals: [new ServicePrincipal('s3.amazonaws.com')],
88+
actions: ['kms:GenerateDataKey*', 'kms:Decrypt'],
89+
resources: ['*'],
90+
}));
7391
}
7492
}
7593

‎packages/aws-cdk-lib/aws-sns/lib/topic-base.ts

+19-1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import { ITopicSubscription } from './subscriber';
55
import { Subscription } from './subscription';
66
import * as notifications from '../../aws-codestarnotifications';
77
import * as iam from '../../aws-iam';
8+
import { IKey } from '../../aws-kms';
89
import { IResource, Resource, ResourceProps, Token } from '../../core';
910
import { ValidationError } from '../../core/lib/errors';
1011

@@ -26,6 +27,17 @@ export interface ITopic extends IResource, notifications.INotificationRuleTarget
2627
*/
2728
readonly topicName: string;
2829

30+
/**
31+
* A KMS Key, either managed by this CDK app, or imported.
32+
*
33+
* This property applies only to server-side encryption.
34+
*
35+
* @see https://docs.aws.amazon.com/sns/latest/dg/sns-server-side-encryption.html
36+
*
37+
* @default None
38+
*/
39+
readonly masterKey?: IKey;
40+
2941
/**
3042
* Enables content-based deduplication for FIFO topics.
3143
*
@@ -73,6 +85,8 @@ export abstract class TopicBase extends Resource implements ITopic {
7385

7486
public abstract readonly topicName: string;
7587

88+
public abstract readonly masterKey?: IKey;
89+
7690
public abstract readonly fifo: boolean;
7791

7892
public abstract readonly contentBasedDeduplication: boolean;
@@ -191,12 +205,16 @@ export abstract class TopicBase extends Resource implements ITopic {
191205
* Grant topic publishing permissions to the given identity
192206
*/
193207
public grantPublish(grantee: iam.IGrantable) {
194-
return iam.Grant.addToPrincipalOrResource({
208+
const ret = iam.Grant.addToPrincipalOrResource({
195209
grantee,
196210
actions: ['sns:Publish'],
197211
resourceArns: [this.topicArn],
198212
resource: this,
199213
});
214+
if (this.masterKey) {
215+
this.masterKey.grant(grantee, 'kms:Decrypt', 'kms:GenerateDataKey*');
216+
}
217+
return ret;
200218
}
201219

202220
/**

‎packages/aws-cdk-lib/aws-sns/lib/topic.ts

+13-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ import { Construct } from 'constructs';
22
import { CfnTopic } from './sns.generated';
33
import { ITopic, TopicBase } from './topic-base';
44
import { IRole } from '../../aws-iam';
5-
import { IKey } from '../../aws-kms';
5+
import { IKey, Key } from '../../aws-kms';
66
import { ArnFormat, Lazy, Names, Stack, Token } from '../../core';
77
import { ValidationError } from '../../core/lib/errors';
88
import { addConstructMetadata, MethodMetadata } from '../../core/lib/metadata-resource';
@@ -219,6 +219,13 @@ export interface TopicAttributes {
219219
*/
220220
readonly topicArn: string;
221221

222+
/**
223+
* KMS encryption key, if this topic is server-side encrypted by a KMS key.
224+
*
225+
* @default - None
226+
*/
227+
readonly keyArn?: string;
228+
222229
/**
223230
* Whether content-based deduplication is enabled.
224231
* Only applicable for FIFO topics.
@@ -261,6 +268,9 @@ export class Topic extends TopicBase {
261268
class Import extends TopicBase {
262269
public readonly topicArn = attrs.topicArn;
263270
public readonly topicName = topicName;
271+
public readonly masterKey = attrs.keyArn
272+
? Key.fromKeyArn(this, 'Key', attrs.keyArn)
273+
: undefined;
264274
public readonly fifo = fifo;
265275
public readonly contentBasedDeduplication = attrs.contentBasedDeduplication || false;
266276
protected autoCreatePolicy: boolean = false;
@@ -273,6 +283,7 @@ export class Topic extends TopicBase {
273283

274284
public readonly topicArn: string;
275285
public readonly topicName: string;
286+
public readonly masterKey?: IKey;
276287
public readonly contentBasedDeduplication: boolean;
277288
public readonly fifo: boolean;
278289

@@ -357,6 +368,7 @@ export class Topic extends TopicBase {
357368
resource: this.physicalName,
358369
});
359370
this.topicName = this.getResourceNameAttribute(resource.attrTopicName);
371+
this.masterKey = props.masterKey;
360372
this.fifo = props.fifo || false;
361373
this.contentBasedDeduplication = props.contentBasedDeduplication || false;
362374

‎packages/aws-cdk-lib/aws-sns/test/sns.test.ts

+47
Original file line numberDiff line numberDiff line change
@@ -304,6 +304,38 @@ describe('Topic', () => {
304304
});
305305
});
306306

307+
test('give publishing permissions with masterKey', () => {
308+
// GIVEN
309+
const stack = new cdk.Stack();
310+
const key = new kms.Key(stack, 'CustomKey');
311+
const topic = new sns.Topic(stack, 'Topic', { masterKey: key });
312+
const user = new iam.User(stack, 'User');
313+
314+
// WHEN
315+
topic.grantPublish(user);
316+
317+
// THEN
318+
Template.fromStack(stack).hasResourceProperties('AWS::IAM::Policy', {
319+
'PolicyDocument': {
320+
Version: '2012-10-17',
321+
'Statement': [
322+
{
323+
'Action': 'sns:Publish',
324+
'Effect': 'Allow',
325+
'Resource': stack.resolve(topic.topicArn),
326+
},
327+
{
328+
'Action': ['kms:Decrypt', 'kms:GenerateDataKey*'],
329+
'Effect': 'Allow',
330+
'Resource': {
331+
'Fn::GetAtt': ['CustomKey1E6D0D07', 'Arn'],
332+
},
333+
},
334+
],
335+
},
336+
});
337+
});
338+
307339
test('give subscribing permissions', () => {
308340
// GIVEN
309341
const stack = new cdk.Stack();
@@ -544,6 +576,21 @@ describe('Topic', () => {
544576
})).toThrow(/Cannot import topic; contentBasedDeduplication is only available for FIFO SNS topics./);
545577
});
546578

579+
test('fromTopicAttributes keyArn', () => {
580+
// GIVEN
581+
const stack = new cdk.Stack();
582+
const keyArn = 'arn:aws:kms:us-east-1:123456789012:key/1234abcd-12ab-34cd-56ef-1234567890ab';
583+
584+
// WHEN
585+
const imported = sns.Topic.fromTopicAttributes(stack, 'Imported', {
586+
topicArn: 'arn:aws:sns:*:123456789012:mytopic',
587+
keyArn,
588+
});
589+
590+
// THEN
591+
expect(imported.masterKey?.keyArn).toEqual(keyArn);
592+
});
593+
547594
test('sets account for imported topic env', () => {
548595
// GIVEN
549596
const stack = new cdk.Stack();

0 commit comments

Comments
 (0)
Please sign in to comment.