From 5c4a9e5d6b8801420019638b2c0423c3b888351c Mon Sep 17 00:00:00 2001 From: Rico Huijbers Date: Thu, 31 Jan 2019 14:35:55 +0100 Subject: [PATCH] fix(sns): create subscription object under subscriber Create the SNS Subscription object under the subscriber instead of the subscribee, avoiding cyclic stack dependencies if those objects would be in different stacks. Fixes #1643, fixes #1534. --- packages/@aws-cdk/aws-sns/lib/topic-ref.ts | 20 ++++++--- ...teg.sns-bucket-notifications.expected.json | 2 +- .../integ.sns-event-rule-target.expected.json | 34 +++++++------- .../test/integ.sns-lambda.expected.json | 45 ++++++++++++------- .../test/integ.sns-sqs.lit.expected.json | 8 ++-- packages/@aws-cdk/aws-sns/test/test.sns.ts | 8 ++-- 6 files changed, 68 insertions(+), 49 deletions(-) diff --git a/packages/@aws-cdk/aws-sns/lib/topic-ref.ts b/packages/@aws-cdk/aws-sns/lib/topic-ref.ts index 4a562651af442..8bff777a2d442 100644 --- a/packages/@aws-cdk/aws-sns/lib/topic-ref.ts +++ b/packages/@aws-cdk/aws-sns/lib/topic-ref.ts @@ -169,14 +169,18 @@ export abstract class TopicBase extends cdk.Construct implements ITopic { * @param queue The target queue */ public subscribeQueue(queue: sqs.IQueue): Subscription { - const subscriptionName = queue.node.id + 'Subscription'; - if (this.node.tryFindChild(subscriptionName)) { + if (!cdk.Construct.isConstruct(queue)) { + throw new Error(`The supplied Queue object must be an instance of Construct`); + } + + const subscriptionName = this.node.id + 'Subscription'; + if (queue.node.tryFindChild(subscriptionName)) { throw new Error(`A subscription between the topic ${this.node.id} and the queue ${queue.node.id} already exists`); } // we use the queue name as the subscription's. there's no meaning to subscribing // the same queue twice on the same topic. - const sub = new Subscription(this, subscriptionName, { + const sub = new Subscription(queue, subscriptionName, { topic: this, endpoint: queue.queueArn, protocol: SubscriptionProtocol.Sqs @@ -203,13 +207,17 @@ export abstract class TopicBase extends cdk.Construct implements ITopic { * @param lambdaFunction The Lambda function to invoke */ public subscribeLambda(lambdaFunction: lambda.IFunction): Subscription { - const subscriptionName = lambdaFunction.id + 'Subscription'; + if (!cdk.Construct.isConstruct(lambdaFunction)) { + throw new Error(`The supplied lambda Function object must be an instance of Construct`); + } + + const subscriptionName = this.node.id + 'Subscription'; - if (this.node.tryFindChild(subscriptionName)) { + if (lambdaFunction.node.tryFindChild(subscriptionName)) { throw new Error(`A subscription between the topic ${this.node.id} and the lambda ${lambdaFunction.id} already exists`); } - const sub = new Subscription(this, subscriptionName, { + const sub = new Subscription(lambdaFunction, subscriptionName, { topic: this, endpoint: lambdaFunction.functionArn, protocol: SubscriptionProtocol.Lambda diff --git a/packages/@aws-cdk/aws-sns/test/integ.sns-bucket-notifications.expected.json b/packages/@aws-cdk/aws-sns/test/integ.sns-bucket-notifications.expected.json index 6e70240782358..b6d814cc219f7 100644 --- a/packages/@aws-cdk/aws-sns/test/integ.sns-bucket-notifications.expected.json +++ b/packages/@aws-cdk/aws-sns/test/integ.sns-bucket-notifications.expected.json @@ -204,4 +204,4 @@ } } } -} +} \ No newline at end of file diff --git a/packages/@aws-cdk/aws-sns/test/integ.sns-event-rule-target.expected.json b/packages/@aws-cdk/aws-sns/test/integ.sns-event-rule-target.expected.json index afa32e0b0348a..39a497da7ea9a 100644 --- a/packages/@aws-cdk/aws-sns/test/integ.sns-event-rule-target.expected.json +++ b/packages/@aws-cdk/aws-sns/test/integ.sns-event-rule-target.expected.json @@ -3,28 +3,12 @@ "MyTopic86869434": { "Type": "AWS::SNS::Topic" }, - "MyTopicMyQueueSubscription3245B11E": { - "Type": "AWS::SNS::Subscription", - "Properties": { - "Endpoint": { - "Fn::GetAtt": [ - "MyQueueE6CA6235", - "Arn" - ] - }, - "Protocol": "sqs", - "TopicArn": { - "Ref": "MyTopic86869434" - } - } - }, "MyTopicPolicy12A5EC17": { "Type": "AWS::SNS::TopicPolicy", "Properties": { "PolicyDocument": { "Statement": [ { - "Sid": "0", "Action": "sns:Publish", "Effect": "Allow", "Principal": { @@ -32,7 +16,8 @@ }, "Resource": { "Ref": "MyTopic86869434" - } + }, + "Sid": "0" } ], "Version": "2012-10-17" @@ -62,6 +47,21 @@ "MyQueueE6CA6235": { "Type": "AWS::SQS::Queue" }, + "MyQueueMyTopicSubscriptionEB66AD1B": { + "Type": "AWS::SNS::Subscription", + "Properties": { + "Endpoint": { + "Fn::GetAtt": [ + "MyQueueE6CA6235", + "Arn" + ] + }, + "Protocol": "sqs", + "TopicArn": { + "Ref": "MyTopic86869434" + } + } + }, "MyQueuePolicy6BBEDDAC": { "Type": "AWS::SQS::QueuePolicy", "Properties": { diff --git a/packages/@aws-cdk/aws-sns/test/integ.sns-lambda.expected.json b/packages/@aws-cdk/aws-sns/test/integ.sns-lambda.expected.json index 800d7778834ae..c4ee7b98f7edf 100644 --- a/packages/@aws-cdk/aws-sns/test/integ.sns-lambda.expected.json +++ b/packages/@aws-cdk/aws-sns/test/integ.sns-lambda.expected.json @@ -3,21 +3,6 @@ "MyTopic86869434": { "Type": "AWS::SNS::Topic" }, - "MyTopicEchoSubscription021036AD": { - "Type": "AWS::SNS::Subscription", - "Properties": { - "Endpoint": { - "Fn::GetAtt": [ - "Echo11F3FB29", - "Arn" - ] - }, - "Protocol": "lambda", - "TopicArn": { - "Ref": "MyTopic86869434" - } - } - }, "EchoServiceRoleBE28060B": { "Type": "AWS::IAM::Role", "Properties": { @@ -34,7 +19,18 @@ "Version": "2012-10-17" }, "ManagedPolicyArns": [ - {"Fn::Join":["",["arn:",{"Ref":"AWS::Partition"},":iam::aws:policy/service-role/AWSLambdaBasicExecutionRole"]]} + { + "Fn::Join": [ + "", + [ + "arn:", + { + "Ref": "AWS::Partition" + }, + ":iam::aws:policy/service-role/AWSLambdaBasicExecutionRole" + ] + ] + } ] } }, @@ -57,6 +53,21 @@ "EchoServiceRoleBE28060B" ] }, + "EchoMyTopicSubscriptionA634C6D7": { + "Type": "AWS::SNS::Subscription", + "Properties": { + "Endpoint": { + "Fn::GetAtt": [ + "Echo11F3FB29", + "Arn" + ] + }, + "Protocol": "lambda", + "TopicArn": { + "Ref": "MyTopic86869434" + } + } + }, "EchoMyTopicF6EBB45F": { "Type": "AWS::Lambda::Permission", "Properties": { @@ -71,4 +82,4 @@ } } } -} +} \ No newline at end of file diff --git a/packages/@aws-cdk/aws-sns/test/integ.sns-sqs.lit.expected.json b/packages/@aws-cdk/aws-sns/test/integ.sns-sqs.lit.expected.json index 40b6fbae330d0..e0dd645a519ea 100644 --- a/packages/@aws-cdk/aws-sns/test/integ.sns-sqs.lit.expected.json +++ b/packages/@aws-cdk/aws-sns/test/integ.sns-sqs.lit.expected.json @@ -3,7 +3,10 @@ "MyTopic86869434": { "Type": "AWS::SNS::Topic" }, - "MyTopicMyQueueSubscription3245B11E": { + "MyQueueE6CA6235": { + "Type": "AWS::SQS::Queue" + }, + "MyQueueMyTopicSubscriptionEB66AD1B": { "Type": "AWS::SNS::Subscription", "Properties": { "Endpoint": { @@ -18,9 +21,6 @@ } } }, - "MyQueueE6CA6235": { - "Type": "AWS::SQS::Queue" - }, "MyQueuePolicy6BBEDDAC": { "Type": "AWS::SQS::QueuePolicy", "Properties": { diff --git a/packages/@aws-cdk/aws-sns/test/test.sns.ts b/packages/@aws-cdk/aws-sns/test/test.sns.ts index 7d9b21872bd2a..3e697b6fb7f5a 100644 --- a/packages/@aws-cdk/aws-sns/test/test.sns.ts +++ b/packages/@aws-cdk/aws-sns/test/test.sns.ts @@ -149,7 +149,7 @@ export = { "TopicName": "topicName" } }, - "MyTopicMyQueueSubscription3245B11E": { + "MyQueueMyTopicSubscriptionEB66AD1B": { "Type": "AWS::SNS::Subscription", "Properties": { "Endpoint": { @@ -233,7 +233,7 @@ export = { "TopicName": "topicName" } }, - "MyTopicMyFuncSubscriptionEAF54A3F": { + "MyFuncMyTopicSubscription708A6535": { "Type": "AWS::SNS::Subscription", "Properties": { "Endpoint": { @@ -368,7 +368,7 @@ export = { "TopicName": "topicName" } }, - "MyTopicMyQueueSubscription3245B11E": { + "MyQueueMyTopicSubscriptionEB66AD1B": { "Type": "AWS::SNS::Subscription", "Properties": { "Endpoint": { @@ -383,7 +383,7 @@ export = { } } }, - "MyTopicMyFuncSubscriptionEAF54A3F": { + "MyFuncMyTopicSubscription708A6535": { "Type": "AWS::SNS::Subscription", "Properties": { "Endpoint": {