/* * 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.boot.smo.payment.adapt.chinaums; import com.alibaba.fastjson.JSONArray; import com.alibaba.fastjson.JSONObject; import com.java110.boot.properties.WechatAuthProperties; import com.java110.boot.smo.DefaultAbstractComponentSMO; import com.java110.boot.smo.payment.adapt.IOweFeeToNotifyAdapt; import com.java110.core.factory.WechatFactory; import com.java110.core.log.LoggerFactory; import com.java110.dto.fee.FeeDto; import com.java110.dto.smallWeChat.SmallWeChatDto; import com.java110.utils.cache.CommonCache; import com.java110.utils.constant.CommonConstant; import com.java110.utils.util.BeanConvertUtil; import com.java110.utils.util.DateUtil; import com.java110.utils.util.PayUtil; import com.java110.utils.util.StringUtil; import org.apache.commons.codec.digest.DigestUtils; import org.slf4j.Logger; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.http.HttpMethod; import org.springframework.http.HttpStatus; import org.springframework.http.ResponseEntity; import org.springframework.stereotype.Component; 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 = "chinaUmsOweFeeToNotifyAdapt") public class ChinaUmsOweFeeToNotifyAdapt extends DefaultAbstractComponentSMO implements IOweFeeToNotifyAdapt { private static final Logger logger = LoggerFactory.getLogger(ChinaUmsOweFeeToNotifyAdapt.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()); } SortedMap paramMap = new TreeMap(); for (String key : map.keySet()) { if ("wId".equals(key)) { continue; } paramMap.put(key, map.get(key).toString()); } 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 orderId = billPayment.get("merOrderId").toString().substring(4); String outTradeNo = map.get("merOrderId").toString(); String orderId = outTradeNo.substring(4); String order = CommonCache.getAndRemoveValue(FeeDto.REDIS_PAY_OWE_FEE + orderId); if (StringUtil.isEmpty(order)) { return 1;// 说明已经处理过了 再不处理 } //查询用户ID JSONObject paramIn = JSONObject.parseObject(order); paramIn.put("oId", orderId); freshFees(paramIn); String url = "fee.payOweFee"; responseEntity = this.callCenterService(getHeaders("-1"), paramIn.toJSONString(), url, HttpMethod.POST); if (responseEntity.getStatusCode() != HttpStatus.OK) { return 0; } return 1; } private void freshFees(JSONObject paramIn) { if (!paramIn.containsKey("fees")) { return; } JSONArray fees = paramIn.getJSONArray("fees"); JSONObject fee = null; for (int feeIndex = 0; feeIndex < fees.size(); feeIndex++) { fee = fees.getJSONObject(feeIndex); if (fee.containsKey("deadlineTime")) { fee.put("startTime", fee.getString("endTime")); fee.put("endTime", fee.getString("deadlineTime")); fee.put("receivedAmount", fee.getString("feePrice")); fee.put("state", ""); } } } private Map getHeaders(String userId) { Map headers = new HashMap<>(); headers.put(CommonConstant.HTTP_APP_ID.toLowerCase(), APP_ID); headers.put(CommonConstant.HTTP_USER_ID.toLowerCase(), userId); headers.put(CommonConstant.HTTP_TRANSACTION_ID.toLowerCase(), UUID.randomUUID().toString()); headers.put(CommonConstant.HTTP_REQ_TIME.toLowerCase(), DateUtil.getDefaultFormateTimeString(new Date())); headers.put(CommonConstant.HTTP_SIGN.toLowerCase(), ""); return headers; } private SmallWeChatDto getSmallWechat(String appId) { ResponseEntity responseEntity = null; responseEntity = this.callCenterService(getHeaders("-1"), "", "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("签名过程中出现错误"); } } }