抖音支付回调验签 go 版本

序言

最近在做抖音小程序支付,由于抖音开放平台的文档写的较为简陋,让人踩了不少坑,在这里整理一下做小程序支付的整个过程,以通用交易系统为例子。

准备条件

1)申请小程序,开通支付功能

这里需要明确你小程序所在的行业,服务范围,这个会影响到后续生成预支付订单所需要的参数

开通支付功能,如果你还希望你的小程序在唤起抖音收银台时,能够选择微信或者支付宝支付,那就在额外开通这块,大概半天就能完成审核,开通完后的截图如下:

除此之外,在解决方案处,还需要将对应的能力开通上,截图如下,默认全都开,可参考这个文档用来检查:https://developer.open-douyin.com/docs/resource/zh-CN/mini-app/open-capacity/trade-system/guide/general/basicapi#9389672b

2)生成并上传密钥

在小程序控制台 【开发】->【开发配置】-> 【密钥管理】生成密钥,如果没有可以先借助工具先生成。

应用公钥和私钥的生成方式可参考:(或使用 在线RSA加密解密,RSA2加密解密(SHA256WithRSA)-BeJSON.com 在线生成)

$ openssl
OpenSSL> genrsa -out private_key.pem 2048
Generating RSA private key, 2048 bit long modulus
....................+++
...........................................................................+++
e is 65537 (0x10001)
OpenSSL> rsa -in private_key.pem -pubout -out public_key.pem
writing RSA key
OpenSSL> exit
$ ls
private_key.pem public_key.pem

上传密钥

3)了解工作流程

  1. 小程序调用后端【订单接口】,生成一个订单数据
  2. 小程序调用后端【抖音支付接口】,后端返回 data 和 byteAuthorization 两个参数,供前端想抖音发起下单操作使用(细节一会展开说)
  3. 小程序拿到两个参数后,调用 tt.requestOrder 组件完成预下单的动作
  4. 完成预下单后,小程序调用 tt.getOrderPayment 组件,拉起收银台,此时用户就能够看到支付页面
  5. 用户完成支付或者取消支付,抖音就会调用事先配置的【回调接口】,后端验签,解析抖音回调的参数后,可根据结果,更新订单状态
  6. 小程序轮询后端【查询订单支付接口】,刷新页面,告知用户。

流程也可以参考如下的时序图:

以上就是整体的流程,下面重点讲述与后端有关的   【抖音支付接口】和 【回调接口】     

抖音支付接口

这一个接口是生成 data 和 byteAuthorization 参数,返回给前端。

第一步:确认产品的商品类型和标签

我所做的是 虚拟产品的会员费用,所以我的 商品类型和标签如下:

// 商品类型
type = 301
// 标签
tag = "tag_group_7272625659888058380"

 其他的,可参考以下文档:

https://developer.open-douyin.com/docs/resource/zh-CN/mini-app/open-capacity/trade-system/guide/general/basicrules#6a1682c4

第二步:参数填写,这里只需要填以下参数即可,给出完整代码如下:

结合官方的文档,再理解代码:使用流程__抖音开放平台


type RequestOrderData struct {SkuList          []*SkuItem       `json:"skuList"`OutOrderNo       string           `json:"outOrderNo"`TotalAmount      int32            `json:"totalAmount"`PayExpireSeconds int32            `json:"payExpireSeconds"`OrderEntrySchema OrderEntrySchema `json:"orderEntrySchema"`PayNotifyUrl     string           `json:"payNotifyUrl"`LimitPayWayList  []int            `json:"limitPayWayList"`
}type SkuItem struct {SkuId      string   `json:"skuId"`Price      int      `json:"price"`Quantity   int      `json:"quantity"`Title      string   `json:"title"`ImageList  []string `json:"imageList"`Type       int      `json:"type"`TagGroupId string   `json:"tagGroupId"`
}type OrderEntrySchema struct {Path   string `json:"path"`Params string `json:"params"`
}func DyPay () {// 构建参数payAmount := 100 // 单位是分requestData := &common.RequestOrderData{SkuList:          make([]*common.SkuItem, 0),OutOrderNo:       "业务系统订单号",TotalAmount:      int32(payAmount),PayExpireSeconds: 3600,PayNotifyUrl:     “”, // 后端的回调地址,不填会默认使用在抖音开放平台后台配置的域名OrderEntrySchema: common.OrderEntrySchema{Path:   "pages/pay/index", // 小程序页面地址,找前端提供Params: "",},LimitPayWayList: []int{}, // 如果需要屏蔽微信和支付宝支付渠道,就填入 1, 2}skuItem := &common.SkuItem{Title:      goodsModel.Name,Price:      int(payAmount.IntPart()),Quantity:   1,SkuId:      "商品id,业务系统自己补充",TagGroupId: "tag_group_7272625659888058380", // 固定ImageList: []string{"" // 商品图片,必填,},Type: 301, // 固定}requestData.SkuList = append(requestData.SkuList, skuItem)// 生成签名data, authorization := s.getByteAuthorizationAndData(requestData)// 打印 或其他操作,就不补充了
}// 获取签名
func getByteAuthorizationAndData(data *RequestOrderData) (string, string) {// 请求时间戳timestamp := strconv.FormatInt(time.Now().Unix(), 10)appId := "" // 获取小程序的 appid // 随机字符串nonceStr := randStr(10)// 应用公钥版本,每次重新上传公钥后需要更新,可通过「开发管理-开发设置-密钥设置」处获取keyVersion := "1" // 对应后台的版本设置,// 应用私钥,用于加签 重要:1.测试时请修改为开发者自行生成的私钥;2.请勿将示例密钥用于生产环境;3.建议开发者不要将私钥文本写在代码中privateKeyStr := ""// 生成好的data jsondataStr, _ := json.Marshal(data)authorization, _ := s.dyPayGetByteAuthorization(privateKeyStr, string(dataStr), appId, nonceStr, timestamp, keyVersion)return string(dataStr), authorization
}// 向抖音发起获取签名接口
func dyPayGetByteAuthorization(privateKeyStr, data, appId, nonceStr, timestamp, keyVersion string) (string, error) {var byteAuthorization string// 读取私钥key, err := base64.StdEncoding.DecodeString(strings.ReplaceAll(privateKeyStr, "\n", ""))if err != nil {return "", err}privateKey, err := x509.ParsePKCS1PrivateKey(key)if err != nil {return "", err}// 生成签名signature, err := s.dyPayGetSignature("POST", "/requestOrder", timestamp, nonceStr, data, privateKey)if err != nil {return "", err}// 构造byteAuthorizationbyteAuthorization = fmt.Sprintf("SHA256-RSA2048 appid=%s,nonce_str=%s,timestamp=%s,key_version=%s,signature=%s", appId, nonceStr, timestamp, keyVersion, signature)return byteAuthorization, nil
}// 拼签名
func dyPayGetSignature(method, url, timestamp, nonce, data string, privateKey *rsa.PrivateKey) (string, error) {fmt.Printf("method:%s\n url:%s\n timestamp:%s\n nonce:%s\n data:%s", method, url, timestamp, nonce, data)targetStr := method + "\n" + url + "\n" + timestamp + "\n" + nonce + "\n" + data + "\n"h := sha256.New()h.Write([]byte(targetStr))digestBytes := h.Sum(nil)signBytes, err := rsa.SignPKCS1v15(rand.Reader, privateKey, crypto.SHA256, digestBytes)if err != nil {return "", err}sign := base64.StdEncoding.EncodeToString(signBytes)return sign, nil
}func randStr(length int) string {b := make([]byte, length)_, err := rand.Read(b)if err != nil {panic(err)}return base64.StdEncoding.EncodeToString(b)
}

在此小程序所需要的参数就已经生成好了,同时你也可以借助抖音签名验证工具(API调试台)进行验证。你也可以阅读抖音的官方文档,了解他们写的签名算法:

https://developer.open-douyin.com/docs/resource/zh-CN/mini-app/develop/server/signature-algorithm#%E7%AD%BE%E5%90%8D%E9%AA%8C%E8%AF%81

抖音支付回调

