/* * 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.plutuspay; import com.alibaba.fastjson.JSON; 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.CommunitySettingFactory; import com.java110.core.factory.PlutusFactory; 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.bouncycastle.util.encoders.Base64; 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.Date; import java.util.HashMap; import java.util.Map; import java.util.UUID; /** * 富友 支付 通知实现 * 说明:信息通过 http 或 https 形式 post 请求递交给前置系统,编码必须为 UTF-8 * Json 格式参数名:如下表 * 参数值:如下表 * 测试地址:商户提供 * 生产地址:待定 *

* 如图中第 6 步中异步回调,下单(主扫)交易的结果是以异步的形式进行回调的。富友在接受到支付宝等支付通道的回调结果以 * 后再回调商户。商户接收回调成功处理成功后返回字符串”1” , 后富友停止回调给商户。最多回调 5 次,每次间隔 30S。 * (重要~重要~重要:不保证通知最终一定能成功,在订单状态不明或者没有收到微信,支付结果通知的情况下, * 建议商户主动调用【2.3 订单查询】确认订单状态) * 只有主扫、公众号/服务窗支付会通过此接口发异步通知,条码支付没有异步通知。 * * @desc add by 吴学文 15:33 */ @Component(value = "plutusOweFeeToNotifyAdapt") public class PlutusOweFeeToNotifyAdapt extends DefaultAbstractComponentSMO implements IOweFeeToNotifyAdapt { private static final Logger logger = LoggerFactory.getLogger(PlutusOweFeeToNotifyAdapt.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 json = JSON.parseObject(param); String signature = json.getString("signature"); String content = json.getString("content"); String appId = WechatFactory.getAppId(wId); SmallWeChatDto smallWeChatDto = getSmallWechat(appId); if (smallWeChatDto == null) { throw new IllegalArgumentException("未配置公众号或者小程序信息"); } String publicKey = CommunitySettingFactory.getRemark(smallWeChatDto.getObjId(), "PLUTUS_PUBLIC_KEY"); //验签 Boolean verify = PlutusFactory.verify256(content, Base64.decode(signature), publicKey); //验签成功 if (!verify) { throw new IllegalArgumentException("支付失败签名失败"); } //解密 byte[] bb = PlutusFactory.decrypt(Base64.decode(content), smallWeChatDto.getPayPassword()); //服务器返回内容 String paramOut = new String(bb); try { JSONObject map = JSONObject.parseObject(paramOut); logger.info("【银联支付回调】 回调数据: \n" + map); //更新数据 int result = confirmPayFee(map, wId); if (result > 0) { //支付成功 return "SUCCESS"; } } catch (Exception e) { logger.error("通知失败", e); return "ERROR"; } return "ERROR"; } 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()); } //String sign = PayUtil.createChinaUmsSign(paramMap, smallWeChatDto.getPayPassword()); //JSONObject billPayment = JSONObject.parseObject(map.getString("billPayment")); String outTradeNo = map.getString("outTransId"); String orderId = outTradeNo; 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("签名过程中出现错误"); } } }