微信官方支付验签源码分析

1.背景

随着微信的迅速崛起,在互联网支付的方式中,微信支付成了举足轻重的一部分。作为程序员,在朝着互联网靠拢的途中,了解微信支付必不可少。此处,笔者分享一下微信官方对于微信回调通知返回的xml数据进行支付验证签名的处理。

2.源码分析

1.官方地址:https://pay.weixin.qq.com/wiki/doc/api/download/WxPayAPI_JAVA_v3.zip

2.源码分析(注释解释)

/**
     * 判断签名是否正确
     *
     * @param xmlStr XML格式数据
     * @param key API密钥
     * @return 签名是否正确
     * @throws Exception
     */
 public static boolean isSignatureValid(String xmlStr, String key) throws Exception {
    //将xml格式数据转化为map格式   
    Map<String, String> data = xmlToMap(xmlStr);
        if (!data.containsKey(WXPayConstants.FIELD_SIGN) ) {
            //如果返回xml数据中不包含sign签名标记数据,则直接返回false
            return false;
        }
        //获取微信返回数据中的sign签名数据
        String sign = data.get(WXPayConstants.FIELD_SIGN);
        //将data和key进行签名组装,与返回数据中的sign签名数据对比
        return generateSignature(data, key).equals(sign);
    }
/**
     * 生成签名
     *
     * @param data 待签名数据
     * @param key API密钥
     * @return 签名
     */
    public static String generateSignature(final Map<String, String> data, String key) throws Exception {
        return generateSignature(data, key, SignType.MD5);
    }
 /**
     * 生成签名. 注意,若含有sign_type字段,必须和signType参数保持一致。
     *
     * @param data 待签名数据
     * @param key API密钥
     * @param signType 签名方式
     * @return 签名
     */
    public static String generateSignature(final Map<String, String> data, String key, SignType signType) throws Exception {
        //通过keySet获取所有的key集合
        Set<String> keySet = data.keySet();
        //将set转化为数组keyArray
        String[] keyArray = keySet.toArray(new String[keySet.size()]);
        //数组升序排序
        Arrays.sort(keyArray);
        //构建StringBuilder字符串变量
        StringBuilder sb = new StringBuilder();
        //for循环key数组
        for (String k : keyArray) {
            //(重点1)如果数组中包含sign,则继续,不做字符串拼接操作
            if (k.equals(WXPayConstants.FIELD_SIGN)) {
                continue;
            }
             // 参数值为空,则不参与签名
            if (data.get(k).trim().length() > 0)
                //字符串拼接形式:key1=value1&key2=value2
                sb.append(k).append("=").append(data.get(k).trim()).append("&");
        }
        //(重点2)拼接密钥,参数上传的密钥
        sb.append("key=").append(key);
        //如果签名加密方式为MD5,则将字符串所有的英文字符转换为大写字母,再做MD5编码,返回md5加密结果
        if (SignType.MD5.equals(signType)) {
            //(重点3)返回加密结果字符串
            return MD5(sb.toString()).toUpperCase();
        }
        //如果签名加密方式为HMACSHA256,则直接将字符串和key密钥直接生成 HMACSHA256
        else if (SignType.HMACSHA256.equals(signType)) {
            //(重点3)返回加密结果字符串
            return HMACSHA256(sb.toString(), key);
        }
        //如果是其他加密方式,则报异常
        else {
            throw new Exception(String.format("Invalid sign_type: %s", signType));
        }
    }

3.回顾

思想:分析微信支付回调通知中的数据,将签名sign过滤掉,替换成API支付密钥,然后做字符串拼接,做MD5或HMACSHA256加密,返回加密结果字符串的一个逆向替换过程。再和微信数据中的sign签名做对比。相同则,验签通过。否则,不通过。

微信支付 SDK 验签的基本流程如下: 1. 从微信支付服务器获取支付结果通知(XML 格式)。 2. 将 XML 格式的支付结果通知转换成 Map 对象。 3. 对 Map 对象中的参数按照 key 值进行字典序排序。 4. 将排序后的参数拼接成一个字符串,格式为 key1=value1&key2=value2&...&keyn=valuen。 5. 将拼接后的字符串与商户密钥进行拼接,得到一个新的字符串。 6. 对新的字符串进行 MD5 签名,得到一个签名值。 7. 将签名值与支付结果通知中的 sign 字段进行比较,如果一致则代表验签通过。 以下是一个 Java 实现的示例代码: ```java import java.util.*; import java.security.*; import javax.xml.bind.DatatypeConverter; public class WxPay { // 商户密钥 private static final String KEY = "your_key_here"; // 验证微信支付通知 public static boolean verify(Map<String, String> params) { // 将参数按照 key 值进行字典序排序 List<String> keys = new ArrayList<String>(params.keySet()); Collections.sort(keys); // 拼接参数字符串 String paramStr = ""; for (String key : keys) { String value = params.get(key); if (!value.isEmpty() && !key.equals("sign")) { paramStr += key + "=" + value + "&"; } } paramStr += "key=" + KEY; // 对参数字符串进行 MD5 签名 String sign = null; try { MessageDigest md5 = MessageDigest.getInstance("MD5"); byte[] bytes = md5.digest(paramStr.getBytes("UTF-8")); sign = DatatypeConverter.printHexBinary(bytes).toLowerCase(); } catch (Exception e) { return false; } // 比较签名值 return sign.equals(params.get("sign")); } } ``` 调用示例: ```java // 获取支付结果通知 Map<String, String> params = ...; // 验证支付通知 if (WxPay.verify(params)) { // 验签通过,处理支付结果 } else { // 验签失败,忽略该支付通知 } ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值