Google订阅补坑

遇到的问题

Google 订阅在未过期之前,用户取消订阅,然后在我们app内购买订阅,Google 会产生一笔费用为0的订单,其实也就是说这个时候应该是不扣款,也不发会员。

正常的回调流程

谷歌扣款成功-->发送消息到回调地址-->验证回调信息的正确性-->查询google订单信息-->订单记录订单并发放会员
之前的设计是:只要查询的谷歌订单有效就会发放会员,用户取消订阅后再从app内购买订阅 就会生成一个新的订单而且认为是有效订单(其实这个订单是0元单)(如果用户在会员未到期之前一直操作取消订阅再重新app内购买订阅, 就会一直薅...)

场景:

假如用户A新订阅一个商品A产生的订单编号GPA.1234567(订单1),支付成功之后会员到账了,用户A取消订阅,再去app重新购买商品A,支付的金额实际为0,谷歌回调的订单编号是GPA.4567890(订单2),按照上边的逻辑只要验证(订单2)是支付成功的就直接给用户A会员到账了

解决思路

(主要的解决思路来源阅于:https://juejin.cn/post/7202211586675687483)


订单回调时会返回一个 purchaseTokenlinkedPurchaseToken 通过这两个字段去验证两笔订单是否有关联,下面根据模拟(为了方便测试谷歌会5分钟后发起续订):
新订阅-->续订-->取消订阅-->app内从新购买订阅(0元单)-->再续订,查看purchaseToken 和 linkedPurchaseToken的关系 及订单的开始结束时间


1).订单1:订单编号GPA.1234567,回调时返回的purchaseToken
{
    "version": "1.0",
    "notificationType": 4,
    "purchaseToken": "oobdohnegiepfgkehjhpniga.AO-",
    "subscriptionId": "subscribe_1"
},
通过purchaseToken解析返回的linkedPurchaseToken=null
开始时间: expiryTimeMillis=1719900697048 转为时间格式:2024-7-2 14:11:37
结束时间:expiryTimeMillis=1719900993387 转为时间格式:2024-7-2 14:16:33

2).续订,订单编号GPA.1234567..0,回调时返回的purchaseToken也会是
{
    "version": "1.0",
    "notificationType": 2,
    "purchaseToken": "oobdohnegiepfgkehjhpniga.AO-",
    "subscriptionId": "subscribe_1"
},
通过purchaseToken解析返回的linkedPurchaseToken=null
开始时间: 1719900697048 转为时间格式:2024-7-2 14:11:37
结束时间: 1719901293387 转为时间格式:2024-7-2 14:21:33

3). 取消订阅,当前的订单编号:GPA.1234567..0,回调时返回的purchaseToken也会是
{
    "version": "1.0",
    "notificationType": 3,
    "purchaseToken": "oobdohnegiepfgkehjhpniga.AO-",
    "subscriptionId": "subscribe_1"
},
通过purchaseToken解析返回的linkedPurchaseToken=null
开始时间: 1719900697048 转为时间格式:2024-7-2 14:11:37
结束时间: 1719901293387 转为时间格式:2024-7-2 14:21:33

4). 取消之后谷歌会发送订单编号:GPA.1234567..0到期的消息
{
    "version": "1.0",
    "notificationType": 13,
    "purchaseToken": "oobdohnegiepfgkehjhpniga.AO-",
    "subscriptionId": "subscribe_1"
}

5). 会员期内新购买订阅,订单编号:GPA.4567890(实际订单金额0),回调时返回的purchaseToken是
{
    "version": "1.0",
    "notificationType": 4,
    "purchaseToken": "gljhdcfkgcaadhnbgeeieiil.AO-",
    "subscriptionId": "subscribe_1"
},
通过purchaseToken解析返回的linkedPurchaseToken=oobdohnegiepfgkehjhpniga.AO-
开始时间: 1719901150359 转为时间格式:2024-7-2 14:19:10
结束时间: 1719901292565 转为时间格式:2024-7-2 14:21:32

6). 再续订,订单编号:GPA.4567890..0(实际订单金额是配置的金额),回调时返回的purchaseToken是
{
    "version": "1.0",
    "notificationType": 2,
    "purchaseToken": "gljhdcfkgcaadhnbgeeieiil.AO-",
    "subscriptionId": "subscribe_1"
},
通过purchaseToken解析返回的linkedPurchaseToken=oobdohnegiepfgkehjhpniga.AO-
开始时间: 1719901150359 转为时间格式:2024-7-2 14:19:10
结束时间: 1719901592742 转为时间格式:2024-7-2 14:26:32

通过上边的流程会发现 取消之后再订阅的订单(订单2) linkedPurchaseToken 与 第一个订单(订单1) purchaseToken相同(即:linkedPurchaseToken 是上一次订单的购买token),且结束时间一致,唯一不同的就是开始时间

所以,在回调处理时 
1. 要验证 linkedPurchaseToken 是否为null,是null 说明这是一个新订单
2. linkedPurchaseToken != null 且 notificationType=4 (新订单),根据 linkedPurchaseToken 查询订单(旧订单)的开始时间与结束时间,
3. 新订单的开始时间, 如果是介于 旧订单的开始时间 + 一个产品的订阅周期 * 续订次数 之内, 那么该订单就是 0元单。此处需要注意每次下单时,0元单也应该记录下来,因为一个订阅周期内可以多次发起重新订阅,导致0元单产生

还是拿订单1和订单2比较:订单2的开始时间介于 订单1的开始时间 至 (订单1的开始时间 + 5分钟*2),就意味着订单2为0元单

相关推荐

  1. Google订阅

    2024-07-11 19:50:03       24 阅读
  2. Django创建RSS订阅

    2024-07-11 19:50:03       50 阅读
  3. Redis-发布与订阅

    2024-07-11 19:50:03       41 阅读

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2024-07-11 19:50:03       66 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2024-07-11 19:50:03       70 阅读
  3. 在Django里面运行非项目文件

    2024-07-11 19:50:03       57 阅读
  4. Python语言-面向对象

    2024-07-11 19:50:03       68 阅读

热门阅读

  1. 低代码开发在金融系统中的应用研究

    2024-07-11 19:50:03       20 阅读
  2. conda 创建新的虚拟环境报错

    2024-07-11 19:50:03       21 阅读
  3. C++处理json数据注意点(url传递接收json数据)

    2024-07-11 19:50:03       21 阅读
  4. Windows批处理指令与Shell的关系

    2024-07-11 19:50:03       19 阅读
  5. 模电基础 - 直流电源

    2024-07-11 19:50:03       24 阅读
  6. Python魔法函数(Magic Methods简介

    2024-07-11 19:50:03       23 阅读
  7. C语言 输出n阶魔方阵

    2024-07-11 19:50:03       24 阅读
  8. ARM/Linux嵌入式面经(十一):地平线嵌入式实习

    2024-07-11 19:50:03       23 阅读
  9. xss攻击

    2024-07-11 19:50:03       23 阅读
  10. Rust简明教程第六章-错误处理&生命周期

    2024-07-11 19:50:03       26 阅读
  11. 【Django】Django 使用连接串配置数据库

    2024-07-11 19:50:03       22 阅读
  12. Sass 和 SCSS

    2024-07-11 19:50:03       19 阅读
  13. 系统迁移从CentOS7.9到Rocky8.9

    2024-07-11 19:50:03       23 阅读
  14. 深入理解CSS中的块格式化上下文(BFC)

    2024-07-11 19:50:03       21 阅读