微信小程序虚拟支付(道具直购)踩坑全记录:从-15005到支付成功

微信小程序虚拟支付(道具直购)踩坑全记录:从-15005到支付成功

一、前言

在实现微信小程序虚拟支付-道具直购功能过程中,遭遇了大量官方文档未明确、错误提示模糊的隐藏坑点。本文把今天真实遇到的问题、错误原因、正确代码完整汇总,可直接作为开发手册使用。


二、核心环境说明

  • 接口:wx.requestVirtualPayment(小程序前端唤起支付)
  • 模式:short_series_goods(道具直购固定模式)
  • 环境:env:1 沙箱 / env:0 正式
  • 错误码:1001 参数错误 / -15005 签名无效

三、今日问题汇总(已脱敏+修复所有错误)

问题1:前端调用报 1001 错误

错误信息

arduino 复制代码
parameter.signature should be String instead of Undefined

原因 以为只需要传 paySig,但道具支付必须传两个签名

  • paySig:订单参数签名
  • signature:用户身份签名

前端必须传的完整参数

js 复制代码
wx.requestVirtualPayment({
  env: 1,
  mode: "short_series_goods",
  offerId: "微信平台分配的道具ID", // 这里不能填productId!
  paySig: "后端计算",
  signData: "后端返回JSON字符串",
  signature: "后端计算",
  success:()=>{},
  fail:(err)=>{}
})

问题2:持续报 -15005 SIGNATURE_INVALID 签名错误

今天最核心、最耗时的问题,一共踩中 3 个坑:

坑1:signData 里多了 platform":"miniprogram"

错误写法

json 复制代码
{"offerId":"xxx",...,"platform":"miniprogram",...}

结论 道具支付不允许加 platform 字段,一加必签错。


坑2:signature 错误对 sessionKey 做了 base64 解码

错误代码

python 复制代码
key_bytes = base64.b64decode(session_key) // 这一行会直接导致-15005

真实规则(虚拟支付专属)

  • sessionKey 不需要解码
  • 直接用字符串 encode("utf-8") 参与 HMAC

最终正确代码

python 复制代码
def calc_signature(self, session_key: str, sign_data: str) -> str:
    return hmac.new(
        key=session_key.encode("utf-8"),
        msg=sign_data.encode("utf-8"),
        digestmod=hashlib.sha256,
    ).hexdigest()

坑3:paySig 拼接规则与 URI 理解错误

正确规则

ini 复制代码
paySig = HMAC-SHA256(appKey, "requestVirtualPayment" + "&" + signData)

正确代码

python 复制代码
def calc_pay_sig(self, uri: str, post_body: str, env: int) -> str:
    app_key = self._get_app_key(env)
    msg = uri + "&" + post_body
    return hmac.new(
        key=app_key.encode("utf-8"),
        msg=msg.encode("utf-8"),
        digestmod=hashlib.sha256,
    ).hexdigest()

问题3:支付页面不显示道具图片/名称

原因(环境隔离规则)

  • env=1 沙箱 → 只能读 开发版道具
  • env=0 正式 → 只能读 线上版道具
  • 调用的 offerId 必须与后台配置完全一致
  • 道具配置修改后必须重新发布/保存才能同步

解决方案

  1. 确认环境与道具版本对应
  2. 检查 offerId 正确(不是 productId)
  3. 重新编辑道具并发布
  4. 检查图片格式/大小符合要求

四、最终正确全套规则(脱敏版)

1. signData 结构(不能多字段)

json 复制代码
{
  "offerId": "微信分配的ID",
  "buyQuantity": 1,
  "env": 1,
  "currencyType": "CNY",
  "productId": "你自定义的商品ID",
  "goodsPrice": 价格,
  "outTradeNo": "订单号",
  "attach": "test"
}

2. paySig 计算

  • 密钥:虚拟支付 appKey
  • 原文:requestVirtualPayment&signData
  • 算法:HMAC-SHA256

3. signature 计算

  • 密钥:sessionKey 直接 encode,不解码
  • 原文:signData
  • 算法:HMAC-SHA256

4. 前端调用参数(最终正确)

js 复制代码
wx.requestVirtualPayment({
  env: 1,
  mode: "short_series_goods",
  offerId: "微信道具ID",
  paySig: "",
  signData: "",
  signature: "",
  success:()=>{},
  fail:()=>{}
})
相关推荐
陈随易7 小时前
Redis 8.8发布,一定要更新
前端·后端·程序员
ssshooter12 小时前
Tauri 应用首次上架 App Store 被驳回了 3 次(iOS)和 12 轮(macOS)的经历
前端·ios·程序员
阿祖zu12 小时前
2026 企业级 Agent 产品落地思考与全流程指南
前端·程序员·aigc
京东云开发者12 小时前
AI助力跨境增长:京点点Oxygen Vision 跨境套图AI生成技术实践与展望
程序员
Patrick_Wilson14 小时前
过早优化是万恶之源:50 年工程史反复在教我们的一件事
程序员·架构·ai编程
一心只读圣贤书14 小时前
QwenCode使用教程
程序员
程序员cxuan16 小时前
Codex 官方:/goal 的正确打开方式
人工智能·后端·程序员
我要改名叫嘟嘟16 小时前
“10w阅读”是怎样体验,以及写出一篇能挣多少钱
程序员
Hilaku16 小时前
如何实现 0 毫秒无感页面跳转?聊聊被低估的 Speculation Rules API
前端·javascript·程序员
用户69190268133919 小时前
【初学者报道】| 全栈实战:基于 RESTful 规范的用户数据接口设计与实现
程序员