抖音支付回调验签 go 版本

news2025/1/22 17:49:43

序言

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

准备条件

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 json
	dataStr, _ := 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
	}
	// 构造byteAuthorization
	byteAuthorization = 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, nil
	default:
		return nil, fmt.Errorf("not rsa public key")
	}
}

总结

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

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

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

相关文章

Linux--基本指令

目录 1.ls指令 ​2.pwd指令 3.cd指令 4.tree指令 5.touch/mkdir/rmdir(rm)指令 6.cp/mv/cat/tac指令 7.head/tail/管道 8.date 9.find/which/grep 10.其它小知识 1.ls指令 补充知识: 2.pwd指令 补充知识: 3.cd指令 补充知识: 4.tree指令 …

Java语法-类和对象之抽象类和接口

1.抽象类 1.1 抽象类的概念 一个类中没有足够的信息来描述一个具体的对象,这样的类就是抽象类 比如: 从图中我们可以看出,只有继承了的类,我们产生的实例,调用的draw方法都是他们本身重写的draw方法,不会调用父类Shape的draw()方法,因此我们可以不管父类里面的draw()方法里面的…

MySQL 之多表设计详解

在实际应用场景中,我们经常需要处理包含多种数据实体及其之间复杂关系的业务逻辑,例如电商平台的用户、商品、订单,社交网络的用户、帖子、评论等等。如果将所有数据都堆砌在一张表中,不仅会造成数据冗余、难以维护,还…

MySQL 8.0.34 从C盘迁移到D盘

因为开始C盘够用,没注意mysql安装位置,如今C盘爆满,只能把mysql转移到D盘,以腾出更多的空间让我折腾。 一、关闭mysql服务 二、找到C盘MySQL安装文件和Data文件 1.找到C盘mysql bin文件目录安装文件路径: C:\Progra…

行为设计模式 -模板方法模式- JAVA

模板方法模式 一 .简介二. 案例2.1 抽象类(Abstract Class)2.2 具体子类(Concrete Class)2.3 测试 三. 结论3.1 优缺点3.2 适用场景3.3 要点 前言 这是我在这个网站整理的笔记,有错误的地方请指出,关注我,接…

linux从入门到精通--从基础学起,逐步提升,探索linux奥秘(六)

linux从入门到精通–从基础学起,逐步提升,探索linux奥秘(六) 一、linux高级指令(1) 1、hostname指令 1)作用:操作服务器的主机名(读取、设置) 2&#xff0…

huggingface的transformers与datatsets的安装与使用

目录 1.安装 2.分词 2.1tokenizer.encode() 2.2tokenizer.encode_plus () 2.3tokenizer.batch_encode_plus() 3.添加新词或特殊字符 3.1tokenizer.add_tokens() 3.2 token…

Python自动收发邮件的详细步骤与使用方法?

Python自动收发邮件教程?Python怎么实现收发邮件? Python作为一种强大的编程语言,提供了丰富的库和工具,使得自动收发邮件变得简单而高效。AokSend将详细介绍如何使用Python自动收发邮件,帮助读者掌握这一实用技能。 …

【ASE】第四课_护盾效果(有碰撞效果)

今天我们一起来学习ASE插件,希望各位点个关注,一起跟随我的步伐 今天我们来学习护盾的效果。 思路: 1.添加纹理贴图和法线贴图(这里省略) 2.添加护盾边缘顶点扰动效果,也可以理解成变形效果 3.添加碰撞…

关于frp Web界面-----frp Server Dashboard 和 frp Client Admin UI

Web 界面 官方文档:https://gofrp.org/zh-cn/docs/features/common/ui/ 目前 frpc 和 frps 分别内置了相应的 Web 界面方便用户使用。 客户端 Admin UI 服务端 Dashboard 服务端 Dashboard 服务端 Dashboard 使用户可以通过浏览器查看 frp 的状态以及代理统计信…

59 双向循环神经网络_by《李沐:动手学深度学习v2》pytorch版

系列文章目录 文章目录 系列文章目录双向RNN推理 总结以下为理论部分双向循环神经网络隐马尔可夫模型中的动态规划双向模型定义模型的计算代价及其应用 (**双向循环神经网络的错误应用**)小结练习 双向RNN 这里理解这个图的时候,不要把正向和逆向认为有上下的关系&a…

计算机毕业设计 基于Python的音乐平台的设计与实现 Python+Django+Vue 前后端分离 附源码 讲解 文档

🍊作者:计算机编程-吉哥 🍊简介:专业从事JavaWeb程序开发,微信小程序开发,定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事,生活就是快乐的。 🍊心愿:点…

IDE 使用技巧与插件推荐(含例说明)

在使用集成开发环境(IDE)进行编程时,掌握一些技巧和使用高效的插件可以显著提高开发效率。以下是一些通用的IDE使用技巧和插件推荐,适用于多种流行的IDE,如IntelliJ IDEA、Visual Studio Code、PyCharm等。每个技巧和插…

泳池异常检测系统源码分享

泳池异常检测检测系统源码分享 [一条龙教学YOLOV8标注好的数据集一键训练_70全套改进创新点发刊_Web前端展示] 1.研究背景与意义 项目参考AAAI Association for the Advancement of Artificial Intelligence 项目来源AACV Association for the Advancement of Computer Vis…

Pencils Protocol 成市场新宠,生态通证$DAPP价值几何

Pencils Protocol 是 Scroll 生态综合性收益平台,其仅在 Scroll 生态单链 TVL 就已经突破了 3.5 亿美元,同时在上线短短几个月的时间里就积累了超 50 万活跃社区用户。现阶段 Pencils Protocol 已经完成了 DAPP 通证的 TGE ,分别在 Tokensoft…

[SAP ABAP] 锁对象

在SAP中使用锁对象,用于避免在数据库中插入或更改数据时出现不一致的情况 1.创建锁对象 数据准备 学校表(ZDBT_SCH_437) 使用事务码SE11创建锁对象 点击"锁对象"单选按钮,输入以E开头的锁定对象的名称,然后点击创建按钮 锁对象名…

关于宝塔PHP getenv无法获取环境变量问题解决办法

今天有用ThinkPHP8接入阿里云OSS时,需要用的用到getenv()来读取环境变量,因为新版OSS SDK是用环境变更来设置AK的。 现象 正常执行PHP文件,可以取到环境变量;但是通过nginxphp-fpm调用脚本取到不到环境变量 原因 php-fpm为了防止…

【软考】高速缓存的组成

目录 1. 说明2. 组成 1. 说明 1.高速缓存用来存放当前最活跃的程序和数据。2.高速缓存位于CPU 与主存之间。3.容量般在几千字节到几兆字节之间。4.速度一般比主存快 5~10 倍,由快速半导体存储器构成。5.其内容是主存局部域的副本,对程序员来说是透明的。…

【C++】C++中如何处理多返回值、C++中的模板

十四、C中如何处理多返回值 本部分也是碎碎念,因为这些点都是很小的点,构不成一篇文章,所以本篇就是想到哪个点就写哪个点。 1、C中如何处理多个返回值 写过python的同学都知道,当你写一个函数的返回时,那是你想返回…

STM32 F1移植FATFS文件系统 USMART组件测试相关函数功能

STM32 F1移植FATFS文件系统 使用USMART调试组件测试相关函数功能 文章目录 STM32 F1移植FATFS文件系统 使用USMART调试组件测试相关函数功能前言部分主要相关代码# USMART介绍1. mf_scan_files 扫描磁盘文件2. mf_mount 挂载磁盘3. mf_open 打开文件4. mf_read 读数据内容5. mf…