WechatUtil.java 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  1. package com.kmall.common.utils.wechat;
  2. import com.alibaba.druid.support.logging.Log;
  3. import com.alibaba.druid.support.logging.LogFactory;
  4. import com.kmall.common.service.pay.wxpay.CommonWxPayPropertiesBuilder;
  5. import com.kmall.common.utils.CharUtil;
  6. import com.kmall.common.utils.MapUtils;
  7. import com.kmall.common.utils.ResourceUtil;
  8. import com.kmall.common.utils.XmlUtil;
  9. import org.apache.http.HttpEntity;
  10. import org.apache.http.HttpResponse;
  11. import org.apache.http.client.HttpClient;
  12. import org.apache.http.client.config.RequestConfig;
  13. import org.apache.http.client.methods.CloseableHttpResponse;
  14. import org.apache.http.client.methods.HttpGet;
  15. import org.apache.http.client.methods.HttpPost;
  16. import org.apache.http.config.RegistryBuilder;
  17. import org.apache.http.conn.socket.ConnectionSocketFactory;
  18. import org.apache.http.conn.socket.PlainConnectionSocketFactory;
  19. import org.apache.http.conn.ssl.SSLConnectionSocketFactory;
  20. import org.apache.http.entity.StringEntity;
  21. import org.apache.http.impl.client.CloseableHttpClient;
  22. import org.apache.http.impl.client.HttpClientBuilder;
  23. import org.apache.http.impl.client.HttpClients;
  24. import org.apache.http.impl.conn.BasicHttpClientConnectionManager;
  25. import org.apache.http.util.EntityUtils;
  26. import java.io.IOException;
  27. import java.io.UnsupportedEncodingException;
  28. import java.math.BigDecimal;
  29. import java.math.MathContext;
  30. import java.net.URLEncoder;
  31. import java.text.SimpleDateFormat;
  32. import java.util.*;
  33. /**
  34. * <p>Title: 微信退款工具类</p>
  35. * <p>Description: 微信退款工具类,通过充值客户端的不同初始化不同的工具类,得到相应微信退款相关的appid和muchid</p>
  36. *
  37. * @author xubo
  38. * @date 2017年6月6日 下午5:05:03
  39. */
  40. public class WechatUtil {
  41. private static Log logger = LogFactory.getLog(WechatUtil.class);
  42. /**
  43. * 充值客户端类型--微信公众号
  44. */
  45. public static Integer CLIENTTYPE_WX = 2;
  46. /**
  47. * 充值客户端类型--app
  48. */
  49. public static Integer CLIENTTYPE_APP = 1;
  50. private static final String EMPTY = "";
  51. private static final String URL_PARAM_CONNECT_FLAG = "&";
  52. /**
  53. * 方法描述:微信退款逻辑
  54. * 创建时间:2017年4月12日 上午11:04:25
  55. * 作者: xubo
  56. *
  57. * @param
  58. * @return
  59. */
  60. public static WechatRefundApiResult wxRefund(String out_trade_no, Double orderMoney, Double refundMoney) {
  61. //初始化请求微信服务器的配置信息包括appid密钥等
  62. //转换金钱格式
  63. BigDecimal bdOrderMoney = new BigDecimal(orderMoney, MathContext.DECIMAL32);
  64. BigDecimal bdRefundMoney = new BigDecimal(refundMoney, MathContext.DECIMAL32);
  65. //构建请求参数
  66. Map<Object, Object> params = buildRequsetMapParam(out_trade_no, bdOrderMoney, bdRefundMoney);
  67. String mapToXml = MapUtils.convertMap2Xml(params);
  68. //请求微信
  69. String reponseXml = sendSSLPostToWx(mapToXml, WechatConfig.getSslcsf());
  70. WechatRefundApiResult result = (WechatRefundApiResult) XmlUtil.xmlStrToBean(reponseXml, WechatRefundApiResult.class);
  71. return result;
  72. }
  73. /**
  74. * 方法描述:得到请求微信退款请求的参数
  75. * 创建时间:2017年6月8日 上午11:27:02
  76. * 作者: xubo
  77. *
  78. * @param
  79. * @return
  80. */
  81. private static Map<Object, Object> buildRequsetMapParam(String out_trade_no, BigDecimal bdOrderMoney, BigDecimal bdRefundMoney) {
  82. Map<Object, Object> params = new HashMap<Object, Object>();
  83. params.put("appid", CommonWxPayPropertiesBuilder.instance().getAppId());//微信分配的公众账号ID(企业号corpid即为此appId)
  84. params.put("mch_id", CommonWxPayPropertiesBuilder.instance().getMchId());//微信支付分配的商户号
  85. params.put("nonce_str", CharUtil.getRandomString(16));//随机字符串,不长于32位。推荐随机数生成算法
  86. params.put("out_trade_no", out_trade_no);//商户传给微信的订单号
  87. params.put("out_refund_no", System.currentTimeMillis()+"");//商户系统内部的退款单号,商户系统内部唯一,同一退款单号多次请求只退一笔
  88. params.put("total_fee", bdOrderMoney.multiply(new BigDecimal(100)).intValue());//订单总金额,单位为分,只能为整数
  89. params.put("refund_fee", bdRefundMoney.multiply(new BigDecimal(100)).intValue());//退款总金额,订单总金额,单位为分,只能为整数
  90. params.put("op_user_id", CommonWxPayPropertiesBuilder.instance().getMchId());//操作员帐号, 默认为商户号
  91. params.put("notify_url", CommonWxPayPropertiesBuilder.instance().getRefundNotifyUrl());
  92. //签名前必须要参数全部写在前面
  93. params.put("sign", arraySign(params, CommonWxPayPropertiesBuilder.instance().getPaySignKey()));//签名
  94. return params;
  95. }
  96. /**
  97. * 请求微信https
  98. **/
  99. public static String sendSSLPostToWx(String mapToXml, SSLConnectionSocketFactory sslcsf) {
  100. logger.info("*******退款(WX Request:" + mapToXml);
  101. System.out.println("*******退款(WX Request:" + mapToXml);
  102. HttpPost httPost = new HttpPost(CommonWxPayPropertiesBuilder.instance().getRefundUrl());
  103. httPost.addHeader("Connection", "keep-alive");
  104. httPost.addHeader("Accept", "*/*");
  105. httPost.addHeader("Content-Type", "application/x-www-form-urlencoded; charset=UTF-8");
  106. httPost.addHeader("Host", "api.mch.weixin.qq.com");
  107. httPost.addHeader("X-Requested-With", "XMLHttpRequest");
  108. httPost.addHeader("Cache-Control", "max-age=0");
  109. httPost.addHeader("User-Agent", "Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 6.0) ");
  110. httPost.setEntity(new StringEntity(mapToXml, "UTF-8"));
  111. CloseableHttpClient httpClient = HttpClients.custom().setSSLSocketFactory(sslcsf).build();
  112. CloseableHttpResponse response = null;
  113. try {
  114. response = httpClient.execute(httPost);
  115. HttpEntity entity = response.getEntity();
  116. String xmlStr = EntityUtils.toString(entity, "UTF-8");
  117. logger.info("*******退款(WX Response:" + xmlStr);
  118. System.out.println("*******退款(WX Response:" + xmlStr);
  119. return xmlStr;
  120. } catch (Exception e) {
  121. logger.error(e.getMessage(), e);
  122. return null;
  123. } finally {
  124. try {
  125. if (response != null) {
  126. response.close();
  127. }
  128. } catch (IOException e) {
  129. logger.error(e.getMessage(), e);
  130. }
  131. }
  132. }
  133. /**
  134. * 方法描述:微信查询退款逻辑
  135. * 创建时间:2017年4月12日 上午11:04:25
  136. * 作者: xubo
  137. *
  138. * @param
  139. * @return
  140. */
  141. public static Map<String, Object> wxRefundquery(String out_trade_no, String out_refund_no) {
  142. Map<Object, Object> params = new HashMap<Object, Object>();
  143. params.put("appid", CommonWxPayPropertiesBuilder.instance().getAppId());//微信分配的公众账号ID(企业号corpid即为此appId)
  144. params.put("mch_id", CommonWxPayPropertiesBuilder.instance().getMchId());//微信支付分配的商户号
  145. params.put("nonce_str", CharUtil.getRandomString(16));//随机字符串,不长于32位。推荐随机数生成算法
  146. params.put("out_trade_no", out_trade_no);//商户侧传给微信的订单号
  147. //签名前必须要参数全部写在前面
  148. params.put("sign", arraySign(params, CommonWxPayPropertiesBuilder.instance().getPaySignKey()));//签名
  149. String mapToXml = MapUtils.convertMap2Xml(params);
  150. HttpPost httPost = new HttpPost(CommonWxPayPropertiesBuilder.instance().getRefundqueryUrl());
  151. httPost.addHeader("Connection", "keep-alive");
  152. httPost.addHeader("Accept", "*/*");
  153. httPost.addHeader("Content-Type", "application/x-www-form-urlencoded; charset=UTF-8");
  154. httPost.addHeader("Host", "api.mch.weixin.qq.com");
  155. httPost.addHeader("X-Requested-With", "XMLHttpRequest");
  156. httPost.addHeader("Cache-Control", "max-age=0");
  157. httPost.addHeader("User-Agent", "Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 6.0) ");
  158. httPost.setEntity(new StringEntity(mapToXml, "UTF-8"));
  159. CloseableHttpClient httpClient = HttpClients.custom().setSSLSocketFactory(WechatConfig.getSslcsf()).build();
  160. CloseableHttpResponse response = null;
  161. try {
  162. response = httpClient.execute(httPost);
  163. HttpEntity entity = response.getEntity();
  164. String xmlStr = EntityUtils.toString(entity, "UTF-8");
  165. System.out.println(xmlStr);
  166. Map<String, Object> result = XmlUtil.xmlStrToMap(xmlStr);//.xmlStrToBean(xmlStr, WechatRefundApiResult.class);
  167. return result;
  168. //将信息保存到数据库
  169. } catch (Exception e) {
  170. logger.error(e.getMessage(), e);
  171. return null;
  172. } finally {
  173. try {
  174. if (response != null) {
  175. response.close();
  176. }
  177. } catch (IOException e) {
  178. logger.error(e.getMessage(), e);
  179. }
  180. }
  181. }
  182. /**
  183. * 支付交易ID
  184. */
  185. public static String getBundleId() {
  186. SimpleDateFormat dateFormat = new SimpleDateFormat("yyyyMMddHHmmssSSS");
  187. String tradeno = dateFormat.format(new Date());
  188. String str = "000000" + (int) (Math.random() * 1000000);
  189. tradeno = tradeno + str.substring(str.length() - 6);
  190. return tradeno;
  191. }
  192. /**
  193. * 方法描述:根据签名加密请求参数
  194. * 创建时间:2017年6月8日 上午11:28:52
  195. * 作者: xubo
  196. *
  197. * @param
  198. * @return
  199. */
  200. public static String arraySign(Map<Object, Object> params, String paySignKey) {
  201. boolean encode = false;
  202. Set<Object> keysSet = params.keySet();
  203. Object[] keys = keysSet.toArray();
  204. Arrays.sort(keys);
  205. StringBuffer temp = new StringBuffer();
  206. boolean first = true;
  207. for (Object key : keys) {
  208. if (first) {
  209. first = false;
  210. } else {
  211. temp.append("&");
  212. }
  213. temp.append(key).append("=");
  214. Object value = params.get(key);
  215. String valueString = "";
  216. if (null != value) {
  217. valueString = value.toString();
  218. }
  219. if (encode) {
  220. try {
  221. temp.append(URLEncoder.encode(valueString, "UTF-8"));
  222. } catch (UnsupportedEncodingException e) {
  223. e.printStackTrace();
  224. }
  225. } else {
  226. temp.append(valueString);
  227. }
  228. }
  229. temp.append("&key=");
  230. temp.append(paySignKey);
  231. System.out.println(temp.toString());
  232. String packageSign = MD5.getMessageDigest(temp.toString());
  233. return packageSign;
  234. }
  235. /**
  236. * 请求,只请求一次,不做重试
  237. *
  238. * @param url
  239. * @param data
  240. * @return
  241. * @throws Exception
  242. */
  243. public static String requestOnce(final String url, String data) {
  244. BasicHttpClientConnectionManager connManager;
  245. connManager = new BasicHttpClientConnectionManager(
  246. RegistryBuilder.<ConnectionSocketFactory>create()
  247. .register("http", PlainConnectionSocketFactory.getSocketFactory())
  248. .register("https", SSLConnectionSocketFactory.getSocketFactory())
  249. .build(),
  250. null,
  251. null,
  252. null
  253. );
  254. HttpClient httpClient = HttpClientBuilder.create()
  255. .setConnectionManager(connManager)
  256. .build();
  257. HttpPost httpPost = new HttpPost(url);
  258. RequestConfig requestConfig = RequestConfig.custom()
  259. .setSocketTimeout(5000)
  260. .setConnectTimeout(5000)
  261. .setConnectionRequestTimeout(10000).build();
  262. httpPost.setConfig(requestConfig);
  263. StringEntity postEntity = new StringEntity(data, "UTF-8");
  264. httpPost.addHeader("Content-Type", "text/xml");
  265. // httpPost.addHeader("User-Agent", "wxpay sdk java v1.0 " + CommonWxPayPropertiesBuilder.instance().getMchId());
  266. httpPost.addHeader("User-Agent", "wxpay sdk java v1.0 " + "1517534731");
  267. httpPost.setEntity(postEntity);
  268. try {
  269. HttpResponse httpResponse = httpClient.execute(httpPost);
  270. HttpEntity httpEntity = httpResponse.getEntity();
  271. String reusltObj = null;
  272. reusltObj = EntityUtils.toString(httpEntity, "UTF-8");
  273. logger.info("请求结果:" + reusltObj);
  274. return reusltObj;
  275. } catch (IOException e) {
  276. e.printStackTrace();
  277. }
  278. return "";
  279. }
  280. /**
  281. * 请求,只请求一次,不做重试
  282. *
  283. * @param url
  284. * @param params
  285. * @return
  286. * @throws Exception
  287. */
  288. public static String requestOnceGet(final String url, Map params) {
  289. StringBuffer strtTotalURL = new StringBuffer(EMPTY);
  290. BasicHttpClientConnectionManager connManager;
  291. connManager = new BasicHttpClientConnectionManager(
  292. RegistryBuilder.<ConnectionSocketFactory>create()
  293. .register("http", PlainConnectionSocketFactory.getSocketFactory())
  294. .register("https", SSLConnectionSocketFactory.getSocketFactory())
  295. .build(),
  296. null,
  297. null,
  298. null
  299. );
  300. HttpClient httpClient = HttpClientBuilder.create()
  301. .setConnectionManager(connManager)
  302. .build();
  303. HttpGet httpGet = new HttpGet(url);
  304. RequestConfig requestConfig = RequestConfig.custom()
  305. .setSocketTimeout(5000)
  306. .setConnectTimeout(5000)
  307. .setConnectionRequestTimeout(10000).build();
  308. httpGet.setConfig(requestConfig);
  309. httpGet.addHeader("Content-Type", "text/xml");
  310. // httpGet.addHeader("User-Agent", "wxpay sdk java v1.0 " + CommonWxPayPropertiesBuilder.instance().getMchId());
  311. httpGet.addHeader("User-Agent", "wxpay sdk java v1.0 " + "1517534731");
  312. if (strtTotalURL.indexOf("?") == -1) {
  313. strtTotalURL.append(url).append("?").append(getUrl(params, "UTF-8"));
  314. } else {
  315. strtTotalURL.append(url).append("&").append(getUrl(params, "UTF-8"));
  316. }
  317. try {
  318. HttpResponse httpResponse = httpClient.execute(httpGet);
  319. HttpEntity httpEntity = httpResponse.getEntity();
  320. String reusltObj = EntityUtils.toString(httpEntity, "UTF-8");
  321. logger.info("请求结果:" + reusltObj);
  322. return reusltObj;
  323. } catch (Exception e) {
  324. e.printStackTrace();
  325. }
  326. return "";
  327. }
  328. /**
  329. * 据Map生成URL字符串
  330. *
  331. * @param map Map
  332. * @param valueEnc URL编码
  333. * @return URL
  334. */
  335. private static String getUrl(Map<String, String> map, String valueEnc) {
  336. if (null == map || map.keySet().size() == 0) {
  337. return (EMPTY);
  338. }
  339. StringBuffer url = new StringBuffer();
  340. Set<String> keys = map.keySet();
  341. for (Iterator<String> it = keys.iterator(); it.hasNext(); ) {
  342. String key = it.next();
  343. if (map.containsKey(key)) {
  344. String val = map.get(key);
  345. String str = val != null ? val : EMPTY;
  346. try {
  347. str = URLEncoder.encode(str, valueEnc);
  348. } catch (UnsupportedEncodingException e) {
  349. e.printStackTrace();
  350. }
  351. url.append(key).append("=").append(str).append(URL_PARAM_CONNECT_FLAG);
  352. }
  353. }
  354. String strURL = EMPTY;
  355. strURL = url.toString();
  356. if (URL_PARAM_CONNECT_FLAG.equals(EMPTY + strURL.charAt(strURL.length() - 1))) {
  357. strURL = strURL.substring(0, strURL.length() - 1);
  358. }
  359. return (strURL);
  360. }
  361. public static void main(String[] args) throws Exception {
  362. Map<Object, Object> parame = new TreeMap<Object, Object>();
  363. parame.put("mch_id", ResourceUtil.getConfigByName("\\conf\\wx-mp", "wx.mchId"));//
  364. String randomStr = CharUtil.getRandomNum(18).toUpperCase();
  365. parame.put("nonce_str", randomStr);//
  366. String sign = WechatUtil.arraySign(parame, ResourceUtil.getConfigByName("\\conf\\wx-mp", "wx.paySignKey"));
  367. parame.put("sign", sign);// 数字签证
  368. String xml = MapUtils.convertMap2Xml(parame);
  369. logger.info("xml:" + xml);
  370. Map<String, Object> resultUn = XmlUtil.xmlStrToMap(WechatUtil.requestOnce("https://apitest.mch.weixin.qq.com/sandboxnew/pay/getsignkey", xml));
  371. System.out.print(resultUn);
  372. }
  373. }