微信小程序支付(服务商模式)解决
前言
微信支付官方文档: 小程序开发与支付、服务商的关系,参考这个文档
1、第三方(服务商)自己申请账号,自己开发,生成指定内页给特约商户用,该模式简称中心化模式。
2、以特约商户身份申请小程序appId,第三方完成开发,该模式简称外包模式。 3、通过开放平台第三方开发者代特约商户进行小程序的开发,该模式简称第三方模式。
本文适用于中心化模式,服务商自己开发一个小程序,但是收款是直接受到对应的特约商户账户中,不收到服务商自己账户中。
基本流程
- 服务商再微信后台申请微信小程序
- 小程序开通微信支付或绑定已开通微信支付的商户号
特约商户需要操作的流程
- 服务商再微信支付商户后台,为特约商户开通微信服务商模式下的微信支付账户
- 特约商户在收到微信邮件发送的登录账号和密码,登录自己的微信支付商户后台,绑定小程序的appId
- 提交审核,被拒绝的话再次提交审核,直到审核通过
- 服务商管理后台中找到"待关联商户号"并确认
- 登录微信支付服务商商户后台,手动为特约商户绑定与服务商主体或特约商户主体一致的公众号,APP或小程序的appId。最多配置5个,手工配置路径:"服务商商户平台-服务商功能-子商户管理-开发配置-特约商户APPID配置"。
支付部分
- appid:注意这里是服务号的appid,不是小程序的
- mch_id:这里是用服务商的id 在我的账号一栏可以找到
- sub_appid: 这里才是小程序的appid
- sub_mch_id: 这里对应特约商户号id 付款到对应商户的凭证就是这个 在注册特约商户的时候邮件里可以找到 这里建议配置到数据库动态传递
- nonce_str: 随机字符串
- body: 这里随意填写,也可以填写商品名称
- out_trade_no: 订单号
- total_fee: 这里必须是整数,单位是分
- trade_type: 公众平台支付或小程序支付填写:JSAPI,如果是APP的填写:APP
- sub_openid: 此参数是在发起支付前在小程序内调起wx.login 方法获得code 然后后台通过置换 获得用户openid
- spbill_create_ip:这里可以随意填写
- notify_url: 支付回调的地址
- sign: 此参数为签名参数 需要将需要传递的参数进行排序并且进行md5签名,需要注意的是需添加参数key 即之前修改的服务商api密钥
好了 参数分析完毕 在后台调用统一下单方法 不出意外是成功的,下单代码如下:
Controller部分
@RestController
@RequestMapping("/payment")
public class WxPayController {
private Logger logger = LoggerFactory.getLogger(WxLoginController.class);
@Autowired
private AppletOrderService appletOrderService;
@Autowired
private WxPayProperties wxPayProperties;
@ResponseBody
@PostMapping(value = "/appletWxPay", produces = "application/json;charset=UTF-8")
public Map<Object, Object> appletWxPay(@RequestParam String openId, String totalFee) throws Exception {
logger.info("[WxPayController].appletWxPay...openId:{}", openId);
SortedMap<Object, Object> resultMap = new TreeMap<Object, Object>();
String body = "测试";
String out_trade_no = String.valueOf(IdWorker.getInstance().nextId());
PreOrderResult preOrderResult = appletOrderService.placeOrder(body, out_trade_no, totalFee, openId);
System.out.println(preOrderResult);
if(WxContants.SUCCESS.equals(preOrderResult.getReturn_code()) && WxContants.SUCCESS.equals(preOrderResult.getResult_code())){
resultMap.put("appId", wxPayProperties.getApp_id());
resultMap.put("timeStamp", Long.toString(System.currentTimeMillis()/1000));
resultMap.put("nonceStr", UUID.randomUUID().toString().replaceAll("-", "").toUpperCase());
resultMap.put("package", "prepay_id="+preOrderResult.getPrepay_id());
resultMap.put("signType", "MD5");
resultMap.put("sign", SignUtils.createSignByMd5(resultMap, wxPayProperties.getKey()));
resultMap.put("returnCode", "SUCCESS");
resultMap.put("returnMsg", "OK");
logger.info("【小程序支付】统一下单成功,返回参数:"+resultMap);
}else{
resultMap.put("returnCode", preOrderResult.getReturn_code());
resultMap.put("returnMsg", preOrderResult.getReturn_msg());
logger.info("【小程序支付】统一下单失败,失败原因:{}" + preOrderResult.getReturn_msg());
}
logger.info("[WxPayController].appletWxPay...CodeUrl:{}", preOrderResult.getCode_url());
return resultMap;
}
}
Serivice部分
@Service
public class AppletOrderServiceImpl implements AppletOrderService{
@Autowired
private WxPayProperties wxPayProperties;
@Override
public PreOrderResult placeOrder(String body, String out_trade_no, String total_fee, String openId) throws Exception {
// 生成预付单对象
PreOrder preOrder = new PreOrder();
preOrder.setAppid(wxPayProperties.getApp_id());
preOrder.setMch_id(wxPayProperties.getMch_id());
preOrder.setSub_appid(wxPayProperties.getSub_app_id());
preOrder.setSub_mch_id(wxPayProperties.getSub_mch_id());
String nonce_str = UUID.randomUUID().toString().replaceAll("-", "").toUpperCase();
preOrder.setNonce_str(nonce_str);
preOrder.setBody(body);
preOrder.setOut_trade_no(out_trade_no);
preOrder.setTotal_fee(new BigDecimal(total_fee));
preOrder.setSpbill_create_ip(wxPayProperties.getSpbill_create_ip());
preOrder.setNotify_url(wxPayProperties.getNotify_url());
preOrder.setTrade_type(WxContants.TRADE_TYPE);
preOrder.setSub_openid(openId);
SortedMap<Object, Object> p = new TreeMap<Object, Object>();
p.put("appid", wxPayProperties.getApp_id());
p.put("mch_id", wxPayProperties.getMch_id());
p.put("sub_appid", wxPayProperties.getSub_app_id());
p.put("sub_mch_id", wxPayProperties.getSub_mch_id());
p.put("body", body);
p.put("nonce_str", nonce_str);
p.put("out_trade_no", out_trade_no);
p.put("total_fee", total_fee);
p.put("spbill_create_ip", wxPayProperties.getSpbill_create_ip());
p.put("notify_url", wxPayProperties.getNotify_url());
p.put("trade_type", WxContants.TRADE_TYPE);
p.put("sub_openid", openId);
// 签名
String sign = SignUtils.createSignByMd5(p, wxPayProperties.getKey());
preOrder.setSign(sign);
String xml = XmlUtil.object2Xml(preOrder, PreOrder.class);
// 调用下单地址
String returnXml = HttpUtil.sendPost(WxContants.pay_url, xml);
System.out.println(returnXml);
// XML转换为Object
PreOrderResult preOrderResult = (PreOrderResult) XmlUtil.xml2Object(returnXml, PreOrderResult.class);
// XML转换为Object
// 一般企业开发中支付流水入库,支付状态更新这些都需要做,此出省略
return preOrderResult;
}
@Override
public PayResult getWxPayResult(HttpServletRequest request) throws Exception {
InputStream inStream = request.getInputStream();
BufferedReader in = null;
String result = "";
in = new BufferedReader(
new InputStreamReader(inStream));
String line;
while ((line = in.readLine()) != null) {
result += line;
}
PayResult pr = (PayResult)XmlUtil.xml2Object(result, PayResult.class);
System.out.println(pr.toString());
return pr;
}
}
小程序配置部分
@Component
public class WxPayProperties {
@Value("${wxpay.app_id}")
private String app_id;
@Value("${wxpay.sub_app_id}")
private String sub_app_id;
@Value("${wxpay.spbill_create_ip}")
private String spbill_create_ip;
@Value("${wxpay.key}")
private String key;
@Value("${wxpay.mch_id}")
private String mch_id;
@Value("${wxpay.sub_mch_id}")
private String sub_mch_id;
@Value("${wxpay.notify_url}")
private String notify_url;
// 此处省略get/set方法
...
}
返回数据
{
"appId": "wxe670bb9ea4775345",
"nonceStr": "536D9056202D4292A909392320E2E5BB",
"package": "prepay_id=wx13143641616855cfa3275610dd2a070000",
"returnCode": "SUCCESS",
"returnMsg": "OK",
"sign": "C512D4025134C356BFA58A2F5699E198",
"signType": "MD5",
"timeStamp": "1610519802"
}
小程序端根据后台返回的参数,拉起支付,代码如下:
wx.requestPayment({
'timeStamp': res.data.timeStamp,
'nonceStr': res.data.nonceStr,
'package': res.data.package,
'signType': res.data.signType,
'paySign': res.data.sign,
'success':function(res){},
'fail':function(res){},
'complete':function(res){}
})
点击支付,总算是来到了这一步:
过程中,可能会遇到如下问题:
出现这个错误的原因是签名不正确,多检查检查是哪一步出现了问题。
最后
一路踩了不少坑,总算还是成功了,因此将解决方法记录下来,后面做小程序支付功能的小伙伴可以避免踩坑。