xiaoyong931011
2023-01-10 73b3813c2d110bf446f251350f120bb1e2b51d0c
src/main/java/cc/mrbird/febs/pay/service/impl/UnipayServiceImpl.java
@@ -1,25 +1,59 @@
package cc.mrbird.febs.pay.service.impl;
import cc.mrbird.febs.common.entity.FebsResponse;
import cc.mrbird.febs.pay.model.UnipayDto;
import cc.mrbird.febs.mall.entity.MallOrderInfo;
import cc.mrbird.febs.mall.mapper.MallOrderInfoMapper;
import cc.mrbird.febs.pay.model.*;
import cc.mrbird.febs.pay.service.UnipayService;
import cc.mrbird.febs.pay.util.HttpRequester;
import cc.mrbird.febs.pay.util.HttpRespons;
import cc.mrbird.febs.pay.util.*;
import cn.hutool.core.date.DateUtil;
import cn.hutool.core.util.ObjectUtil;
import cn.hutool.core.util.StrUtil;
import cn.hutool.json.JSONUtil;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.codec.digest.DigestUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import java.io.IOException;
import java.lang.reflect.Field;
import java.math.BigDecimal;
import java.math.RoundingMode;
import java.text.DecimalFormat;
import java.util.*;
@Slf4j
@Service
public class UnipayServiceImpl implements UnipayService {
    //汇聚支付平台公钥
    public static final String platformPublicKey = "MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQCwSAEXsiC0IYh" +
            "a6a94imKq8VfOkk7WjDRAQWMBRnoKOZeEUeMrHYiblcrqeMYXGpV13288iUOkuyKwkPXkYXyIQK8emvJIbQOhtB5bS" +
            "lAbodsPgncM9Ney1GFiz+7ogBxyt58mP8AA9UHtMw7u78zZoQ1+dUWwUUowVXml3Q0cVQIDAQAB";
    //本地生成的公钥
    public static final String paySecretKey =
            "MIICdgIBADANBgkqhkiG9w0BAQEFAASCAmAwggJcAgEAAoGBAJSsoLHAy4P+H76l4i6IMZ/JKKTVphvrJBSB777N8h" +
                    "gzN38M+66IqazRQLNjjyn2UZxm4eSfuJ4OBxCTGLJgDOw8B9l9lVo+0bzhNxsy5ae20EOlQa3h8xfCjCCU" +
                    "Xb8VP8yknG43Wk1WtwLX5VYWmJFTwA1I0dWQR4RlbHH2q2aVAgMBAAECgYBZNyoK4JV/tFwaTHLo12Nn7g" +
                    "9MssRGFpmFEN/sEKuJKBrSS9kvx+SBxuPbgg/j72LTxs0hI1NmzSYiJRL53zeBFM0wM2/D2zW0pZfogUPN" +
                    "7Mb3FkFOoE0CxFwn7pjjBkvAsZQJS2fZFY6cf/WYhVI+XCMzjiQWJSQUXKEfWzUflQJBAMpLzinyAsBHKe" +
                    "6lmAkSh/yaQX76cnxNZGmfc/gj62qM5YDzpt3+Za5Y4NSq5kYu36NAgorf2ipJPRwpgS13fK8CQQC8JKVP" +
                    "2YgZ39fa+xVJu3yTT48k9EnYIEfsv/Bc7AVdpaZKDeEBuGpBb0JQzuIg3Zd6yNQCzH3oBqp8l6l/fgn7Ak" +
                    "EAtFluvHB4yWjoVk0lRPlTaP0w5P5ssKrimVPBtPh4+a4RMayHGKSjjBLKpm6SCwHg+Q8bEqpNOqO+qmvK" +
                    "MXm0GwJAZd4Bk8ZYJopIOUyRLibRQIFnI78Q7HAuAUW7QtSX4yh5bMcu+Nt8zIkNAuvBC8Ju7hAmmo1V7n" +
                    "cNgAAtydXYWQJALLOFCjCkRgeRVL8YE8bVi4U16b8ltAN1DlbWEzui6VFy2vIga3IryesNVAOOdornyAwf" +
                    "1huqB2lYfuQwtrIBKg==";
    public static final String MD5KEY = "2e95f6a3e11e47fa8a4386d6aefe1735";
    public static final String notifyUrl = "http://47.111.90.145:8800/api/unipay/unipayCallBack";
    public static final String agreementPayNotifyUrl = "http://47.111.90.145:8800/api/unipay/agreeMentPayCallBack";
    public static final String singlePayNotifyUrl = "http://47.111.90.145:8800/api/unipay/singlePayCallBack";
    public static final String p1MerchantNo = "888118000001971";/** 商户编号 */
    public static final String aesKey = "1234567891234567";/** 商户编号 */
    @Autowired
    private MallOrderInfoMapper orderInfoMapper;
    @Override
    @Transactional
@@ -27,7 +61,7 @@
        String key = "2e95f6a3e11e47fa8a4386d6aefe1735";/** md5密钥商户后台-商户中心-商户设置-密钥管理获取 必填!*/
        Map<String, String> map = new HashMap<String, String>();
        map.put("p0_Version", "1.0");/** 版本号 */
        map.put("p1_MerchantNo", "888118000001971");/** 商户编号 */
        map.put("p1_MerchantNo", p1MerchantNo);/** 商户编号 */
        map.put("p2_OrderNo", unipayDto.getOrderNo()); /**商户订单号*/
        map.put("p3_Amount", unipayDto.getAmount().toString());/**订单金额*/
        map.put("p4_Cur", "1"); /**交易币种 */
@@ -131,4 +165,381 @@
    }
    @Override
    @Transactional
    public String agreementPay(AgreementPayDto agreementPayDto) {
        String secretKey =paySecretKey;
        String key = aesKey;//敏感信息加密key
        Map<String, Object> map = new HashMap<>();
        RequestParam requestParam = new RequestParam();
        requestParam.setMch_no(p1MerchantNo);// 商户编号
        requestParam.setMethod("fastPay.agreement.pay");// 固定方法名
        requestParam.setVersion("1.0");// 版本号
        requestParam.setRand_str("12345678901234567890123456789012");// 随机字符串
        requestParam.setSign_type("2");// 签名类型
        try {
            requestParam.setSec_key(RSAUtil.encryptByPublicKey(platformPublicKey,aesKey));//加密密钥
        } catch (Exception e) {
            e.printStackTrace();
        }
        map.put("mch_order_no",agreementPayDto.getOrderNo()); // 商户订单号
        map.put("order_amount", agreementPayDto.getOrderAmount()); // 订单金额;金额保留两位小数
        map.put("mch_req_time", agreementPayDto.getOrderTime()); // 订单时间
        map.put("order_desc", agreementPayDto.getOrderDesc()); // 商品名称
        map.put("callback_url", agreementPayNotifyUrl); // 异步通知地址
        map.put("bank_card_no",  AESUtil.Aes256Encode(agreementPayDto.getBankNo(),aesKey)); // 签约银行卡号
        requestParam.setData(JsonUtil.toString(map));
        String signStr = null;
        try {
            signStr = getSortedString(requestParam);
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println("待签名字符串:"+signStr);// 待签名字符串
        // 签名
        if (SignTypeEnum.RSA.getValue().equals(requestParam.getSign_type())) {
            try {
                signStr =(RSAUtil.sign(signStr, secretKey, false));
            } catch (Exception e) {
                e.printStackTrace();
            }
        } else {
            signStr = "";
            System.out.println("未预期的签名类型:" + requestParam.getSign_type());
        }
        requestParam.setSign(signStr);
        // Map转json字符串
        String reqBodyJson = JSON.toJSONString(requestParam);
        System.out.println(reqBodyJson);
        String httpResponseJson = null;
        try {
            httpResponseJson = HttpClientUtil.sendHttpPost("https://api.joinpay.com/fastpay", reqBodyJson);
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println(httpResponseJson);
        if(StrUtil.isNotBlank(httpResponseJson)){
            return httpResponseJson;
        }else{
            return "fail";
        }
    }
    @Override
    @Transactional
    public String singlePay(SinglePayDto singlePayDto) {
        String singlePay = null;
        String key = MD5KEY;
        Map<String, Object> map = new HashMap<>();
        map.put("userNo", p1MerchantNo);// 商户编号
        map.put("productCode", "BANK_PAY_DAILY_ORDER");//产品类型
        map.put("requestTime", DateUtil.now()); // 交易请求时间
        map.put("merchantOrderNo", singlePayDto.getMerchantOrderNo());//商户订单号
        map.put("receiverAccountNoEnc", singlePayDto.getReceiverAccountNoEncBankNo()); // 收款账户号
        map.put("receiverNameEnc", singlePayDto.getReceiverAccountNoEncName()); // 收款人
        map.put("receiverAccountType", singlePayDto.getReceiverAccountType());//账户类型
        map.put("paidAmount", singlePayDto.getPaidAmount()); //交易金额
        map.put("currency", singlePayDto.getCurrency());//币种
        map.put("isChecked", singlePayDto.getIsChecked());//是否复核
        map.put("paidDesc", singlePayDto.getPaidDesc());//代付说明
        map.put("paidUse", singlePayDto.getPaidUse());//代付用途
        map.put("callbackUrl", singlePayNotifyUrl);//商户通知地址
        String reqSign = getRequestSign(map);
        // 签名
        String hmac = Md5_Sign.SignByMD5(reqSign, key);
        map.put("hmac", hmac);/** 签名数据 */
        // Map转json字符串
        String reqBodyJson = JSON.toJSONString(map);
        System.out.println("reqBodyJson:" + reqBodyJson);
        String httpResponseJson = null;
        try {
            httpResponseJson = HttpClientUtil
                    .sendHttpPost("https://www.joinpay.com/payment/pay/singlePay",reqBodyJson);
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println(httpResponseJson);
        if(StrUtil.isNotBlank(httpResponseJson)){
            try {
                singlePay = doResponseInfo(httpResponseJson, key);
            } catch (Exception e) {
                e.printStackTrace();
            }
            return singlePay;
        }else{
            return "fail";
        }
    }
    /**
     * 对单笔代付响应信息的处理
     *
     * @param httpResponseJson 响应信息json字符串
     * @param key 商户秘钥
     * @throws Exception 异常实体类
     */
    @SuppressWarnings("unchecked")
    private static String doResponseInfo(String httpResponseJson, String key) throws Exception {
        // 响应信息map集合
        Map<String, Object> httpResponseMap = (Map<String, Object>) JSONObject.parse(httpResponseJson);
        // 业务数据map集合
        Map<String, Object> dataMap = (Map<String, Object>) httpResponseMap.get("data");
        dataMap.put("statusCode", httpResponseMap.get("statusCode"));
        dataMap.put("message", httpResponseMap.get("message"));
        // 请求签名串
        String reqSign = getRequestSign(httpResponseMap);
        // 响应签名串
        String respSign = getResponseSign(dataMap);
        // 请求数据的加密签名
        String reqHmac = Md5_Sign.SignByMD5(respSign, key);
        // 请求数据的加密签名
        String respHmac = (String) dataMap.get("hmac");
        System.out.println("reqHmac:" + reqHmac);
        System.out.println("respSign:" + respHmac);
        reqHmac=reqHmac.toUpperCase();
        respHmac=respHmac.toUpperCase();
        boolean isMatch = reqHmac.equals(respHmac);
        if (isMatch) {
            if("2001".equals(httpResponseMap.get("statusCode").toString())
                    && ObjectUtil.isEmpty(dataMap.get("errorCode"))){
                System.out.println("验签成功");
                return dataMap.get("merchantOrderNo").toString();
            }
        }
        return "fail";
    }
    /**
     * 获取响应数据签名串信息
     * 必须按新代付接口文档应答参数信息顺序来进行字符串的拼接,详情请参考新代付接口文档的应答报文
     *
     * @param params 响应数据参数
     * @return 返回响应签名串
     */
    public static String getResponseSign(Map<String, Object> params) {
        StringBuilder stringBuilder = new StringBuilder();
        stringBuilder.append(params.get("statusCode")).append(params.get("message")).append(params.get("errorCode"))
                .append(params.get("errorDesc")).append(params.get("userNo")).append(params.get("merchantOrderNo"));
        return stringBuilder.toString();
    }
    /**
     * 获取请求数据签名串信息
     * 必须按新代付接口文档请求参数信息顺序来进行字符串的拼接,详情请参考新代付接口文档请求报文
     *
     * @param params 请求数据参数
     * @return 返回请求签名串
     */
    public static String getRequestSign(Map<String, Object> params) {
        StringBuilder stringBuilder = new StringBuilder();
        stringBuilder.append(params.get("userNo"))
                .append(params.get("productCode"))
                .append(params.get("requestTime"))
                .append(params.get("merchantOrderNo"))
                .append(params.get("receiverAccountNoEnc"))
                .append(params.get("receiverNameEnc"))
                .append(params.get("receiverAccountType"))
                .append(params.get("paidAmount"))
                .append(params.get("currency"))
                .append(params.get("isChecked"))
                .append(params.get("paidDesc"))
                .append(params.get("paidUse"))
                .append(params.get("callbackUrl"));
        System.out.println("reqSign:" + stringBuilder.toString());
        return stringBuilder.toString();
    }
    public static void main(String[] args) {
//        BigDecimal value = new BigDecimal("0.10858").setScale(2,BigDecimal.ROUND_HALF_UP);
        BigDecimal value = new BigDecimal("1231.1").setScale(2, BigDecimal.ROUND_DOWN);
        DecimalFormat decimalFormat = new DecimalFormat("0.00#");
        String strVal = decimalFormat.format(value);
        System.out.println(strVal);
    }
    @Override
    public String getAgreeMentPaySms(AgreeMentPaySmsDto agreeMentPaySmsDto) {
        String secretKey = paySecretKey;
        Map<String, Object> map = new HashMap<>();
        RequestParam requestParam = new RequestParam();
        requestParam.setMch_no(p1MerchantNo);// 商户编号
        requestParam.setMethod("fastPay.agreement.signSms");// 固定方法名
        requestParam.setVersion("1.0");// 版本号
        requestParam.setRand_str("12345678901234567890123456789012");// 随机字符串
        requestParam.setSign_type("2");// 签名类型
        try {
            requestParam.setSec_key(RSAUtil.encryptByPublicKey(platformPublicKey,aesKey));//加密密钥
        } catch (Exception e) {
            e.printStackTrace();
        }
        MallOrderInfo orderInfo = orderInfoMapper.selectByOrderNo(agreeMentPaySmsDto.getOrderNo());
        map.put("mch_order_no", "NO_"+orderInfo.getOrderNo()); // 商户订单号
        BigDecimal value = orderInfo.getAmount().setScale(2, BigDecimal.ROUND_DOWN);
        DecimalFormat decimalFormat = new DecimalFormat("0.00#");
        String strVal = decimalFormat.format(value);
        map.put("order_amount", strVal); // 订单金额
        map.put("mch_req_time", DateUtil.now()); // 下单时间
        map.put("payer_name",AESUtil.Aes256Encode(agreeMentPaySmsDto.getName(),aesKey)); // 姓名
        map.put("id_type", agreeMentPaySmsDto.getIdType()); // 证件类型
        map.put("id_no", AESUtil.Aes256Encode(agreeMentPaySmsDto.getIdCardNum(),aesKey)); // 证件号码
        map.put("bank_card_no", AESUtil.Aes256Encode(agreeMentPaySmsDto.getBankNo(),aesKey)); // 银行卡号
        map.put("mobile_no", AESUtil.Aes256Encode(agreeMentPaySmsDto.getPhone(), aesKey)); // 手机号
        requestParam.setData(JsonUtil.toString(map));
        String signStr = null;
        try {
            signStr = getSortedString(requestParam);
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println("待签名字符串:"+signStr);// 待签名字符串
        // 签名
        if (SignTypeEnum.RSA.getValue().equals(requestParam.getSign_type())) {
            try {
                signStr =(RSAUtil.sign(signStr, secretKey, false));
            } catch (Exception e) {
                e.printStackTrace();
            }
        } else {
            signStr = "";
            System.out.println("未预期的签名类型:" + requestParam.getSign_type());
        }
        requestParam.setSign(signStr);
        // Map转json字符串
        String reqBodyJson = JSON.toJSONString(requestParam);
        System.out.println(reqBodyJson);
        String httpResponseJson = null;
        try {
            httpResponseJson = HttpClientUtil.sendHttpPost("https://api.joinpay.com/fastpay", reqBodyJson);
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println(httpResponseJson);
        cn.hutool.json.JSONObject jsonObject = JSONUtil.parseObj(httpResponseJson);
        System.out.println(jsonObject);
        String biz_code = jsonObject.get("biz_code").toString();
        String biz_msg = jsonObject.get("biz_msg").toString();
        if("JS000000".equals(biz_code)){
            return biz_code;
        }else{
            return biz_msg;
        }
    }
    @Override
    public String agreementSign(AgreementSignDto agreementSignDto) {
        String secretKey = paySecretKey;
        Map<String, Object> map = new HashMap<>();
        RequestParam requestParam = new RequestParam();
        requestParam.setMch_no(p1MerchantNo);// 商户编号
        requestParam.setMethod("fastPay.agreement.smsSign");// 固定方法名
        requestParam.setVersion("1.0");// 版本号
        requestParam.setRand_str("12345678901234567890123456789012");// 随机字符串
        requestParam.setSign_type("2");// 签名类型
        try {
            requestParam.setSec_key(RSAUtil.encryptByPublicKey(platformPublicKey,aesKey));
        } catch (Exception e) {
            e.printStackTrace();
        }
        map.put("mch_order_no",  "NO_"+agreementSignDto.getOrderNo()); // 商户订单号
        map.put("sms_code", agreementSignDto.getSmsCode()); // 签约短信验证码
        requestParam.setData(JsonUtil.toString(map));
        String signStr = null;
        try {
            signStr = getSortedString(requestParam);
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println(signStr);// 待签名字符串
        // 签名
        if (SignTypeEnum.RSA.getValue().equals(requestParam.getSign_type())) {
            try {
                signStr =(RSAUtil.sign(signStr, secretKey, false));
            } catch (Exception e) {
                e.printStackTrace();
            }
        } else {
            signStr = "";
            System.out.println("未预期的签名类型:" + requestParam.getSign_type());
        }
        requestParam.setSign(signStr);
        // Map转json字符串
        String reqBodyJson = JSON.toJSONString(requestParam);
        System.out.println(reqBodyJson);
        String httpResponseJson = null;
        try {
            httpResponseJson = HttpClientUtil.sendHttpPost("https://api.joinpay.com/fastpay", reqBodyJson);
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println(httpResponseJson);
        if(StrUtil.isNotBlank(httpResponseJson)){
            return httpResponseJson;
        }else{
            return "fail";
        }
    }
    protected static String getSortedString(Object obj) throws Exception {
        Field[] fields = obj.getClass().getDeclaredFields();
        Map<String, String> map = new HashMap<String, String>();
        for (int i = 0; i < fields.length; i++) {
            Field filed = fields[i];
            String name = filed.getName();
            if (SignUtil.NOT_SIGN_PARAM.contains(name)) {// 不参与签名或验签的参数直接跳过
                continue;
            }
            filed.setAccessible(true);
            map.put(name, String.valueOf(filed.get(obj)));
        }
        StringBuffer content = new StringBuffer();
        List<String> keys = new ArrayList(map.keySet());
        Collections.sort(keys); // 排序map
        for (int i = 0; i < keys.size(); i++) {
            String key = keys.get(i);
            String value = map.get(key);
            if (i != 0) {
                content.append(SignUtil.SIGN_SEPARATOR);
            }
            content.append(key).append(SignUtil.SIGN_EQUAL).append(value);
        }
        return content.toString();
    }
}