官方文档:功能概述__抖音开放平台

  • 由于网络波动等原因,可能会产生重复的通知消息,接入方需要做好幂等,正确处理。​
  • •​回调可能存在延时,若实时性要求高,开发者可以通过主动请求查询订单信息,确认支付结果。​
  • •​支付回调一定要做验签处理,防止收到假通知,可参考下文验签示例代码。​
  • •​在开发者服务端收到回调且处理成功后,需要按以下正常返回示例返回并且 HTTP 响应状态码设为 200,否则会认为通知失败进行重试。重试频率为 15s/30s/1m/2m/4m/8m/16m/32m/64m/128m - 总共4小时12m。​
  • •​JS API 下单 有 payNotifyUrl 字段,如果在下单传了该字段,则会优先使用下单的 payNotifyUrl 回调地址,否则使用解决方案配置的回调地址。​

先看官方文档,核心就在验签,这里最容易出错的,其实就是 request body 的获取,一定是要直接原格式获取的,不要进行其他加工

type DYResultCallBack struct {Version string `json:"version"`Msg     string `json:"msg"`Type    string `json:"type"`
}type DYResultCallBackData struct {AppId          string `json:"app_id"`OutOrderNo     string `json:"out_order_no"`OrderId        string `json:"order_id"`Status         string `json:"status"`TotalAmount    int    `json:"total_amount"`EventTime      int64  `json:"event_time"`DiscountAmount int64  `json:"discount_amount"`PayChannel     int    `json:"pay_channel"`UserBillPayId  string `json:"user_bill_pay_id"`Message        string `json:"message"`Extra          string `json:"extra"`
}func (c *DyPayController) DyMiniPayNotify(ctx *gin.Context) {// 参数赋值request := &pay.DYResultCallBack{}if err := ctx.ShouldBindJSON(request); err != nil {c.Fail(ctx, errorcode.BASE_STRUCT_TRANS_FAILED, "无效的参数")return}callbackMsg := &pay.DYResultCallBackData{}err := json.Unmarshal([]byte(request.Msg), callbackMsg)if err != nil {c.Fail(ctx, errorcode.BASE_STRUCT_TRANS_FAILED, "解析消息体失败")return}// 验签代码verifyInfoRequest := &verifyInfo{}verifyInfoRequest.Nonce = ctx.Request.Header.Get("Byte-Nonce-Str")verifyInfoRequest.Signature = ctx.Request.Header.Get("Byte-Signature")verifyInfoRequest.Timestamp = ctx.Request.Header.Get("Byte-Timestamp")bodyStr, _ := json.Marshal(request)verifyInfoRequest.Body = fmt.Sprintf("%s", bodyStr)valid, err := c.verifySignature(verifyInfoRequest)if err != nil {c.Fail(ctx, errorcode.BASE_STRUCT_TRANS_FAILED, "验签失败")return}if !valid {c.Fail(ctx, errorcode.BASE_STRUCT_TRANS_FAILED, "验签失败-1")return}// 处理支付信息if callbackMsg.Status != "SUCCESS" {c.Fail(ctx, errorcode.BASE_STRUCT_TRANS_FAILED, "支付失败")return}// todo 处理业务逻辑 c.Success(ctx)
}// verifySignature 验证签名
func (c *DyPayController) verifySignature(reqInfo *verifyInfo) (bool, error) {// 打开 PEM 文件 平台的公钥data, err := os.ReadFile("./dy_platform_public_key.pem")if err != nil {return false, err}return c.CheckSign(reqInfo.Timestamp, reqInfo.Nonce, reqInfo.Body, reqInfo.Signature, string(data))
}func (c *DyPayController) Fail(ctx *gin.Context, code int, msg string) {result := map[string]interface{}{"err_tips": msg,"err_no":   code,}ctx.JSON(http.StatusInternalServerError, result)
}func (c *DyPayController) Success(ctx *gin.Context) {result := gin.H{"err_no":   0,"err_tips": "success",}ctx.JSON(http.StatusOK, result)
}func (c *DyPayController) CheckSign(timestamp, nonce, body, signature, pubKeyStr string) (bool, error) {pubKey, err := c.PemToRSAPublicKey(pubKeyStr) // 注意验签时publicKey使用平台公钥而非应用公钥if err != nil {return false, err}hashed := sha256.Sum256([]byte(timestamp + "\n" + nonce + "\n" + body + "\n"))signBytes, err := base64.StdEncoding.DecodeString(signature)if err != nil {return false, err}err = rsa.VerifyPKCS1v15(pubKey, crypto.SHA256, hashed[:], signBytes)return err == nil, nil
}func (c *DyPayController) PemToRSAPublicKey(pemKeyStr string) (*rsa.PublicKey, error) {block, _ := pem.Decode([]byte(pemKeyStr))if block == nil || len(block.Bytes) == 0 {return nil, fmt.Errorf("empty block in pem string")}key, err := x509.ParsePKIXPublicKey(block.Bytes)if err != nil {return nil, err}switch key := key.(type) {case *rsa.PublicKey:return key, nildefault:return nil, fmt.Errorf("not rsa public key")}
}

