/*
* 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("签名过程中出现错误");
}
}
}