package com.java110.acct.payment.adapt.fuiou;
|
|
import com.alibaba.fastjson.JSONObject;
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
import com.java110.acct.payment.IPaymentFactoryAdapt;
|
import com.java110.core.context.ICmdDataFlowContext;
|
import com.java110.core.factory.CommunitySettingFactory;
|
import com.java110.core.factory.WechatFactory;
|
import com.java110.core.log.LoggerFactory;
|
import com.java110.dto.app.AppDto;
|
import com.java110.dto.owner.OwnerAppUserDto;
|
import com.java110.dto.payment.NotifyPaymentOrderDto;
|
import com.java110.dto.payment.PaymentOrderDto;
|
import com.java110.dto.payment.PaymentPoolDto;
|
import com.java110.dto.payment.PaymentPoolValueDto;
|
import com.java110.dto.wechat.SmallWeChatDto;
|
import com.java110.intf.acct.IPaymentPoolV1InnerServiceSMO;
|
import com.java110.intf.acct.IPaymentPoolValueV1InnerServiceSMO;
|
import com.java110.intf.store.ISmallWechatV1InnerServiceSMO;
|
import com.java110.intf.user.IOwnerAppUserInnerServiceSMO;
|
import com.java110.utils.cache.MappingCache;
|
import com.java110.utils.cache.UrlCache;
|
import com.java110.utils.constant.MappingConstant;
|
import com.java110.utils.constant.WechatConstant;
|
import com.java110.utils.util.*;
|
import org.slf4j.Logger;
|
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Value;
|
import org.springframework.http.*;
|
import org.springframework.stereotype.Service;
|
import org.springframework.web.client.RestTemplate;
|
|
import java.util.*;
|
import java.util.stream.Collectors;
|
|
/**
|
* 富友 支付
|
* 此实现方式为 通过 富友支付 去下单不直接去掉微信
|
* <p>
|
* 商户调用此接口则用户可使用支付宝或微信进行支付。
|
* 本接口支持: 微信公众号、 微信小程序、 微信 APP, 支付宝服务窗等
|
* <p>
|
* <p>
|
* 说明:信息通过 http 或 https 形式 post 请求递交给前置系统,编码必须为 UTF-8
|
* Json 格式参数名:如下表
|
* 参数值:如下表
|
* 测试地址: https://aipaytest.fuioupay.com/aggregatePay/wxPreCreate
|
* 生产地址: https://aipay.fuioupay.com/aggregatePay/wxPreCreate
|
* 生产地址 2: https://aipay-xs.fuioupay.com/aggregatePay/wxPreCreate
|
* <p>
|
* 该接口常应用于聚合二维码(静态二维码、统一收款码、台卡等不同叫法),用户扫二维码进入微信公众号/支付宝服务窗
|
* /QQJS 页面,页面调此接口生成订单,接受订单参数后调起官方支付接口支付。详见公众号/服务窗对接流程
|
* 步骤 1:用户通过支付宝(服务窗)、微信(公众号)进入到商户 H5 页面,或者是通过扫描台卡进入。
|
* 步骤 2:用户选择商品、输入支付金额等进行下单支付
|
* 步骤 3:商户将订单信息发送给富友,返回支付信息(用于调起支付宝、微信的参数)。
|
* 步骤 4:商户拿到支付信息后调起微信或者支付宝进行支付
|
* 步骤 5:支付结果以回调(2.5)的方式通知到商户
|
*
|
* @desc add by 吴学文 15:33
|
*/
|
@Service("fuiouPaymentFactory")
|
public class FuiouPaymentFactoryAdapt implements IPaymentFactoryAdapt {
|
|
private static final Logger logger = LoggerFactory.getLogger(FuiouPaymentFactoryAdapt.class);
|
|
|
//微信支付
|
public static final String DOMAIN_WECHAT_PAY = "WECHAT_PAY";
|
// 微信服务商支付开关
|
public static final String WECHAT_SERVICE_PAY_SWITCH = "WECHAT_SERVICE_PAY_SWITCH";
|
|
//开关ON打开
|
public static final String WECHAT_SERVICE_PAY_SWITCH_ON = "ON";
|
|
|
private static final String WECHAT_SERVICE_APP_ID = "SERVICE_APP_ID";
|
|
private static final String WECHAT_SERVICE_MCH_ID = "SERVICE_MCH_ID";
|
|
public static final String TRADE_TYPE_NATIVE = "NATIVE";
|
public static final String TRADE_TYPE_JSAPI = "JSAPI";
|
public static final String TRADE_TYPE_MWEB = "MWEB";
|
public static final String TRADE_TYPE_APP = "APP";
|
public static final String TRADE_TYPE_LETPAY = "LETPAY";
|
|
@Value("${fuiou.pay.unified-order-url}")
|
public String PAY_UNIFIED_ORDER_URL;
|
@Value("${fuiou.pay.wx-order-url}")
|
public String PAY_WX_ORDER_URL;
|
private static final String VERSION = "1.0";
|
|
|
@Autowired
|
private ISmallWechatV1InnerServiceSMO smallWechatV1InnerServiceSMOImpl;
|
|
|
@Autowired
|
private IOwnerAppUserInnerServiceSMO ownerAppUserInnerServiceSMOImpl;
|
|
@Autowired
|
private RestTemplate outRestTemplate;
|
@Autowired
|
private IPaymentPoolV1InnerServiceSMO paymentPoolV1InnerServiceSMOImpl;
|
|
@Autowired
|
private IPaymentPoolValueV1InnerServiceSMO paymentPoolValueV1InnerServiceSMOImpl;
|
|
|
@Override
|
public Map java110Payment(PaymentOrderDto paymentOrderDto, JSONObject reqJson, ICmdDataFlowContext context) throws Exception {
|
|
SmallWeChatDto smallWeChatDto = getSmallWechat(reqJson);
|
|
smallWeChatDto.setAppId(reqJson.getString("appId"));
|
|
String appId = context.getReqHeaders().get("app-id");
|
String userId = context.getReqHeaders().get("user-id");
|
String tradeType = reqJson.getString("tradeType");
|
String notifyUrl = UrlCache.getOwnerUrl() + "/app/payment/notify/wechat/"+appId+"/"+reqJson.getString("paymentPoolId");
|
|
String openId = reqJson.getString("openId");
|
|
if(StringUtil.isEmpty(openId)) {
|
//由于现在只有006,所以写死WECHAT,后续如果有多种支付方式则重新设计
|
// String appType = OwnerAppUserDto.APP_TYPE_WECHAT_MINA;
|
// if (AppDto.WECHAT_OWNER_APP_ID.equals(appId)) {
|
// appType = OwnerAppUserDto.APP_TYPE_WECHAT;
|
// } else if (AppDto.WECHAT_MINA_OWNER_APP_ID.equals(appId)) {
|
// appType = OwnerAppUserDto.APP_TYPE_WECHAT_MINA;
|
// } else {
|
// appType = OwnerAppUserDto.APP_TYPE_APP;
|
// }
|
String appType = OwnerAppUserDto.APP_TYPE_WECHAT;
|
OwnerAppUserDto ownerAppUserDto = new OwnerAppUserDto();
|
ownerAppUserDto.setUserId(userId);
|
ownerAppUserDto.setAppType(appType);
|
List<OwnerAppUserDto> ownerAppUserDtos = ownerAppUserInnerServiceSMOImpl.queryOwnerAppUsers(ownerAppUserDto);
|
|
// Assert.listOnlyOne(ownerAppUserDtos, "未找到开放账号信息");
|
try{
|
openId = ownerAppUserDtos.get(0).getOpenId();
|
}catch (Exception e){
|
|
}
|
}
|
|
|
logger.debug("【小程序支付】 统一下单开始, 订单编号=" + paymentOrderDto.getOrderId());
|
SortedMap<String, String> resultMap = new TreeMap<String, String>();
|
//生成支付金额,开发环境处理支付金额数到0.01、0.02、0.03元
|
double payAmount = PayUtil.getPayAmountByEnv(MappingCache.getValue(MappingConstant.ENV_DOMAIN,"HC_ENV"), paymentOrderDto.getMoney());
|
//添加或更新支付记录(参数跟进自己业务需求添加)
|
|
JSONObject resMap = null;
|
if (TRADE_TYPE_LETPAY.equals(tradeType)){
|
resMap = this.java110UnifieldOrder(paymentOrderDto.getName(),
|
paymentOrderDto.getOrderId(),
|
tradeType,
|
payAmount,
|
openId,
|
smallWeChatDto,
|
notifyUrl
|
);
|
}else{
|
|
resMap = this.java110UnifieldOrder(paymentOrderDto.getName(),
|
paymentOrderDto.getOrderId(),
|
tradeType,
|
payAmount,
|
openId,
|
smallWeChatDto,
|
notifyUrl
|
);
|
}
|
|
if ("000000".equals(resMap.getString("result_code"))) {
|
if (TRADE_TYPE_LETPAY.equals(tradeType)) {
|
resultMap.putAll(JSONObject.toJavaObject(JSONObject.parseObject(resMap.getString("reserved_pay_info")), Map.class));
|
resultMap.put("sign", resultMap.get("paySign"));
|
resultMap.put("timeStamp", resMap.get("sdk_timestamp").toString());
|
resultMap.put("nonceStr", resMap.get("sdk_noncestr").toString());
|
resultMap.put("package", resMap.get("sdk_package").toString());
|
resultMap.put("signType", resMap.get("sdk_signtype").toString());
|
resultMap.put("paySign", resMap.get("sdk_paysign").toString());
|
resultMap.put("payAppId", resMap.get("sdk_appid").toString());
|
} else if (TRADE_TYPE_APP.equals(tradeType)) {
|
resultMap.put("appId", smallWeChatDto.getAppId());
|
resultMap.put("timeStamp", PayUtil.getCurrentTimeStamp());
|
resultMap.put("nonceStr", PayUtil.makeUUID(32));
|
resultMap.put("partnerid", smallWeChatDto.getMchId());
|
resultMap.put("prepayid", resMap.getString("session_id"));
|
//resultMap.put("signType", "MD5");
|
resultMap.put("sign", PayUtil.createSign(resultMap, smallWeChatDto.getAppSecret()));
|
} else if (TRADE_TYPE_NATIVE.equals(tradeType)) {
|
resultMap.put("prepayId", resMap.getString("session_id"));
|
resultMap.put("codeUrl", resMap.getString("qr_code"));
|
}
|
resultMap.put("orderId", paymentOrderDto.getOrderId());
|
resultMap.put("qrCode", resMap.getString("qr_code"));
|
resultMap.put("code", "0");
|
resultMap.put("msg", "下单成功");
|
logger.info("【小程序支付】统一下单成功,返回参数:" + resultMap);
|
} else {
|
resultMap.put("code", paymentOrderDto.getOrderId());
|
resultMap.put("msg", resMap.getString("result_msg"));
|
logger.info("【小程序支付】统一下单失败,失败原因:" + resMap.get("return_msg"));
|
}
|
return resultMap;
|
}
|
|
|
private JSONObject java110UnifieldOrder(String feeName, String orderNum,
|
String tradeType, double payAmount, String openid,
|
SmallWeChatDto smallWeChatDto, String notifyUrl) throws Exception {
|
|
//String systemName = MappingCache.getValue(WechatConstant.WECHAT_DOMAIN, WechatConstant.PAY_GOOD_NAME);
|
if (feeName.length() > 127) {
|
feeName = feeName.substring(0, 126);
|
}
|
// String orderPre = CommunitySettingFactory.getValue(smallWeChatDto.getObjId(), "FUIOU_ORDER_PRE");
|
String orderPre = smallWeChatDto.getOrderPre();
|
JSONObject paramMap = new JSONObject();
|
if (tradeType == null){
|
paramMap.put("version", VERSION);
|
paramMap.put("mchnt_cd", smallWeChatDto.getMchId()); // 富友分配给二级商户的商户号
|
paramMap.put("random_str", PayUtil.makeUUID(32));
|
paramMap.put("order_amt", PayUtil.moneyToIntegerStr(payAmount));
|
paramMap.put("mchnt_order_no", orderPre + orderNum);
|
paramMap.put("txn_begin_ts", DateUtil.getNow(DateUtil.DATE_FORMATE_STRING_DEFAULT));
|
paramMap.put("goods_des", "cesdasw");
|
paramMap.put("term_id", "abcdefgh");
|
paramMap.put("term_ip", PayUtil.getLocalIp());
|
paramMap.put("notify_url", notifyUrl + "?wId=" + WechatFactory.getWId(smallWeChatDto.getAppId()));
|
paramMap.put("order_type", "WECHAT");
|
paramMap.put("sub_openid", openid);
|
paramMap.put("sub_appid", smallWeChatDto.getAppId());
|
paramMap.put("sign", createSignByQrCode(paramMap, smallWeChatDto.getAppSecret()));
|
|
logger.debug("调用支付统一下单接口" + paramMap.toJSONString());
|
HttpHeaders headers = new HttpHeaders();
|
headers.add("Content-Type", "application/json");
|
HttpEntity httpEntity = new HttpEntity(paramMap.toJSONString(), headers);
|
ResponseEntity<String> responseEntity = outRestTemplate.exchange(
|
PAY_UNIFIED_ORDER_URL, HttpMethod.POST, httpEntity, String.class);
|
|
logger.debug("统一下单返回" + responseEntity);
|
|
if (responseEntity.getStatusCode() != HttpStatus.OK) {
|
throw new IllegalArgumentException("支付失败" + responseEntity.getBody());
|
}
|
return JSONObject.parseObject(responseEntity.getBody());
|
}
|
paramMap.put("version", VERSION);
|
paramMap.put("mchnt_cd", smallWeChatDto.getMchId()); // 富友分配给二级商户的商户号
|
paramMap.put("random_str", PayUtil.makeUUID(32));
|
paramMap.put("order_amt", PayUtil.moneyToIntegerStr(payAmount));
|
paramMap.put("mchnt_order_no", orderPre + orderNum);
|
paramMap.put("txn_begin_ts", DateUtil.getNow(DateUtil.DATE_FORMATE_STRING_DEFAULT));
|
paramMap.put("goods_des", "cesdasw");
|
paramMap.put("term_id", "abcdefgh");
|
paramMap.put("term_ip", PayUtil.getLocalIp());
|
paramMap.put("notify_url", notifyUrl + "?wId=" + WechatFactory.getWId(smallWeChatDto.getAppId()));
|
paramMap.put("trade_type", tradeType);
|
paramMap.put("sub_openid", openid);
|
paramMap.put("sub_appid", smallWeChatDto.getAppId());
|
paramMap.put("sign", createSign(paramMap, smallWeChatDto.getAppSecret()));
|
|
logger.debug("调用支付统一下单接口" + paramMap.toJSONString());
|
HttpHeaders headers = new HttpHeaders();
|
headers.add("Content-Type", "application/json");
|
HttpEntity httpEntity = new HttpEntity(paramMap.toJSONString(), headers);
|
ResponseEntity<String> responseEntity = outRestTemplate.exchange(
|
PAY_WX_ORDER_URL, HttpMethod.POST, httpEntity, String.class);
|
|
logger.debug("统一下单返回" + responseEntity);
|
|
if (responseEntity.getStatusCode() != HttpStatus.OK) {
|
throw new IllegalArgumentException("支付失败" + responseEntity.getBody());
|
}
|
return JSONObject.parseObject(responseEntity.getBody());
|
}
|
|
|
@Override
|
public PaymentOrderDto java110NotifyPayment(NotifyPaymentOrderDto notifyPaymentOrderDto) {
|
String param = notifyPaymentOrderDto.getParam();
|
PaymentOrderDto paymentOrderDto = new PaymentOrderDto();
|
|
JSONObject resJson = new JSONObject();
|
resJson.put("result_code", "010002");
|
resJson.put("result_msg", "失败");
|
try {
|
JSONObject map = JSONObject.parseObject(param);
|
logger.info("【富友支付回调】 回调数据: \n" + map);
|
String resultCode = map.getString("result_code");
|
if ("000000".equals(resultCode)) {
|
//更新数据
|
int result = confirmPayFee(map, paymentOrderDto,notifyPaymentOrderDto);
|
if (result > 0) {
|
//支付成功
|
resJson.put("result_code", "000000");
|
resJson.put("result_msg", "成功");
|
}
|
}
|
} catch (Exception e) {
|
logger.error("通知失败", e);
|
resJson.put("result_msg", "鉴权失败");
|
}
|
paymentOrderDto.setResponseEntity(new ResponseEntity<String>(resJson.toJSONString(), HttpStatus.OK));
|
return paymentOrderDto;
|
}
|
|
public int confirmPayFee(JSONObject map, PaymentOrderDto paymentOrderDto,NotifyPaymentOrderDto notifyPaymentOrderDto) {
|
String appId = null;
|
//兼容 港币交易时 或者微信有时不会掉参数的问题
|
if (map.containsKey("wId")) {
|
String wId = map.get("wId").toString();
|
wId = wId.replace(" ", "+");
|
appId = WechatFactory.getAppId(wId);
|
} else {
|
appId = map != null && map.get("appid") != null ? map.get("appid").toString() : "不存在appid";
|
}
|
SortedMap<String, String> paramMap = new TreeMap<String, String>();
|
ResponseEntity<String> responseEntity = null;
|
for (String key : map.keySet()) {
|
if ("wId".equals(key)) {
|
continue;
|
}
|
paramMap.put(key, map.get(key).toString());
|
}
|
//String appId = WechatFactory.getAppId(wId);
|
JSONObject paramIn = new JSONObject();
|
paramIn.put("appId", appId);
|
paramIn.put("communityId",notifyPaymentOrderDto.getCommunityId());
|
SmallWeChatDto smallWeChatDto = getSmallWechat(paramIn);
|
|
String sign = createSign(map, smallWeChatDto.getAppSecret());
|
|
if (!sign.equals(map.get("sign"))) {
|
// throw new IllegalArgumentException("鉴权失败");
|
}
|
|
String outTradeNo = map.get("mchnt_order_no").toString();
|
paymentOrderDto.setOrderId(outTradeNo);
|
return 1;
|
}
|
|
|
private SmallWeChatDto getSmallWechat(JSONObject paramIn) {
|
|
SmallWeChatDto smallWeChatDto = new SmallWeChatDto();
|
//由于富有支付方式要从数据库里面取值,所以此处要查询
|
PaymentPoolDto paymentPoolDto = new PaymentPoolDto();
|
paymentPoolDto.setCommunityId(paramIn.getString("communityId"));
|
paymentPoolDto.setPage(1);
|
paymentPoolDto.setRow(10);
|
List<PaymentPoolDto> paymentPoolDtos = paymentPoolV1InnerServiceSMOImpl.queryPaymentPools(paymentPoolDto);
|
List<PaymentPoolDto> collect = paymentPoolDtos.stream().filter(e -> e.getPaymentType().equals("FUIOU")).collect(Collectors.toList());
|
PaymentPoolValueDto paymentPoolValueDto = new PaymentPoolValueDto();
|
paymentPoolValueDto.setPpId(collect.get(0).getPpId());
|
paymentPoolValueDto.setCommunityId(paramIn.getString("communityId"));
|
List<PaymentPoolValueDto> values = paymentPoolValueV1InnerServiceSMOImpl.queryPaymentPoolValues(paymentPoolValueDto);
|
Map<String, List<PaymentPoolValueDto>> payMap = values.stream().collect(Collectors.groupingBy(PaymentPoolValueDto::getColumnKey));
|
if(payMap.containsKey("FUIOU_APP_ID")){
|
smallWeChatDto.setAppId(payMap.get("FUIOU_APP_ID").get(0).getColumnValue());
|
}else{
|
smallWeChatDto.setAppId(MappingCache.getValue(WechatConstant.WECHAT_DOMAIN, "appId"));
|
}
|
if(payMap.containsKey("FUIOU_MCHNT_KEY")){
|
smallWeChatDto.setAppSecret(payMap.get("FUIOU_MCHNT_KEY").get(0).getColumnValue());
|
}else{
|
smallWeChatDto.setAppSecret(MappingCache.getValue(WechatConstant.WECHAT_DOMAIN, "appSecret"));
|
}
|
if(payMap.containsKey("FUIOU_MERCHANT_ID")){
|
smallWeChatDto.setMchId(payMap.get("FUIOU_MERCHANT_ID").get(0).getColumnValue());
|
}else{
|
smallWeChatDto.setMchId(MappingCache.getValue(MappingConstant.WECHAT_STORE_DOMAIN, "mchId"));
|
}
|
if(payMap.containsKey("FUIOU_ORDER_PRE")){
|
smallWeChatDto.setOrderPre(payMap.get("FUIOU_ORDER_PRE").get(0).getColumnValue());
|
}else{
|
smallWeChatDto.setOrderPre("1066");
|
}
|
smallWeChatDto.setPayPassword(MappingCache.getValue(MappingConstant.WECHAT_STORE_DOMAIN, "key"));
|
smallWeChatDto.setObjId(paramIn.getString("communityId"));
|
return smallWeChatDto;
|
// }
|
|
// return BeanConvertUtil.covertBean(smallWeChatDtos.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("trade_type") + "|"
|
+ paramMap.getString("order_amt") + "|"
|
+ paramMap.getString("mchnt_order_no") + "|"
|
+ paramMap.getString("txn_begin_ts") + "|"
|
+ paramMap.getString("goods_des") + "|"
|
+ paramMap.getString("term_id") + "|"
|
+ paramMap.getString("term_ip") + "|"
|
+ paramMap.getString("notify_url") + "|"
|
+ paramMap.getString("random_str") + "|"
|
+ paramMap.getString("version") + "|"
|
+ payPassword;
|
return PayUtil.md5(str);
|
}
|
|
|
|
private String createSignByQrCode(JSONObject paramMap, String payPassword) {
|
String str = paramMap.getString("mchnt_cd") + "|"
|
+ paramMap.getString("order_type") + "|"
|
+ paramMap.getString("order_amt") + "|"
|
+ paramMap.getString("mchnt_order_no") + "|"
|
+ paramMap.getString("txn_begin_ts") + "|"
|
+ paramMap.getString("goods_des") + "|"
|
+ paramMap.getString("term_id") + "|"
|
+ paramMap.getString("term_ip") + "|"
|
+ paramMap.getString("notify_url") + "|"
|
+ paramMap.getString("random_str") + "|"
|
+ paramMap.getString("version") + "|"
|
+ payPassword;
|
return PayUtil.md5(str);
|
}
|
}
|