总结

抖音文档会将大概的逻辑讲一下,但是在实操方面的补充会比较少,希望这篇文章能够帮助到,有问题,欢迎在评论区留言,我看到一定会马上回复。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/54927.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

鸿蒙开发(NEXT/API 12)【硬件(传感器开发)】传感器服务

使用场景 Sensor Service Kit(传感器服务)使应用程序能够从传感器获取原始数据,并提供振感控制能力。 Sensor(传感器)模块是应用访问底层硬件传感器的一种设备抽象概念。开发者可根据传感器提供的相关接口订阅传感器…

Mitsuba 渲染基础

Mitsuba 渲染基础 0. Abstract1. 安装 Mitsuba21.1 下载 Mitsuba2 源码1.2 选择后端 (variants)1.3 编译 2. [Mitsuba2PointCloudRenderer](https://github.com/tolgabirdal/Mitsuba2PointCloudRenderer)2.1 Mitsuba2 渲染 XML2.2 Scene 场景的 XML 文件格式2.2.1 chair.npy to…

Comfyui 学习笔记2

在潜空间放大,三种方法:NNLatentUpscale、Upscale Latent、Upscale Latent,其中只有NNLatentUpscale自带模型优化,其他两种需要KSample重新绘画,NNLatentUpscale后也可以接KSmaple。 像素空间放大,同理&am…

大模型推理任务Nvidia GPU选型指南

大型语言模型 (LLM)(如 GPT-4、BERT 和其他基于 Transformer 的模型)彻底改变了 AI 格局。这些模型需要大量计算资源来进行训练和推理。选择合适的 GPU 进行 LLM 推理可以极大地影响性能、成本效益和可扩展性。 在本文中,我们将探索最适合 L…

Spring的热部署工具和数据库密码加盐操作

1.布置热部署 引言:在程序运行起来后,如果我们对代码进行了修改,需要重新测试修改后的程序,就得重新启动程序,这样很麻烦。于是引入热部署之后,我们就不需要重新启动程序,会自动更正。 1.配置po…

牛顿迭代法求解x 的平方根

牛顿迭代法是一种可以用来快速求解函数零点的方法。 为了叙述方便,我们用 C C C表示待求出平方根的那个整数。显然, C C C的平方根就是函数 f ( x ) x c − C f(x)x^c-C f(x)xc−C 的零点。 牛顿迭代法的本质是借助泰勒级数,从初始值开始快…

前端大模型入门:使用Transformers.js手搓纯网页版RAG(二)- qwen1.5-0.5B - 纯前端不调接口

书接上文,本文完了RAG的后半部分,在浏览器运行qwen1.5-0.5B实现了增强搜索全流程。但受限于浏览器和模型性能,仅适合于研究、离线和高隐私场景,但对前端小伙伴来说大模型也不是那么遥不可及了,附带全部代码&#xff0c…

【深度学习】(5)--搭建卷积神经网络

文章目录 搭建卷积神经网络一、数据预处理1. 下载数据集2. 创建DataLoader(数据加载器) 二、搭建神经网络三、训练数据四、优化模型 总结 搭建卷积神经网络 一、数据预处理 1. 下载数据集 在PyTorch中,有许多封装了很多与图像相关的模型、…

vue3 通过 axios + jsonp 实现根据公网 ip, 查询天气信息

前提 安装 axios 的 jsonp 适配器。 pnpm install pingtou/axios-jsonp 简单使用说明:当与后端约定的请求 callback 参数名称不为为 callback 时,可修改。一般无需添加。 1. 获取当前电脑 ip 和城市信息 请求地址: https://whois.pconl…

Linux之我不会

一、常用命令 1.系统管理 1.1 systemctl start | stop | restart | status 服务名 案例实操 1 查看防火墙状态 systemctl status firewalld2 停止防火墙服务 systemctl stop firewalld3 启动防火墙服务 systemctl start firewalld4 重启防火墙服务 systemctl restart f…

【Canvas与诗词】秋夕.杜牧(银烛秋光冷画屏......)

【成图】 【代码】 <!DOCTYPE html> <html lang"utf-8"> <meta http-equiv"Content-Type" content"text/html; charsetutf-8"/> <head><title>金六边形外圈绿色底录杜牧秋夕诗</title><style type"…

【电商搜索】现代工业级电商搜索技术-Facebook语义搜索技术QueSearch

【电商搜索】现代工业级电商搜索技术-Facebook语义搜索技术Que2Search 目录 文章目录 【电商搜索】现代工业级电商搜索技术-Facebook语义搜索技术Que2Search目录0. 论文信息1. 研究背景&#xff1a;2. 技术背景和发展历史&#xff1a;3. 算法建模3.1 模型架构3.1.1 双塔与分类 …

NLP:BERT的介绍

1. BERT 1.1 Transformer Transformer架构是一种基于自注意力机制(self-attention)的神经网络架构&#xff0c;它代替了以前流行的循环神经网络和长短期记忆网络&#xff0c;已经应用到多个自然语言处理方向。   Transformer架构由两个主要部分组成&#xff1a;编码器(Encod…

【HarmonyOS】应用引用media中的字符串资源如何拼接字符串

【HarmonyOS】应用引用media中的字符串资源如何拼接字符串 一、问题背景&#xff1a; 鸿蒙应用中使用字符串资源加载&#xff0c;一般文本放置在resoutces-base-element-string.json字符串配置文件中。便于国际化的处理。当然小项目一般直接引用字符串&#xff0c;不需要加载s…

python爬虫:从12306网站获取火车站信息

代码逻辑 初始化 (init 方法)&#xff1a; 设置请求头信息。设置车站版本号。 同步车站信息 (synchronization 方法)&#xff1a; 发送GET请求获取车站信息。返回服务器响应的文本。 提取信息 (extract 方法)&#xff1a; 从服务器响应中提取车站信息字符串。去掉字符串末尾的…

如何通过Dockfile更改docker中ubuntu的apt源

首先明确我们有一个宿主机和一个docker环境&#xff0c;接下来的步骤是基于他们两个完成的 1.在宿主机上创建Dockerfile 随便将后面创建的Dockerfile放在一个位置,我这里选择的是 /Desktop 使用vim前默认你已经安装好了vim 2.在输入命令“vim Dockerfile”之后&#xff0c;…

知识付费APP开发指南:基于在线教育系统源码的技术详解

本篇文章&#xff0c;我们将探讨基于在线教育系统源码的知识付费APP开发的技术细节&#xff0c;帮助开发者和企业快速入门。 一、选择合适的在线教育系统源码 选择合适的在线教育系统源码是开发的关键一步。市场上有许多开源和商业化的在线教育系统源码&#xff0c;开发者需要…

花都狮岭寄宿自闭症学校:开启孩子的生命之门

在花都狮岭这片充满温情的土地上&#xff0c;有一所特别的学校&#xff0c;它像一把钥匙&#xff0c;轻轻旋转&#xff0c;为自闭症儿童们开启了一扇通往无限可能的生命之门——这就是广州星贝育园自闭症儿童寄宿制学校。这所学校不仅是知识的摇篮&#xff0c;更是孩子们心灵成…

React 启动时webpack版本冲突报错

报错信息&#xff1a; 解决办法&#xff1a; 找到全局webpack的安装路径并cmd 删除全局webpack 安装所需要的版本

Python(六)-拆包,交换变量名,lambda

目录 拆包 交换变量值 引用 lambda函数 lambda实例 字典的lambda 推导式 列表推导式 列表推导式if条件判断 for循环嵌套列表推导式 字典推导式 集合推导式 拆包 看一下在Python程序中的拆包&#xff1a;把组合形成的元组形式的数据&#xff0c;拆分出单个元素内容…