All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.github.binarywang.wxpay.util.SignUtils Maven / Gradle / Ivy

There is a newer version: 4.7.1.B
Show newest version
package com.github.binarywang.wxpay.util;

import com.github.binarywang.wxpay.bean.request.BaseWxPayRequest;
import com.github.binarywang.wxpay.bean.result.BaseWxPayResult;
import com.github.binarywang.wxpay.constant.WxPayConstants.SignType;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.thoughtworks.xstream.annotations.XStreamAlias;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.codec.digest.DigestUtils;
import org.apache.commons.lang3.ArrayUtils;
import org.apache.commons.lang3.StringUtils;

import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.util.*;

/**
 * 
 * 签名相关工具类.
 * Created by Binary Wang on 2017-3-23.
 * 
* * @author binarywang(Binary Wang) */ @Slf4j public class SignUtils { /** * 签名的时候不携带的参数 */ private static final List NO_SIGN_PARAMS = Lists.newArrayList("sign", "key", "xmlString", "xmlDoc", "couponList"); /** * 请参考并使用 {@link #createSign(Object, String, String, String[])}. * * @param xmlBean the xml bean * @param signKey the sign key * @return the string */ @Deprecated public static String createSign(Object xmlBean, String signKey) { return createSign(xmlBean2Map(xmlBean), signKey); } /** * 请参考并使用 {@link #createSign(Map, String, String, String[])} . * * @param params the params * @param signKey the sign key * @return the string */ @Deprecated public static String createSign(Map params, String signKey) { return createSign(params, SignType.MD5, signKey, new String[0]); } /** * 微信支付签名算法(详见:https://pay.weixin.qq.com/wiki/doc/api/tools/cash_coupon.php?chapter=4_3). * * @param xmlBean Bean里的属性如果存在XML注解,则使用其作为key,否则使用变量名 * @param signType 签名类型,如果为空,则默认为MD5 * @param signKey 签名Key * @param ignoredParams 签名时需要忽略的特殊参数 * @return 签名字符串 string */ public static String createSign(Object xmlBean, String signType, String signKey, String[] ignoredParams) { Map map = null; if (XmlConfig.fastMode) { if (xmlBean instanceof BaseWxPayRequest) { map = ((BaseWxPayRequest) xmlBean).getSignParams(); } } if (map == null) { map = xmlBean2Map(xmlBean); } return createSign(map, signType, signKey, ignoredParams); } /** * 微信支付签名算法(详见:https://pay.weixin.qq.com/wiki/doc/api/tools/cash_coupon.php?chapter=4_3). * * @param params 参数信息 * @param signType 签名类型,如果为空,则默认为MD5 * @param signKey 签名Key * @param ignoredParams 签名时需要忽略的特殊参数 * @return 签名字符串 string */ public static String createSign(Map params, String signType, String signKey, String[] ignoredParams) { StringBuilder toSign = new StringBuilder(); for (String key : new TreeMap<>(params).keySet()) { String value = params.get(key); boolean shouldSign = false; if (StringUtils.isNotEmpty(value) && !ArrayUtils.contains(ignoredParams, key) && !NO_SIGN_PARAMS.contains(key)) { shouldSign = true; } if (shouldSign) { toSign.append(key).append("=").append(value).append("&"); } } toSign.append("key=").append(signKey); if (SignType.HMAC_SHA256.equals(signType)) { return me.chanjar.weixin.common.util.SignUtils.createHmacSha256Sign(toSign.toString(), signKey); } else { return DigestUtils.md5Hex(toSign.toString()).toUpperCase(); } } /** * 企业微信签名 * * @param signType md5 目前接口要求使用的加密类型 */ public static String createEntSign(String actName, String mchBillNo, String mchId, String nonceStr, String reOpenid, Integer totalAmount, String wxAppId, String signKey, String signType) { Map sortedMap = new HashMap<>(8); sortedMap.put("act_name", actName); sortedMap.put("mch_billno", mchBillNo); sortedMap.put("mch_id", mchId); sortedMap.put("nonce_str", nonceStr); sortedMap.put("re_openid", reOpenid); sortedMap.put("total_amount", totalAmount + ""); sortedMap.put("wxappid", wxAppId); return toSignBuilder(sortedMap, signKey, signType); } /** * 企业微信签名 * @param signType md5 目前接口要求使用的加密类型 */ public static String createEntSign(Integer totalAmount, String appId, String description, String mchId, String nonceStr, String openid, String partnerTradeNo, String wwMsgType, String signKey, String signType) { Map sortedMap = new HashMap<>(8); sortedMap.put("amount", String.valueOf(totalAmount)); sortedMap.put("appid", appId); sortedMap.put("desc", description); sortedMap.put("mch_id", mchId); sortedMap.put("nonce_str", nonceStr); sortedMap.put("openid", openid); sortedMap.put("partner_trade_no", partnerTradeNo); sortedMap.put("ww_msg_type", wwMsgType); return toSignBuilder(sortedMap, signKey, signType); } private static String toSignBuilder(Map sortedMap, String signKey, String signType) { Iterator> iterator = new TreeMap<>(sortedMap).entrySet().iterator(); StringBuilder toSign = new StringBuilder(); while (iterator.hasNext()) { Map.Entry entry = iterator.next(); String value = entry.getValue(); boolean shouldSign = false; if (StringUtils.isNotEmpty(value)) { shouldSign = true; } if (shouldSign) { toSign.append(entry.getKey()).append("=").append(value).append("&"); } } //企业微信这里字段名不一样 toSign.append("secret=").append(signKey); if (SignType.HMAC_SHA256.equals(signType)) { return me.chanjar.weixin.common.util.SignUtils.createHmacSha256Sign(toSign.toString(), signKey); } else { return DigestUtils.md5Hex(toSign.toString()).toUpperCase(); } } /** * 校验签名是否正确. * * @param xmlBean Bean需要标记有XML注解 * @param signType 签名类型,如果为空,则默认为MD5 * @param signKey 校验的签名Key * @return true - 签名校验成功,false - 签名校验失败 */ public static boolean checkSign(Object xmlBean, String signType, String signKey) { return checkSign(xmlBean2Map(xmlBean), signType, signKey); } /** * 校验签名是否正确. * * @param params 需要校验的参数Map * @param signType 签名类型,如果为空,则默认为MD5 * @param signKey 校验的签名Key * @return true - 签名校验成功,false - 签名校验失败 */ public static boolean checkSign(Map params, String signType, String signKey) { String sign = createSign(params, signType, signKey, new String[0]); return sign.equals(params.get("sign")); } /** * 将bean按照@XStreamAlias标识的字符串内容生成以之为key的map对象. * * @param bean 包含@XStreamAlias的xml bean对象 * @return map对象 map */ public static Map xmlBean2Map(Object bean) { Map result = Maps.newHashMap(); List fields = new ArrayList<>(Arrays.asList(bean.getClass().getDeclaredFields())); fields.addAll(Arrays.asList(bean.getClass().getSuperclass().getDeclaredFields())); if (bean.getClass().getSuperclass().getSuperclass() == BaseWxPayRequest.class) { fields.addAll(Arrays.asList(BaseWxPayRequest.class.getDeclaredFields())); } if (bean.getClass().getSuperclass().getSuperclass() == BaseWxPayResult.class) { fields.addAll(Arrays.asList(BaseWxPayResult.class.getDeclaredFields())); } for (Field field : fields) { try { boolean isAccessible = field.isAccessible(); field.setAccessible(true); if (field.get(bean) == null) { field.setAccessible(isAccessible); continue; } if (field.isAnnotationPresent(XStreamAlias.class)) { result.put(field.getAnnotation(XStreamAlias.class).value(), field.get(bean).toString()); } else if (!Modifier.isStatic(field.getModifiers())) { //忽略掉静态成员变量 result.put(field.getName(), field.get(bean).toString()); } field.setAccessible(isAccessible); } catch (SecurityException | IllegalArgumentException | IllegalAccessException e) { log.error(e.getMessage(), e); } } return result; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy