AWS JDK 2创建队列并订阅主题

时间:2019-12-05 17:33:33

标签: amazon-sqs amazon-sns aws-java-sdk-2.x

我正在使用具有SNS / SQS的AWS进行工作,并且我正尝试在SQS主题上创建SNS订阅。我正在使用AWS Java SDK2。我可以正确创建主题和队列,我面临的问题是在主题和队列之间创建订阅时出现错误。以下是我的代码段:

public SubscribeResponse subscribeQueue(String topicArn, String queueUrl){
    SubscribeRequest subscribeRequest = SubscribeRequest.builder()
            .topicArn(topicArn)
            .protocol("sqs")
            .endpoint(queueUrl)
            .returnSubscriptionArn(true).build();
    return getAmazonSNSClient().subscribe(subscribeRequest);
}

以下是我得到的错误:

{ "message": "Internal Server Error: software.amazon.awssdk.services.sns.model.InvalidParameterException: Invalid parameter: SQS endpoint ARN (Service: Sns, Status Code: 400, Request ID: dabaeeb3-e77a-5873-861e-b8d0a7bf5c7f)" }

有人可以指出我在做什么吗?

1 个答案:

答案 0 :(得分:1)

使用AWS Java SDK v1,您可以使用方法Topics.subscribeQueue将SQS队列订阅到SNS主题,其中 endpoint 对象将 < em> queueUrl 如下所示:

Topics.subscribeQueue(sns, sqs, myTopicARN, queueURL);

在AWS Java SDK v2中,您可以使用方法SunscribeRequest.builder将SQS队列订阅到SNS主题,其中 endpoint 对象将加入 queueArn

下面是一个示例Java代码段,我已经对其进行了测试,以成功将SQS队列订阅到SNS主题:

  public static void subscribeSQStoSNS(String queueUrl, String topicArn, SqsClient sqsClient, SnsClient snsClient) {


   /* with AWS Java SDK v1, you could subscribe an SQS queue to an SNS topic by
     * calling "Topics.subscribeQueue"()[1] where the endpoint object takes in queueUrl.
     * 
     * with AWS Java SDK v2, to subscribe an SQS queue to an SNS topic
     * one needs to use "SunscribeRequest.builder"() [2] where the endpoint object takes in queueArn.
     * 
     * [1] https://docs.aws.amazon.com/AWSJavaSDK/latest/javadoc/com/amazonaws/services/sns/util/Topics.html
     * [2] https://sdk.amazonaws.com/java/api/2.0.0-preview-11/software/amazon/awssdk/services/sns/model/SubscribeRequest.Builder.html
     *
     */

  // Step 1: We call "get-queue-attributes" API to retrieve the SQS queue Arn

  GetQueueAttributesRequest QueueAttributesRequest = GetQueueAttributesRequest.builder()
            .queueUrl(queueUrl)
            .attributeNamesWithStrings("All")
            .build();

   GetQueueAttributesResponse QueueAttributesResult = sqsClient.getQueueAttributes(QueueAttributesRequest);
    Map<String, String> sqsAttributeMap = QueueAttributesResult.attributesAsStrings();

  System.out.println("\n\n"+ QueueAttributesRequest);
    String queueArn = sqsAttributeMap.get("QueueArn");
    //System.out.println("\n\n"+ queueArn);


  // Step 2: We call "SubscribeRequest.builder" to subscribe the SQS queue to the SNS topic

   SubscribeRequest Qrequest = SubscribeRequest.builder()
            .protocol("sqs")
            .endpoint(queueArn)
            .returnSubscriptionArn(true)
            .topicArn(topicArn)
            .build();

   SubscribeResponse Qresponse = snsClient.subscribe(Qrequest);
    System.out.println("\n\nCreated Subscription ARN: " + Qresponse.subscriptionArn()+ " " + " and StatusCode : " + Qresponse.sdkHttpResponse().statusCode());}

注意:

1。“ Topics.subscribeQueue”()方法自动为订阅创建策略,以允许主题将消息发送到队列。

2。“ SunscribeRequest.builder”()方法不会自动为订阅添加策略( SQS队列),因此以允许SNS主题将消息发送到队列。因此,通过将以下示例策略添加到SQS队列中,如下图所示,也按照here所述,确保已授予SNS主题发布到SQS队列的权限:

    {
      "Statement": [{
        "Effect":"Allow",
        "Principal":"*",
        "Action":"sqs:SendMessage",
        "Resource":"arn:aws:sqs:us-east-2:123456789012:MyQueue",
        "Condition":{
          "ArnEquals":{
            "aws:SourceArn":"arn:aws:sns:us-east-2:123456789012:MyTopic"
          }
        }
      }]
    }

您可以在github中找到完整的代码:https://gist.github.com/syumaK/79c946ed4e618c1f871f50ecb1819a0f

希望这会有所帮助。