package com.java110.acct.payment.adapt.chinaUms;
import com.alibaba.fastjson.JSONObject;
import com.java110.acct.payment.IPaymentFactoryAdapt;
import com.java110.core.context.ICmdDataFlowContext;
import com.java110.core.factory.ChinaUmsFactory;
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.PaymentOrderDto;
import com.java110.dto.smallWeChat.SmallWeChatDto;
import com.java110.intf.store.ISmallWechatV1InnerServiceSMO;
import com.java110.intf.user.IOwnerAppUserInnerServiceSMO;
import com.java110.utils.cache.MappingCache;
import com.java110.utils.constant.MappingConstant;
import com.java110.utils.constant.WechatConstant;
import com.java110.utils.util.*;
import org.apache.commons.codec.digest.DigestUtils;
import org.slf4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.*;
import org.springframework.stereotype.Service;
import org.springframework.web.client.RestTemplate;
import java.io.UnsupportedEncodingException;
import java.util.List;
import java.util.Map;
import java.util.SortedMap;
import java.util.TreeMap;
/**
* 富友 支付
* 此实现方式为 通过 富友支付 去下单不直接去掉微信
*
* 商户调用此接口则用户可使用支付宝或微信进行支付。
* 本接口支持: 微信公众号、 微信小程序、 微信 APP, 支付宝服务窗等
*
*
* 说明:信息通过 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
*
* 该接口常应用于聚合二维码(静态二维码、统一收款码、台卡等不同叫法),用户扫二维码进入微信公众号/支付宝服务窗
* /QQJS 页面,页面调此接口生成订单,接受订单参数后调起官方支付接口支付。详见公众号/服务窗对接流程
* 步骤 1:用户通过支付宝(服务窗)、微信(公众号)进入到商户 H5 页面,或者是通过扫描台卡进入。
* 步骤 2:用户选择商品、输入支付金额等进行下单支付
* 步骤 3:商户将订单信息发送给富友,返回支付信息(用于调起支付宝、微信的参数)。
* 步骤 4:商户拿到支付信息后调起微信或者支付宝进行支付
* 步骤 5:支付结果以回调(2.5)的方式通知到商户
*
* @desc add by 吴学文 15:33
*/
@Service("chinaUmsPaymentFactory")
public class ChinaUmsPaymentFactoryAdapt implements IPaymentFactoryAdapt {
private static final Logger logger = LoggerFactory.getLogger(ChinaUmsPaymentFactoryAdapt.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 PAY_UNIFIED_ORDER_URL = "https://api-mop.chinaums.com/v1/netpay/wx/unified-order";
private static final String VERSION = "1.0";
@Autowired
private ISmallWechatV1InnerServiceSMO smallWechatV1InnerServiceSMOImpl;
@Autowired
private IOwnerAppUserInnerServiceSMO ownerAppUserInnerServiceSMOImpl;
@Autowired
private RestTemplate outRestTemplate;
@Override
public Map java110Payment(PaymentOrderDto paymentOrderDto, JSONObject reqJson, ICmdDataFlowContext context) throws Exception {
SmallWeChatDto smallWeChatDto = getSmallWechat(reqJson);
String appId = context.getReqHeaders().get("app-id");
String userId = context.getReqHeaders().get("user-id");
String tradeType = reqJson.getString("tradeType");
String notifyUrl = MappingCache.getValue(MappingConstant.URL_DOMAIN,"OWNER_WECHAT_URL") + "/app/payment/notify/chinaums/992020011134400001";
String openId = reqJson.getString("openId");
if(StringUtil.isEmpty(openId)) {
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;
}
OwnerAppUserDto ownerAppUserDto = new OwnerAppUserDto();
ownerAppUserDto.setUserId(userId);
ownerAppUserDto.setAppType(appType);
List ownerAppUserDtos = ownerAppUserInnerServiceSMOImpl.queryOwnerAppUsers(ownerAppUserDto);
Assert.listOnlyOne(ownerAppUserDtos, "未找到开放账号信息");
openId = ownerAppUserDtos.get(0).getOpenId();
}
logger.debug("【小程序支付】 统一下单开始, 订单编号=" + paymentOrderDto.getOrderId());
SortedMap resultMap = new TreeMap();
//生成支付金额,开发环境处理支付金额数到0.01、0.02、0.03元
double payAmount = PayUtil.getPayAmountByEnv(MappingCache.getValue("HC_ENV"), paymentOrderDto.getMoney());
//添加或更新支付记录(参数跟进自己业务需求添加)
JSONObject resMap = null;
resMap = this.java110UnifieldOrder(paymentOrderDto.getName(),
paymentOrderDto.getOrderId(),
tradeType,
payAmount,
openId,
smallWeChatDto,
notifyUrl
);
if ("SUCCESS".equals(resMap.getString("errCode"))) {
if (TRADE_TYPE_JSAPI.equals(tradeType)) {
resultMap.putAll(JSONObject.toJavaObject(JSONObject.parseObject(resMap.getString("jsPayRequest")), Map.class));
resultMap.put("sign",resultMap.get("paySign"));
} 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.getPayPassword()));
} else if (TRADE_TYPE_NATIVE.equals(tradeType)) {
resultMap.put("prepayId", resMap.getString("session_id"));
resultMap.put("codeUrl", resMap.getString("qr_code"));
}
resultMap.put("code", "0");
resultMap.put("msg", "下单成功");
logger.info("【小程序支付】统一下单成功,返回参数:" + resultMap);
} else {
resultMap.put("code", resMap.getString("errCode"));
resultMap.put("msg", resMap.getString("errMsg"));
logger.info("【小程序支付】统一下单失败,失败原因:" + resMap.get("errMsg"));
}
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);
}
JSONObject paramMap = new JSONObject();
paramMap.put("requestTimestamp", DateUtil.getNow(DateUtil.DATE_FORMATE_STRING_A));
paramMap.put("mid", smallWeChatDto.getMchId()); // 富友分配给二级商户的商户号
paramMap.put("tid", "CV5EW7IM"); //终端号
paramMap.put("instMid", "YUEDANDEFAULT");
paramMap.put("merOrderId", "11WP"+orderNum);
paramMap.put("totalAmount", PayUtil.moneyToIntegerStr(payAmount));
paramMap.put("notifyUrl", notifyUrl + "?wId=" + WechatFactory.getWId(smallWeChatDto.getAppId()));
paramMap.put("tradeType", tradeType);
paramMap.put("subopenId", openid);
paramMap.put("subAppId", smallWeChatDto.getAppId());
logger.debug("调用支付统一下单接口" + paramMap.toJSONString());
HttpHeaders headers = new HttpHeaders();
headers.add("Content-Type", "application/json");
headers.add("Authorization", ChinaUmsFactory.getAccessToken(smallWeChatDto));
HttpEntity httpEntity = new HttpEntity(paramMap.toJSONString(), headers);
ResponseEntity 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());
}
@Override
public PaymentOrderDto java110NotifyPayment(String param) {
PaymentOrderDto paymentOrderDto = new PaymentOrderDto();
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, paymentOrderDto);
if (result > 0) {
//支付成功
resJson.put("errCode", "SUCCESS");
resJson.put("errMsg", "成功");
}
} catch (Exception e) {
logger.error("通知失败", e);
resJson.put("result_msg", "鉴权失败");
}
paymentOrderDto.setResponseEntity(new ResponseEntity(resJson.toJSONString(), HttpStatus.OK));
return paymentOrderDto;
}
public int confirmPayFee(JSONObject map, PaymentOrderDto paymentOrderDto) {
String appId;
//兼容 港币交易时 或者微信有时不会掉参数的问题
if (map.containsKey("wId")) {
String wId = map.get("wId").toString();
wId = wId.replace(" ", "+");
appId = WechatFactory.getAppId(wId);
} else {
appId = map.get("appid").toString();
}
JSONObject paramIn = new JSONObject();
paramIn.put("appId", appId);
SmallWeChatDto smallWeChatDto = getSmallWechat(paramIn);
//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();
paymentOrderDto.setOrderId(outTradeNo);
return 1;
}
private SmallWeChatDto getSmallWechat(JSONObject paramIn) {
SmallWeChatDto smallWeChatDto = new SmallWeChatDto();
smallWeChatDto.setObjId(paramIn.getString("communityId"));
smallWeChatDto.setAppId(paramIn.getString("appId"));
smallWeChatDto.setPage(1);
smallWeChatDto.setRow(1);
List smallWeChatDtos = smallWechatV1InnerServiceSMOImpl.querySmallWechats(smallWeChatDto);
if (smallWeChatDtos == null || smallWeChatDtos.size() < 1) {
smallWeChatDto = new SmallWeChatDto();
smallWeChatDto.setAppId(MappingCache.getValue(WechatConstant.WECHAT_DOMAIN, "appId"));
smallWeChatDto.setAppSecret(MappingCache.getValue(WechatConstant.WECHAT_DOMAIN, "appSecret"));
smallWeChatDto.setMchId(MappingCache.getValue(WechatConstant.WECHAT_DOMAIN, "mchId"));
smallWeChatDto.setPayPassword(MappingCache.getValue(WechatConstant.WECHAT_DOMAIN, "key"));
return smallWeChatDto;
}
return BeanConvertUtil.covertBean(smallWeChatDtos.get(0), SmallWeChatDto.class);
}
// 根据编码类型获得签名内容byte[]
public static byte[] getContentBytes(String content) {
try {
return content.getBytes("UTF-8");
} catch (UnsupportedEncodingException e) {
throw new RuntimeException("签名过程中出现错误");
}
}
}