无法从本地假 PubSub 服务器执行请求订阅的 GET 请求

Posted

技术标签:

【中文标题】无法从本地假 PubSub 服务器执行请求订阅的 GET 请求【英文标题】:Can't do GET request for pull subscription from local fake PubSub Server 【发布时间】:2021-05-10 20:54:43 【问题描述】:

TL;DR:发送 GET 请求以从 Pub/Sub 服务器进行拉取订阅的正确方法是什么。使用的正确 URL 是什么?

我正在使用 gcloud beta emulators pubsub start 运行本地 Google Pub/Sub 伪造,我已经能够使用我编写的 ruby​​ 脚本成功发布到它,但是我无法进行请求订阅。

我的目标是仅使用 GET 请求而不是脚本来完成此操作。我发现的问题之一是documentation 中的tons 与客户端或pull subscriptions 或gcloud 一起使用,但很少通过URL 访问服务器。也许我误解了什么是可能的 - 但我想使用 ruby​​ 客户端将消息发布到 pub/sub,然后使用 Postman 向 pub/sub 服务器发出 GET 请求以检索消息。

我很确定问题在于我如何发出获取请求,但我已经复制了下面的所有其他内容

Ruby 发布代码

require "google/cloud/pubsub"
require 'json'

class Publisher
    def publish(event)

        puts "1==============>>>>>>>> publishing..."
        pubsub = Google::Cloud::PubSub.new(
            project_id: "grpc-demo-proj",
            emulator_host: "localhost:8085"
        )

        topic_id = "event_topic"
        topic = pubsub.topic topic_id

        begin
            topic.publish_async "receive_event#event",
                    event: JSON.generate(event) do |result|
                raise "Failed to publish the message." unless result.succeeded?
                puts "2==============>>>>>>>>  Message published asynchronously."
            end

            # Stop the async_publisher to send all queued messages immediately.
            topic.async_publisher.stop.wait!
            rescue StandardError => e
            puts "3==============>>>>>>>> Received error while publishing: #e.message"
        end
    end
end

这似乎有效,因为我得到了

1==============>>>>>>>> publishing...
DEBUG  GRPC : calling localhost:8085:/google.pubsub.v1.Publisher/GetTopic
DEBUG  GRPC : calling localhost:8085:/google.pubsub.v1.Publisher/Publish
2==============>>>>>>>>  Message published asynchronously.

在我的终端中。

我还使用以下 shell 脚本运行 Pub/Sub 服务器。

#!/bin/bash

# Kill the existing process if it's already running
if [ "$(lsof -i:8085)" ]; then
  kill $(lsof -t -i:8085)
fi

# Kick off the new process
gcloud beta emulators pubsub start --project=grpc-demo-proj

# Connect to environment variables
$(gcloud beta emulators pubsub env-init)

PubSub 设置脚本

#!/bin/bash

# Wait for the pubsub emulator to boot up
sleep 7
while [[ ! "$(lsof -i:8085)" ]]
do
  echo '#===> PUBSUB EMULATOR SETUP: Waiting for PubSub Emulator to start...'
  sleep 3
done

  # Create topics
curl --header "Content-Type: application/json" \
  --request PUT \
  http://localhost:8085/v1/projects/grpc-demo-proj/topics/event_topic


# Create test subscriptions for each topic
curl --header "Content-Type: application/json" \
  --request PUT \
  --data '"topic": "projects/grpc-demo-proj/topics/event_topic"' \
  http://localhost:8085/v1/projects/grpc-demo-proj/subscriptions/event_topic.test_sub1

再次。这些似乎运作良好。

我遇到的麻烦... 正在使用 GET 请求(来自 PostMan,或仅在浏览器的 URL 栏中)从 pub/sub 服务器进行拉取订阅

http://localhost:8085/v1/projects/grpc-demo-proj/subscriptions/event_topic.test_sub1:pull

返回


    "error": 
        "code": 400,
        "message": "Invalid [subscriptions] name: (name=projects/grpc-demo-proj/subscriptions/event_topic.test_sub1:pull)",
        "status": "INVALID_ARGUMENT"
    

但是订阅名是有效的,如

http://localhost:8085/v1/projects/grpc-demo-proj/subscriptions/event_topic.test_sub1

返回


    "name": "projects/grpc-demo-proj/subscriptions/event_topic.test_sub1",
    "topic": "projects/grpc-demo-proj/topics/event_topic",
    "pushConfig": ,
    "ackDeadlineSeconds": 10,
    "messageRetentionDuration": "604800s"

这似乎可以确认服务器正在运行,并且主题和订阅已成功创建。

虽然-不是-我正在寻找的解决方案,但我尝试在命令行中使用 gcloud:

bgc@jadzia:~$ gcloud beta pubsub subscriptions pull test_sub1
ERROR: (gcloud.beta.pubsub.subscriptions.pull) NOT_FOUND: Resource not found (resource=test_sub1).

尽管其他消息来源似乎证实此订阅确实存在。

虽然可能是 Ruby 错误地说它成功发布了消息,或者服务器出了问题。我怀疑我只是没有正确执行 GET 请求。我已经尝试了上述 GET 请求的几种变体,但不会在这里全部列出。

那么,在不使用脚本的情况下,如何从发布/订阅服务器返回消息?理想情况下,我可以将 GET 请求的 URL 插入 PostMan,但基于命令行的解决方案也可以在这里工作。

【问题讨论】:

诚然(道歉)只是简要地扫描了您的问题。下面是创建订阅的 REST (PUT) 方法:cloud.google.com/pubsub/docs/reference/rest/v1/… 这里是 Goggle 的(推荐的)用于 Pub/Sub 的 Ruby SDK:cloud.google.com/pubsub/docs/… 我已经创建了一个订阅,我无法尝试从一个订阅中获取消息。我也不打算为此使用 Ruby SDK,我只是将它用于发布。 也就是说,您的第一个链接中的差异页面看起来很有希望:cloud.google.com/pubsub/docs/reference/rest/v1/… - 我尝试了它提到的 HTTP 请求。它与我在问题中尝试的 URL 相同,但作为 POST 请求,而不是 GET 请求。不幸的是,它仍然不起作用,给出 404 和“订阅不存在”,即使再次在设置脚本中运行 curl 命令以再次创建订阅也会给出“ALREADY_EXISTS:订阅已存在”。所以它不存在,但它也存在。薛定谔的 Pub/Sub 订阅? 【参考方案1】:

我确实使用您发布的所有脚本复制了您本地的假 Pub/sub 服务器。正如您评论的那样,我使用 POST 而不是 GET 并得到了响应。 Pub/Sub subscriptions pull reference

发布https://pubsub.googleapis.com/v1/subscription:pull

订阅请求的 POST 请求:

curl --header "Content-Type: application/json" \
--request POST \
--data "
  "maxMessages": "1"
" \
http://localhost:8085/v1/projects/my-project/subscriptions/event_topic.test_sub1:pull

订阅拉取的输出:

Pubsub Message 以 64 进制编码。请注意,我在 Google Cloud Shell 中运行了所有内容(创建发布订阅服务器、主题、订阅者、发布消息、拉取消息)。

编辑 1:

正如布赖恩所说,这是对他有用的请求。这也适用于我的测试!

curl --header "Content-Type: application/json" \
--request POST \
localhost:8085/v1/projects/my-prject/subscriptions/event_topic.test_sub1:pull?maxMessages=5

输出:

【讨论】:

谢谢,看起来我在做正确的事情,至少在理论上是这样。我尝试了 curl 命令而不是使用 PostMan,但得到了相同的结果“订阅不存在”。 我不确定如何确认我的消息是以 base 64 编码的,但我不确定这是否会导致此错误。我想如果是这样的话,它就不会发布了。 好的。解决了!订阅问题是因为我需要在末尾指定 topic.subscription:pull,而不仅仅是 subscription.pull。此外 - 有一个必需的参数 maxMessages,它并不那么明显。工作请求是:POST localhost:8085/v1/projects/grpc-demo-proj/subscriptions/event_topic.test_sub1:pull?maxMessages=5 如果你想相应地编辑你的答案,我很乐意接受它 @BrianC 当然,让我也测试一下。因为它是编写 POST 请求的一种更有效的方式!

以上是关于无法从本地假 PubSub 服务器执行请求订阅的 GET 请求的主要内容,如果未能解决你的问题,请参考以下文章

Pubsub.pull 请求无法正常工作 - 去吧

Google PubSub 订阅无法从 StatusCode.UNAVAILABLE [code=8a75] 错误中恢复

Google Pubsub - 接收推送订阅的传递尝试

为啥使用 PUBSUB 订阅时无法 PING?

xmpp pubsub service(XEP-0060) 收到订阅请求后能否创建节点?

如何通过 terraform 使用服务帐户创建谷歌云 pubsub 订阅?