/* * Copyright 2017-2020 吴学文 and java110 team. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.java110.front.smo.payment.adapt.chinaums; import com.alibaba.fastjson.JSONArray; import com.alibaba.fastjson.JSONObject; import com.java110.core.factory.WechatFactory; import com.java110.dto.smallWeChat.SmallWeChatDto; import com.java110.front.properties.WechatAuthProperties; import com.java110.front.smo.payment.adapt.IPayNotifyAdapt; import com.java110.utils.constant.CommonConstant; import com.java110.utils.constant.ServiceConstant; import com.java110.utils.util.BeanConvertUtil; import com.java110.utils.util.DateUtil; import com.java110.utils.util.PayUtil; import org.apache.commons.codec.digest.DigestUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.http.*; import org.springframework.stereotype.Component; import org.springframework.web.client.HttpStatusCodeException; import org.springframework.web.client.RestTemplate; import java.io.UnsupportedEncodingException; import java.util.*; /** * 富友 支付 通知实现 * 说明:信息通过 http 或 https 形式 post 请求递交给前置系统,编码必须为 UTF-8 * Json 格式参数名:如下表 * 参数值:如下表 * 测试地址:商户提供 * 生产地址:待定 *

* 如图中第 6 步中异步回调,下单(主扫)交易的结果是以异步的形式进行回调的。富友在接受到支付宝等支付通道的回调结果以 * 后再回调商户。商户接收回调成功处理成功后返回字符串”1” , 后富友停止回调给商户。最多回调 5 次,每次间隔 30S。 * (重要~重要~重要:不保证通知最终一定能成功,在订单状态不明或者没有收到微信,支付结果通知的情况下, * 建议商户主动调用【2.3 订单查询】确认订单状态) * 只有主扫、公众号/服务窗支付会通过此接口发异步通知,条码支付没有异步通知。 * * @desc add by 吴学文 15:33 */ @Component(value = "chinaUmsPayNotifyAdapt") public class ChinaUmsPayNotifyAdapt implements IPayNotifyAdapt { private static final Logger logger = LoggerFactory.getLogger(ChinaUmsPayNotifyAdapt.class); private static final String APP_ID = "992020011134400001"; @Autowired private RestTemplate restTemplate; @Autowired private WechatAuthProperties wechatAuthProperties; /** * 预下单 * * @param param * @return * @throws Exception */ public String confirmPayFee(String param, String wId) { JSONObject resJson = new JSONObject(); resJson.put("errCode", "INTERNAL_ERROR"); resJson.put("errMsg", "失败"); try { JSONObject map = JSONObject.parseObject(param); logger.info("【银联支付回调】 回调数据: \n" + map); //更新数据 int result = confirmPayFee(map, wId); if (result > 0) { //支付成功 resJson.put("errCode", "SUCCESS"); resJson.put("errMsg", "成功"); } } catch (Exception e) { logger.error("通知失败", e); resJson.put("result_msg", "鉴权失败"); } return resJson.toJSONString(); } public int confirmPayFee(JSONObject map, String wId) { wId = wId.replace(" ", "+"); ResponseEntity responseEntity = null; String appId = WechatFactory.getAppId(wId); SmallWeChatDto smallWeChatDto = getSmallWechat(appId); if (smallWeChatDto == null) { //从配置文件中获取 小程序配置信息 smallWeChatDto = new SmallWeChatDto(); smallWeChatDto.setAppId(wechatAuthProperties.getAppId()); smallWeChatDto.setAppSecret(wechatAuthProperties.getSecret()); smallWeChatDto.setMchId(wechatAuthProperties.getMchId()); smallWeChatDto.setPayPassword(wechatAuthProperties.getKey()); } TreeMap paramMap = new TreeMap(); for (String key : map.keySet()) { // if ("wId".equals(key)) { // continue; // } paramMap.put(key, map.get(key).toString()); } //String sign = PayUtil.createChinaUmsSign(paramMap, smallWeChatDto.getPayPassword()); String preSign = map.getString("preSign"); String text = preSign + smallWeChatDto.getPayPassword(); System.out.println("待签名字符串:" + text); String sign = DigestUtils.sha256Hex(getContentBytes(text)).toUpperCase(); if (!sign.equals(map.get("sign"))) { throw new IllegalArgumentException("鉴权失败"); } //JSONObject billPayment = JSONObject.parseObject(map.getString("billPayment")); String outTradeNo = map.get("merOrderId").toString(); //查询用户ID JSONObject paramIn = new JSONObject(); paramIn.put("oId", outTradeNo.substring(4)); String url = ServiceConstant.SERVICE_API_URL + "/api/fee.payFeeConfirm"; responseEntity = this.callCenterService(restTemplate, "-1", paramIn.toJSONString(), url, HttpMethod.POST); if (responseEntity.getStatusCode() != HttpStatus.OK) { return 0; } return 1; } //SJ=lJQi& //B7C091FCE2AFC3DDEE16DEDD04C234CF public static void main(String[] args) { JSONObject data = JSONObject.parseObject("{\"msgType\":\"wx.notify\",\"payTime\":\"2021-09-03 02:47:35\",\"buyerCashPayAmt\":\"100\",\"connectSys\":\"UNIONPAY\",\"sign\":\"B7C091FCE2AFC3DDEE16DEDD04C234CF\",\"merName\":\"青海德坤电力有限公司\",\"mid\":\"898630149000110\",\"invoiceAmount\":\"100\",\"settleDate\":\"2021-09-03\",\"billFunds\":\"现金:100\",\"buyerId\":\"otdJ_uCsgFQi-XigMpadM9gB4h0w\",\"mchntUuid\":\"2d9081bd76d235d20176da1bf4f62bc9\",\"tid\":\"CV5EW7IM\",\"instMid\":\"YUEDANDEFAULT\",\"receiptAmount\":\"100\",\"couponAmount\":\"0\",\"SJ\":\"lJQi\",\"targetOrderId\":\"4200001198202109032729935220\",\"signType\":\"MD5\",\"billFundsDesc\":\"现金支付1.00元。\",\"subBuyerId\":\"oBFo-5-xs50SKaC5hjYf2Ux_Ww2g\",\"orderDesc\":\"青海德坤电力有限公司\",\"seqId\":\"23332339885N\",\"merOrderId\":\"1017102021090304700052\",\"targetSys\":\"WXPay\",\"bankInfo\":\"OTHERS\",\"totalAmount\":\"100\",\"wId\":\"hFXywDBfLkpKik7ZLPlAsRUQ4qORS1n8\",\"createTime\":\"2021-09-03 02:47:29\",\"buyerPayAmount\":\"100\",\"notifyId\":\"2f02e4a2-b54f-4d48-9b8a-16c924a95c98\",\"subInst\":\"103800\",\"status\":\"TRADE_SUCCESS\"}"); TreeMap paramMap = new TreeMap(); for (String key : data.keySet()) { paramMap.put(key, data.get(key).toString()); } StringBuffer sb = new StringBuffer(); Set es = paramMap.entrySet(); Iterator it = es.iterator(); while (it.hasNext()) { Map.Entry entry = (Map.Entry) it.next(); String k = (String) entry.getKey(); if (entry.getValue() != null || !"".equals(entry.getValue())) { String v = String.valueOf(entry.getValue()); if (null != v && !"".equals(v) && !"sign".equals(k)) { sb.append(k + "=" + v + "&"); } } } String data1 = sb.toString().substring(0, sb.length() - 1) + "JkENP4taKmyH2aBsxXZbnpJDGZ7pBhasCKcYxpt7xyNP4QXS"; data1 = "bankInfo=OTHERS&billFunds=现金:100&billFundsDesc=现金支付1.00元。&buyerCashPayAmt=100&buyerId=otdJ_uCsgFQi-XigMpadM9gB4h0w&buyerPayAmount=100&connectSys=UNIONPAY&couponAmount=0&createTime=2021-09-03 02:47:29&instMid=YUEDANDEFAULT&invoiceAmount=100&mchntUuid=2d9081bd76d235d20176da1bf4f62bc9&merName=青海德坤电力有限公司&merOrderId=1017102021090304700052&mid=898630149000110&msgType=wx.notify¬ifyId=2f02e4a2-b54f-4d48-9b8a-16c924a95c98&orderDesc=青海德坤电力有限公司&payTime=2021-09-03 02:47:35&receiptAmount=100&seqId=23332339885N&settleDate=2021-09-03&signType=MD5&SJ=lJQi&status=TRADE_SUCCESS&subBuyerId=oBFo-5-xs50SKaC5hjYf2Ux_Ww2g&subInst=103800&targetOrderId=4200001198202109032729935220&targetSys=WXPay&tid=CV5EW7IM&totalAmount=100&wId=hFXywDBfLkpKik7ZLPlAsRUQ4qORS1n8JkENP4taKmyH2aBsxXZbnpJDGZ7pBhasCKcYxpt7xyNP4QXS"; //sb.append(key); logger.debug("加密前串:" + data1); String sign = PayUtil.md5(data1.toString()).toUpperCase(); System.out.printf("sign:" + sign); } /** * 调用中心服务 * * @return */ protected ResponseEntity callCenterService(RestTemplate restTemplate, String userId, String param, String url, HttpMethod httpMethod) { ResponseEntity responseEntity = null; HttpHeaders header = new HttpHeaders(); header.add(CommonConstant.HTTP_APP_ID.toLowerCase(), APP_ID); header.add(CommonConstant.HTTP_USER_ID.toLowerCase(), userId); header.add(CommonConstant.HTTP_TRANSACTION_ID.toLowerCase(), UUID.randomUUID().toString()); header.add(CommonConstant.HTTP_REQ_TIME.toLowerCase(), DateUtil.getDefaultFormateTimeString(new Date())); header.add(CommonConstant.HTTP_SIGN.toLowerCase(), ""); HttpEntity httpEntity = new HttpEntity(param, header); //logger.debug("请求中心服务信息,{}", httpEntity); try { responseEntity = restTemplate.exchange(url, httpMethod, httpEntity, String.class); } catch (HttpStatusCodeException e) { //这里spring 框架 在4XX 或 5XX 时抛出 HttpServerErrorException 异常,需要重新封装一下 responseEntity = new ResponseEntity(e.getResponseBodyAsString(), e.getStatusCode()); } catch (Exception e) { responseEntity = new ResponseEntity(e.getMessage(), HttpStatus.INTERNAL_SERVER_ERROR); } finally { logger.debug("请求地址为,{} 请求中心服务信息,{},中心服务返回信息,{}", url, httpEntity, responseEntity); } return responseEntity; } private SmallWeChatDto getSmallWechat(String appId) { ResponseEntity responseEntity = null; responseEntity = this.callCenterService(restTemplate, "-1", "", ServiceConstant.SERVICE_API_URL + "/api/smallWeChat.listSmallWeChats?appId=" + appId + "&page=1&row=1", HttpMethod.GET); if (responseEntity.getStatusCode() != HttpStatus.OK) { return null; } JSONObject smallWechatObj = JSONObject.parseObject(responseEntity.getBody().toString()); JSONArray smallWeChats = smallWechatObj.getJSONArray("smallWeChats"); if (smallWeChats == null || smallWeChats.size() < 1) { return null; } return BeanConvertUtil.covertBean(smallWeChats.get(0), SmallWeChatDto.class); } /** * 富友 生成sign 方法 * * @param paramMap * @param payPassword * @return */ private String createSign(JSONObject paramMap, String payPassword) { String str = paramMap.getString("mchnt_cd") + "|" + paramMap.getString("mchnt_order_no") + "|" + paramMap.getString("settle_order_amt") + "|" + paramMap.getString("order_amt") + "|" + paramMap.getString("txn_fin_ts") + "|" + paramMap.getString("reserved_fy_settle_dt") + "|" + paramMap.getString("random_str") + "|" + payPassword; return PayUtil.md5(str); } // 根据编码类型获得签名内容byte[] public static byte[] getContentBytes(String content) { try { return content.getBytes("UTF-8"); } catch (UnsupportedEncodingException e) { throw new RuntimeException("签名过程中出现错误"); } } }