Google 发布/订阅订阅数据与应用不匹配
Posted
技术标签:
【中文标题】Google 发布/订阅订阅数据与应用不匹配【英文标题】:Google pub/sub subscription data doesn't match with the app 【发布时间】:2021-03-25 17:18:40 【问题描述】:我正在尝试在服务器上监听我的 Google Play 应用的订阅更改(新的和现有的)。这是我正在使用的代码。这使用了google/cloud-pubsub
composer 包:
$projectId = 'app-name';
$keyFile = file_get_contents(storage_path('app/app-name.json'));
$pubsub = new PubSubClient([
'projectId' => $projectId,
'keyFile' => json_decode($keyFile, true)
]);
$httpPostRequestBody = file_get_contents('php://input');
$requestData = json_decode($httpPostRequestBody, true);
info(json_encode($requestData));
$message = $pubsub->consume($requestData);
info(json_encode($message));
上面的代码有效,但问题是我获得的数据与我在应用程序端获得的数据不匹配。这是一个示例数据:
"message":
"data":"eyJ2ZXJ...",
"messageId":"16797998xxxxxxxxx",
"message_id":"1679799xxxxxxxxx",
"publishTime":"2020-12-15T02:09:23.27Z",
"publish_time":"2020-12-15T02:09:23.27Z"
,
"subscription":"projects\/app-name\/subscriptions\/test-subs"
如果你base64_decode()
数据,你会得到这样的东西:
version: "1.0",
packageName: "com.dev.app",
eventTimeMillis: "1607997631636",
subscriptionNotification:
version: "1.0",
notificationType: 4,
purchaseToken: "kmloa....",
subscriptionId: "app_subs1"
这是我希望purchaseToken
与我从客户端获得的相同的地方。
这是客户端的代码。我正在使用 Expo 应用内购买来实现订阅:
setPurchaseListener(async ( responseCode, results, errorCode ) =>
if (responseCode === IAPResponseCode.OK)
const orderId, purchaseToken, acknowledged = results[0];
if (!acknowledged)
await instance.post("/subscribe",
order_id: orderId,
order_token: purchaseToken,
data: JSON.stringify(results[0]),
);
finishTransactionAsync(results[0], true);
alert(
"You're now subscribed! You can now use the full functionality of the app."
);
);
我希望从results[0]
提取的purchaseToken
与Google 服务器在将通知推送到端点时返回的相同。但事实并非如此。
更新
我认为我的主要问题是我假设我需要的所有数据都来自 Google Pay,所以我只是依赖于用户在应用中订阅时 Google 发布的数据。
这实际上不是发布消息的人:
await instance.post("/subscribe")
它只是用购买令牌更新数据库。我可以用它来订阅用户,但不能保证请求是合法的。有人可以根据现有用户构建必要的凭据,他们几乎可以订阅而无需支付任何费用。另外,此方法不能用于保持用户订阅。所以数据确实必须来自谷歌。
根据下面的答案,我现在意识到您应该从自己的服务器触发发布?然后你听听吗?所以当我从客户端调用这个时:
await instance.post("/subscribe",
purchaseToken
);
我实际上需要像这样发布包含购买令牌的消息:
$pubsub = new PubSubClient([
'projectId' => $projectId,
]);
$topic = $pubsub->topic($topicName);
$message = [
'purchaseToken' => request('purchaseToken')
];
$topic->publish(['data' => $message]);
这就是你的意思吗?但是这种方法唯一的问题是如何验证购买令牌是否合法,以及如何在服务器中续订订阅?我有一个需要每月更新的字段,以便用户在服务器眼中保持“订阅”状态。
也许,我只是使用 pub/sub 使事情变得过于复杂。如果实际上有一个 API,我可以定期(使用 cron)从中提取数据,这允许我保持用户订阅数据的更新,那么这也是可以接受的答案。
【问题讨论】:
你有什么进展吗? 【参考方案1】:首先 - 由于 php PubSubClient,我对 php 和 pubsub 的体验非常糟糕。如果您的脚本只是在等待推送和检查消息,则删除 pubsub 包并用几行代码处理它。
示例:
$message = file_get_contents('php://input');
$message = json_decode($message, true);
if (is_array($message))
$message = (isset($message['message']) && isset($message['message']['data'])) ? base64_decode($message['message']['data']) : false;
if (is_string($message))
$message = json_decode($message, true);
if (is_array($message))
$type = (isset($message['type'])) ? $message['type'] : null;
$data = (isset($message['data'])) ? $message['data'] : [];
我不确定你这边的一切如何,但如果这部分发布消息:
await instance.post("/subscribe",
order_id: orderId,
order_token: purchaseToken,
data: JSON.stringify(results[0]),
);
它看起来像是发布消息的代理方法。因为与它一起发送的有效负载不像PubSub described schema,并且在最终消息中它看起来不像IAPQueryResponse
如果我遇到你的情况,我会检查一些东西来调试问题:
我如何向 PubSub 发布/读取消息(主题、订阅和消息负载) 我将按照Google PubSub publish documentation 中的描述编写发布机制 我会检查我的项目、主题和订阅 如果一切设置正确,我将比较所有其他消息数据 如果问题仍然存在,那么我将尝试将最少量的数据发布到 PubSub - 只需purchaseToken
在开始时检查是什么破坏了消息
为了方便调试:
创建拉取订阅 当您发布消息时,请使用“查看消息”检查拉取订阅消息对我来说,问题不直接在 PubSub 中,而是在您的消息发布/接收实现中。
2020 年 12 月 21 日更新:
流程:
客户创建/续订订阅 Publish to pubsub with authentication PubSub 通过“推送”将消息传输到分析应用程序以进行分析。如果您需要以下信息:
新订阅者计数 续订次数 活动订阅数 您可以创建自己的分析应用程序,但如果您需要更复杂的东西,则必须选择一种工具来满足您的需求。您也可以使用“pull”从 pubsub 获取消息,但我遇到的情况很少:
上次我使用 pull pubsub 返回随机数量的消息 - 如果我的限制是 50 并且队列中有超过 50 条消息,我希望收到 50 条消息,但有时 pubsub 给我的消息更少。李> PubSub 以随机顺序返回消息 - 现在有一个使用排序键的选项,但它是新事物。 要实现“拉”,您必须运行 crons 或带有“推”的东西,您会尽快收到消息。 使用“pull”,您必须依赖库/包(或任何其他语言),但使用“push”,您只需几行代码即可处理消息,就像我的 php 示例一样。【讨论】:
谢谢你。请查看帖子中的更新。 我猜你的问题已经回答了我 - 我不知道实例是什么,所以这只是一个猜测。正如我在回答中提到的那样,您必须将您的消息发布到 pubsub,然后您必须将订阅指向您的订阅以将消息发布到您的验证机。在那里你可以得到消息,你可以用它做任何你想做的事情。对我来说,最好的实现是在创建/更新订阅时创建一个 pubsub 消息,然后 pubsub 将消息发布到您必须进行分析的软件。您还可以使用服务帐户作为 pubsub 身份验证:)以上是关于Google 发布/订阅订阅数据与应用不匹配的主要内容,如果未能解决你的问题,请参考以下文章