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

com.github.binarywang.wxpay.bean.request.BaseWxPayRequest Maven / Gradle / Ivy

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

import com.github.binarywang.wxpay.config.WxPayConfig;
import com.github.binarywang.wxpay.exception.WxPayException;
import com.github.binarywang.wxpay.util.SignUtils;
import com.github.binarywang.wxpay.util.XmlConfig;
import com.thoughtworks.xstream.XStream;
import com.thoughtworks.xstream.annotations.XStreamAlias;
import lombok.Data;
import lombok.experimental.Accessors;
import me.chanjar.weixin.common.error.WxErrorException;
import me.chanjar.weixin.common.error.WxRuntimeException;
import me.chanjar.weixin.common.util.BeanUtils;
import me.chanjar.weixin.common.util.json.WxGsonBuilder;
import me.chanjar.weixin.common.util.xml.XStreamInitializer;
import org.apache.commons.lang3.StringUtils;
import org.dom4j.Document;
import org.dom4j.DocumentHelper;
import org.dom4j.Element;

import java.io.Serializable;
import java.math.BigDecimal;
import java.util.HashMap;
import java.util.Map;
import java.util.Optional;

import static com.github.binarywang.wxpay.constant.WxPayConstants.SignType.ALL_SIGN_TYPES;

/**
 * 
 *  微信支付请求对象共用的参数存放类
 * Created by Binary Wang on 2016-10-24.
 * 
* * @author Binary Wang */ @Data @Accessors(chain = true) public abstract class BaseWxPayRequest implements Serializable { private static final long serialVersionUID = -4766915659779847060L; /** *
   * 字段名:公众账号ID.
   * 变量名:appid
   * 是否必填:是
   * 类型:String(32)
   * 示例值:wxd678efh567hg6787
   * 描述:微信分配的公众账号ID(企业号corpid即为此appId)
   * 
*/ @XStreamAlias("appid") protected String appid; /** *
   * 字段名:商户号.
   * 变量名:mch_id
   * 是否必填:是
   * 类型:String(32)
   * 示例值:1230000109
   * 描述:微信支付分配的商户号
   * 
*/ @XStreamAlias("mch_id") protected String mchId; /** *
   * 字段名:服务商模式下的子商户公众账号ID.
   * 变量名:sub_appid
   * 是否必填:是
   * 类型:String(32)
   * 示例值:wxd678efh567hg6787
   * 描述:微信分配的子商户公众账号ID
   * 
*/ @XStreamAlias("sub_appid") protected String subAppId; /** *
   * 字段名:服务商模式下的子商户号.
   * 变量名:sub_mch_id
   * 是否必填:是
   * 类型:String(32)
   * 示例值:1230000109
   * 描述:微信支付分配的子商户号,开发者模式下必填
   * 
*/ @XStreamAlias("sub_mch_id") protected String subMchId; /** *
   * 字段名:随机字符串.
   * 变量名:nonce_str
   * 是否必填:是
   * 类型:String(32)
   * 示例值:5K8264ILTKCH16CQ2502SI8ZNMTM67VS
   * 描述:随机字符串,不长于32位。推荐随机数生成算法
   * 
*/ @XStreamAlias("nonce_str") protected String nonceStr; /** *
   * 字段名:签名.
   * 变量名:sign
   * 是否必填:是
   * 类型:String(32)
   * 示例值:C380BEC2BFD727A4B6845133519F3AD6
   * 描述:签名,详见签名生成算法
   * 
*/ @XStreamAlias("sign") protected String sign; /** *
   * 签名类型.
   * sign_type
   * 否
   * String(32)
   * HMAC-SHA256
   * 签名类型,目前支持HMAC-SHA256和MD5
   * 
*/ @XStreamAlias("sign_type") private String signType; /** * 企业微信签名 */ @XStreamAlias("workwx_sign") private String workWxSign; public String getWorkWxSign() { return workWxSign; } public void setWorkWxSign(String workWxSign) { this.workWxSign = workWxSign; } /** * 将单位为元转换为单位为分. * * @param yuan 将要转换的元的数值字符串 * @return the integer */ public static Integer yuanToFen(String yuan) { return new BigDecimal(yuan).setScale(2, BigDecimal.ROUND_HALF_UP).multiply(new BigDecimal(100)).intValue(); } /** * 元转分 */ public static Integer yuan2Fen(BigDecimal yuan) { return yuan.multiply(BigDecimal.valueOf(100)).setScale(2, BigDecimal.ROUND_HALF_UP).intValue(); } /** * 分转元 */ public static BigDecimal fen2Yuan(BigDecimal fen) { return fen.divide(BigDecimal.valueOf(100)).setScale(2, BigDecimal.ROUND_HALF_UP); } /** * 检查请求参数内容,包括必填参数以及特殊约束. */ private void checkFields() throws WxPayException { //check required fields try { BeanUtils.checkRequiredFields(this); } catch (WxErrorException e) { throw new WxPayException(e.getError().getErrorMsg(), e); } //check other parameters this.checkConstraints(); } /** * 检查约束情况. * * @throws WxPayException the wx pay exception */ protected abstract void checkConstraints() throws WxPayException; /** * 是否需要nonce_str */ protected boolean needNonceStr() { return true; } /** * 如果配置中已经设置,可以不设置值. * * @param appid 微信公众号appid */ public void setAppid(String appid) { this.appid = appid; } /** * 如果配置中已经设置,可以不设置值. * * @param mchId 微信商户号 */ public void setMchId(String mchId) { this.mchId = mchId; } /** * 默认采用时间戳为随机字符串,可以不设置. * * @param nonceStr 随机字符串 */ public void setNonceStr(String nonceStr) { this.nonceStr = nonceStr; } @Override public String toString() { return WxGsonBuilder.create().toJson(this); } /** * To xml string. * * @return the string */ public String toXML() { //涉及到服务商模式的两个参数,在为空值时置为null,以免在请求时将空值传给微信服务器 this.setSubAppId(StringUtils.trimToNull(this.getSubAppId())); this.setSubMchId(StringUtils.trimToNull(this.getSubMchId())); if (XmlConfig.fastMode) { return toFastXml(); } XStream xstream = XStreamInitializer.getInstance(); xstream.processAnnotations(this.getClass()); return xstream.toXML(this); } /** * 使用快速算法组装xml */ private String toFastXml() { try { Document document = DocumentHelper.createDocument(); Element root = document.addElement(xmlRootTagName()); Map signParams = getSignParams(); signParams.put("sign", sign); for (Map.Entry entry : signParams.entrySet()) { if (entry.getValue() == null) { continue; } Element elm = root.addElement(entry.getKey()); elm.addText(entry.getValue()); } return document.asXML(); } catch (Exception e) { throw new WxRuntimeException("generate xml error", e); } } /** * 返回xml结构的根节点名称 * * @return 默认返回"xml", 特殊情况可以在子类中覆盖 */ protected String xmlRootTagName() { return "xml"; } /** * 签名时,是否忽略appid. * * @return the boolean */ protected boolean ignoreAppid() { return false; } /** * 签名时,是否忽略sub_appid. * * @return the boolean */ protected boolean ignoreSubAppId() { return false; } protected boolean ignoreSubMchId() { return false; } /** * 是否是企业微信字段 */ protected boolean isWxWorkSign() { return false; } /** * 签名时,忽略的参数. * * @return the string [ ] */ protected String[] getIgnoredParamsForSign() { return new String[0]; } /** * 获取签名时需要的参数. * 注意:不含sign属性 */ public Map getSignParams() { Map map = new HashMap<>(8); map.put("appid", appid); map.put("mch_id", mchId); map.put("sub_appid", subAppId); map.put("sub_mch_id", subMchId); map.put("nonce_str", nonceStr); map.put("sign_type", signType); storeMap(map); return map; } /** * 将属性组装到一个Map中,供签名和最终发送XML时使用. * 这里需要将所有的属性全部保存进来,签名的时候会自动调用getIgnoredParamsForSign进行忽略, * 不用担心。否则最终生成的XML会缺失。 * * @param map 传入的属性Map */ protected abstract void storeMap(Map map); /** *
   * 检查参数,并设置签名.
   * 1、检查参数(注意:子类实现需要检查参数的而外功能时,请在调用父类的方法前进行相应判断)
   * 2、补充系统参数,如果未传入则从配置里读取
   * 3、生成签名,并设置进去
   * 
* * @param config 支付配置对象,用于读取相应系统配置信息 * @throws WxPayException the wx pay exception */ public void checkAndSign(WxPayConfig config) throws WxPayException { this.checkFields(); if (!ignoreAppid()) { if (StringUtils.isBlank(getAppid())) { this.setAppid(config.getAppId()); } } if (StringUtils.isBlank(getMchId())) { this.setMchId(config.getMchId()); } if (!ignoreSubAppId()) { if (StringUtils.isBlank(getSubAppId())) { this.setSubAppId(config.getSubAppId()); } } if (!ignoreSubMchId()) { if (StringUtils.isBlank(getSubMchId())) { this.setSubMchId(config.getSubMchId()); } } if (StringUtils.isBlank(getSignType())) { if (config.getSignType() != null && !ALL_SIGN_TYPES.contains(config.getSignType())) { throw new WxPayException("非法的signType配置:" + config.getSignType() + ",请检查配置!"); } this.setSignType(StringUtils.trimToNull(config.getSignType())); } else { if (!ALL_SIGN_TYPES.contains(this.getSignType())) { throw new WxPayException("非法的sign_type参数:" + this.getSignType()); } } if (needNonceStr() && StringUtils.isBlank(getNonceStr())) { this.setNonceStr(String.valueOf(System.currentTimeMillis())); } //设置签名字段的值 this.setSign(SignUtils.createSign(this, this.getSignType(), config.getMchKey(), this.getIgnoredParamsForSign())); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy