rentaiming
2024-05-22 10748fcb8994a0c93b024a97e6df9336e94bc0f7
写拍卖师端接口
3个文件已修改
464 ■■■■■ 已修改文件
ruoyi-modules/ruoyi-order/pom.xml 6 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
ruoyi-modules/ruoyi-order/src/main/java/com/ruoyi/order/controller/PaylogController.java 426 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
ruoyi-modules/ruoyi-order/src/main/java/com/ruoyi/order/util/tencent/WXPay.java 32 ●●●● 补丁 | 查看 | 原始文档 | blame | 历史
ruoyi-modules/ruoyi-order/pom.xml
@@ -129,6 +129,12 @@
    </dependency>
    <dependency>
      <groupId>com.alipay.sdk</groupId>
      <artifactId>alipay-sdk-java</artifactId>
      <version>3.4.27.ALL</version>
    </dependency>
    <dependency>
      <groupId>cn.afterturn</groupId>
      <artifactId>easypoi-spring-boot-starter</artifactId>
      <version>4.0.0</version>
ruoyi-modules/ruoyi-order/src/main/java/com/ruoyi/order/controller/PaylogController.java
@@ -1,9 +1,43 @@
package com.ruoyi.order.controller;
import com.alipay.api.AlipayClient;
import com.alipay.api.DefaultAlipayClient;
import com.alipay.api.request.AlipayTradeRefundRequest;
import com.alipay.api.response.AlipayTradeRefundResponse;
import com.ruoyi.common.core.domain.R;
import com.ruoyi.order.domain.pojo.Paylog;
import com.ruoyi.order.service.IPaylogService;
import com.ruoyi.order.util.DateUtil;
import com.ruoyi.order.util.SinataUtil;
import com.ruoyi.order.util.alipay.config.AlipayConfig;
import com.ruoyi.order.util.alipay.util.PayDemoActivity;
import com.ruoyi.order.util.tencent.common.Configure;
import com.ruoyi.order.util.tencent.common.Signature;
import com.ruoyi.order.util.tencent.common.XMLParser;
import com.ruoyi.order.util.tencent.protocol.AppPayReqData;
import com.ruoyi.order.util.tencent.protocol.UnifiedorderReqData;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.bind.annotation.RestController;
import javax.annotation.Resource;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.time.LocalDateTime;
import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import static com.ruoyi.order.util.tencent.WXPay.requestRefundService;
import static com.ruoyi.order.util.tencent.WXPay.requestUnifiedorderService;
/**
 * <p>
@@ -16,5 +50,397 @@
@RestController
@RequestMapping("/paylog")
public class PaylogController {
    /**
     * 获取支付信息
     *
     * @param userId
     *            用户ID
     * @param type
     *            支付类型 1=支付宝,2 = 微信
     * @param ordernum
     *            订单编号
     * @param price
     *            金额
     * @return
     */
    @Resource
    private IPaylogService iPaylogService;
    Logger log = LoggerFactory.getLogger(getClass());
    @ResponseBody
    @RequestMapping("getPayInfo")
    public R<Map<String, Object>> getPayInfo(Integer uid, Integer type, String subject, String body, String orderID,
                                          HttpServletRequest request) {
        Double price = 0.0;
        try {
            if(judgeContainsStr(orderID)){
                price=  Double.valueOf(1);
                body = "商品支付";
                subject = "商品支付";
                Map<String, Object> map = new HashMap<String, Object>();
                if (type == 1) {
                    // 支付宝预下单
                    return alipay("1", subject, body, price, request);
                } else {
                    // 微信预下单
                    return wxpay(1, "1", body, price, request);
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
        return R.fail("获取异常");
    }
    /**
     * 支付宝订单退款
     *
     * @author tzj
     * @param out_trade_no
     *            商品订单号
     * @param trade_no
     *            支付宝交易号
     * @param refund_amount
     *            退款金额
     * @return
     */
    public boolean refundForAlipay(String out_trade_no, String trade_no, Double refund_amount) {
        try {
            // 实例化客户端
            AlipayClient alipayClient = new DefaultAlipayClient("https://openapi.alipay.com/gateway.do",
                    AlipayConfig.app_id, AlipayConfig.private_key, "json", AlipayConfig.input_charset,
                    AlipayConfig.ali_public_key,"RSA2");
            // 实例化具体API对应的request类,类名称和接口名称对应
            AlipayTradeRefundRequest request = new AlipayTradeRefundRequest();
            // SDK已经封装掉了公共参数,这里只需要传入业务参数
            // 此次只是参数展示,未进行字符串转义,实际情况下请转义
            request.setBizContent("{" + "\"out_trade_no\":\"" + out_trade_no + "\", " + "\"trade_no\":\"" + trade_no
                    + "\", " + "\"refund_amount\":" + refund_amount + ", " + "\"refund_reason\":\"正常退款\","
                    + "\"out_request_no\":\"HZ01RF001\"" + "}");
            AlipayTradeRefundResponse response = alipayClient.execute(request);
            // 调用成功,则处理业务逻辑
            if (response.isSuccess()) {
                return true;
            }
        } catch (Exception e) {
            e.printStackTrace();
            return false;
        }
        return false;
    }
    /**
     * 获取支付信息支付(签约app支付2.0)
     *
     * @param subject
     * @param body
     * @param price
     * @param request
     * @return
     */
    public  R<Map<String, Object>> alipay(String orderNo, String subject, String body, Double price,
                                             HttpServletRequest request) {
        try {
            // 接口封装支付宝请求参数
            // Map<String, Object> mData = new HashMap<String, Object>();
            // 构建请求支付签名参数
            Map<String, Object> pay = PayDemoActivity.appPay(subject, body, price, orderNo);
            /*
             * Set<Entry<String, String>> entrySet = pay.entrySet();
             * Iterator<Entry<String, String>> iterator = entrySet.iterator();
             * while (iterator.hasNext()) { Entry<String, String> next =
             * iterator.next(); mData.put(next.getKey(), next.getValue()); }
             */
            return R.ok(pay);
        } catch (Exception e) {
            e.printStackTrace();
            return R.fail("支付异常!");
        }
    }
    /**
     * 服务器异步通知处理支付宝
     *
     * @param request
     * @param res
     */
    @RequestMapping("/alipay/notify")
    public void notifyUrl(HttpServletRequest request, HttpServletResponse res) {
        HttpServletResponse response = (HttpServletResponse) res;
        response.setContentType("text/html;charset=UTF-8");
        PrintWriter out;
        try {
            out = response.getWriter();
            // 获取支付宝POST过来反馈信息
            Map<String, String> params = new HashMap<String, String>();
            Map requestParams = request.getParameterMap();
            log.debug("AlipayController.notifyUrl__requestParams:\n" + requestParams);
            for (Iterator iter = requestParams.keySet().iterator(); iter.hasNext();) {
                String name = (String) iter.next();
                String[] values = (String[]) requestParams.get(name);
                String valueStr = "";
                for (int i = 0; i < values.length; i++) {
                    valueStr = (i == values.length - 1) ? valueStr + values[i] : valueStr + values[i] + ",";
                }
                // 乱码解决,这段代码在出现乱码时使用。如果mysign和sign不相等也可以使用这段代码转化
                valueStr = new String(valueStr.getBytes("ISO-8859-1"), "gbk");
                params.put(name, valueStr);
            }
            Paylog paylog = getPayLog_alipay(request);
            // 验证成功
            if ("TRADE_FINISHED".equals(paylog.getTradeStatus())) {
                System.out.println("AlipayController.notifyUrl__验证成功:success");
                // 支付失败
            } else if ("TRADE_SUCCESS".equals(paylog.getTradeStatus())) {
                // 支付成功
                try {
                    if(judgeContainsStr(paylog.getOutTradeNo())){
                    }
                    // ——请根据您的业务逻辑来编写程序(以上代码仅作参考)——
                    log.debug("AlipayController.notifyUrl__回调处理:success");
                    out.println("success"); // 请不要修改或删除
                // ——请根据您的业务逻辑来编写程序(以上代码仅作参考)——
                log.debug("AlipayController.notifyUrl__回调处理:success");
                out.println("success"); // 请不要修改或删除
            } catch (Exception e) {
                log.debug("AlipayController.notifyUrl__回调逻辑代码处理异常!fail");
                // 返回失败
                out.println("fail");
                e.printStackTrace();
            }
            // ////////////////////////////////////////////////////////////////////////////////////////
        } else {// 验证失败
            log.debug("AlipayController.notifyUrl__回调处理失败!fail");
            out.println("fail");
        }
    } catch (
    IOException e) {
        log.debug("AlipayController.notifyUrl__支付宝服务器异步通知数据处理失败!");
        e.printStackTrace();
    }
}
///////////////////////////////////////////////////////////////////////////////////////////////////
/**
 * 统一下单
 *
 * @param apptype
 * @param outTradeNo
 * @param body
 * @param price
 * @param request
 * @return
 */
public static R<Map<String, Object>> wxpay(Integer apptype, String outTradeNo, String body, Double price,
                                        HttpServletRequest request) {
    // 获取预支付接口返回参数
    Map<String, Object> map = new HashMap<String, Object>();
    Map<String, Object> appPayMap = new HashMap<String, Object>();
    try {
        // 构建接口请求参数
        UnifiedorderReqData unifiedorderReqData = new UnifiedorderReqData(apptype, outTradeNo, body, price,
                Configure.wx_notify_url);
        // 请求接口获取返回接口
        String result = requestUnifiedorderService(apptype, unifiedorderReqData);
        System.out.println(result);
        System.out.println("WxpayController.createOrder__result:\n" + result);
        // 获取预支付接口返回参数
        map = XMLParser.getMapFromXML(result);
        System.out.println("WxpayController.createOrder__result:\n" + result);
        // 捕获预支付接口错误提示
        if ("FAIL".equals(map.get("result_code")) || "FAIL".equals(map.get("return_code"))) {
            return R.fail(String.valueOf(map.get("return_msg")));
        }
        // 对获取预支付返回接口参数进行封装(生成支付订单接口数据)
        AppPayReqData appPay = new AppPayReqData(apptype, (String) map.get("appid"), (String) map.get("mch_id"),
                (String) map.get("prepay_id"), unifiedorderReqData.getNonce_str());
        // 对获取预支付返回接口参数进行封装(生成支付订单接口数据)
        appPayMap.put("appid", appPay.getAppid());// 公众账号ID
        appPayMap.put("nonceStr", appPay.getNoncestr());// 随机字符串(32位)
        appPayMap.put("package", appPay.get_package());// 扩展字段(暂填写固定值Sign=WXPay)
        appPayMap.put("partnerId", appPay.getPartnerid());// 商户号
        appPayMap.put("prepayId", appPay.getPrepayid());// 预支付编号(微信返回的支付交易会话ID)
        appPayMap.put("timeStamp", appPay.getTimestamp());// 时间戳
        appPayMap.put("sign", appPay.getSign());// 根据API给的签名规则进行签名
        return R.ok(appPayMap);
    } catch (Exception e) {
        System.out.println("统一下单_API_处理异常!");
        e.printStackTrace();
    }
    return R.fail("统一下单失败");
}
/**
 * 微信支付回调(参考财付通回调接口)
 *
 * @param request
 * @param response
 */
@RequestMapping("/wxpay/notify")
public void wxnotify(HttpServletRequest request, HttpServletResponse response) {
    try {
        System.out.println("微信支付回调!!!!!!!!!!!!!!!!!!!!!!!!!");
        // 异步通知返回报文
        StringBuffer notityXml = new StringBuffer();
        String inputLine;
        while ((inputLine = request.getReader().readLine()) != null) {
            notityXml.append(inputLine);
        }
        request.getReader().close();
        // log.debug("WxpayController.notify__notityXml:\n" + notityXml);
        System.out.println("WxpayController.notify__notityXml:\n" + notityXml);
        // 验证签名
        if (Signature.checkIsSignValidFromResponseString(1, notityXml.toString())
                || Signature.checkIsSignValidFromResponseString(2, notityXml.toString())) {
            Map<String, Object> map = XMLParser.getMapFromXML(notityXml.toString());
            // log.debug("WxpayController.notify__map:\n" + map);
            // 接口返回状态
            String result_code = (String) map.get("result_code");
            if ("SUCCESS".equals(result_code)) {
                // // 商户订单号
                String out_trade_no = (String) map.get("out_trade_no");
                // // 微信支付交易号
                String trade_no = (String) map.get("transaction_id");
                // // 金额,以分为单位
                String total_fee = (String) map.get("total_fee");
                // // 优惠金额
                // String discount = (String) map.get("discount");
                // // 支付完成时间
                String time_end = (String) map.get("time_end");
                // // 支付者唯一Id(对应买家账号的一个加密串 )
                String buyer_id = (String) map.get("buyer_id");
                ///////////////////////////// 这里程序处理支付回调逻辑
                ///////////////////////////// ////////////////////
                Paylog paylog = new Paylog();
                paylog.setOutTradeNo(out_trade_no);
                paylog.setPayType(2);
                paylog.setBuyerId(buyer_id);
                paylog.setTradeNo(trade_no);
                paylog.setPayMoney(Double.parseDouble(total_fee) / 100);
                paylog.setState(1);
                System.out.println("支付完成时间"+time_end);
                paylog.setAddTime(LocalDateTime.now());
                iPaylogService.save(paylog);
                if(judgeContainsStr(paylog.getOutTradeNo())){
                    log.debug("WxpayController.notify__回调处理成功:SUCCESS");
                    response.getOutputStream().print("success");
                }
                log.debug("WxpayController.notify__回调处理成功:SUCCESS");
                response.getOutputStream().print("success");
            } else {
                log.debug("WxpayController.notify__回调处理:验证状态错误!" + result_code);
                System.out.println("验证状态错误!" + result_code);
            }
        } else {
            log.debug("WxpayController.notify__回调处理:通知签名验证失败!");
            System.out.println("通知签名验证失败!");
        }
    } catch (Exception e) {
        e.printStackTrace();
    }
}
/**
 * 微信退款
 *
 * @param transactionID
 *            【支付交易号】是微信系统为每一笔支付交易分配的订单号,通过这个订单号可以标识这笔交易,
 *            它由支付订单API支付成功时返回的数据里面获取到。建议优先使用
 * @param outTradeNo
 *            【商品订单编号】商户系统内部的订单号,transaction_id
 *            、out_trade_no二选一,如果同时存在优先级:transaction_id>out_trade_no
 * @param outRefundNo
 *            【退款编号】商户系统内部的退款单号,商户系统内部唯一,同一退款单号多次请求只退一笔
 *            订单总金额,单位为分
 * @param refundFee
 *            退款总金额,单位为分,可以做部分退款
 * @return
 */
public static boolean refundForWxpay(Integer apptype, String transactionID, String outTradeNo, String outRefundNo,
                                     Integer totalFee, Integer refundFee, String pay_type) {
    Map<String, Object> map = new HashMap<String, Object>();
    try {
        // 构建接口请求参数
        com.tencent.protocol.RefundReqData refundReqData = new com.tencent.protocol.RefundReqData(transactionID, outTradeNo, outRefundNo, totalFee, refundFee,
                pay_type);
        // 请求接口返回结果
        String result = requestRefundService(apptype, refundReqData);
        System.out.println("微信退款返回内容:" + result);
        // 获取预支付接口返回参数
        map = XMLParser.getMapFromXML(result);
        System.out.println("微信退款返回参数:" + map);
        // 退款成功处理
        if ("SUCCESS".equals(map.get("result_code"))) {
            return true;
        }
    } catch (Exception e) {
        e.printStackTrace();
        return false;
    }
    return false;
}
/**
 * 获取支付宝
 *
 * @param request
 * @return
 * @throws Exception
 */
private Paylog getPayLog_alipay(HttpServletRequest request) throws IOException {
    //////// 获取支付宝的通知返回参数,可参考技术文档中页面跳转同步通知参数列表(以下仅供参考)////////
    // 商户订单号
    String out_trade_no = new String(request.getParameter("out_trade_no").getBytes("ISO-8859-1"), "UTF-8");
    // 支付宝交易号
    String trade_no = new String(request.getParameter("trade_no").getBytes("ISO-8859-1"), "UTF-8");
    // 交易状态
    String trade_status = new String(request.getParameter("trade_status").getBytes("ISO-8859-1"), "UTF-8");
    // 支付者唯一Id
    String buyer_id = new String(request.getParameter("buyer_id").getBytes("ISO-8859-1"), "UTF-8");
    // 支付帐号
    String buyer_email = "";
    if (SinataUtil.isNotEmpty(request.getParameter("buyer_logon_id"))) {
        buyer_email = new String(request.getParameter("buyer_logon_id").getBytes("ISO-8859-1"), "UTF-8");
    }
    // 支付金额
    String total_fee = new String(request.getParameter("total_amount").getBytes("ISO-8859-1"), "UTF-8");
    // 支付时间
    String notify_time = new String(request.getParameter("notify_time").getBytes("ISO-8859-1"), "UTF-8");
    //////// 获取支付宝的通知返回参数,可参考技术文档中页面跳转同步通知参数列表(以下仅供参考)////////
    Paylog paylog = new Paylog();
    paylog.setOutTradeNo(out_trade_no);
    paylog.setPayType(1);
    paylog.setBuyerId(buyer_id);
    paylog.setTradeNo(trade_no);
    paylog.setPayMoney(Double.parseDouble(total_fee));
    paylog.setState(1);
    paylog.setAddTime(LocalDateTime.now());
    paylog.setTradeStatus(trade_status.toString());
    iPaylogService.save(paylog);
    return paylog;
}
public boolean judgeContainsStr(String cardNum) {
    String regex=".*[a-zA-Z]+.*";
    Matcher m= Pattern.compile(regex).matcher(cardNum);
    return m.matches();
}
}
ruoyi-modules/ruoyi-order/src/main/java/com/ruoyi/order/util/tencent/WXPay.java
@@ -1,17 +1,17 @@
package com.tencent;
package com.ruoyi.order.util.tencent;
import java.util.Map;
import com.tencent.common.Configure;
import com.tencent.common.WXPayConfig;
import com.tencent.common.WXPayConstants;
import com.tencent.common.WXPayConstants.SignType;
import com.tencent.common.WXPayUtil;
import com.ruoyi.order.util.tencent.common.Configure;
import com.ruoyi.order.util.tencent.common.WXPayConfig;
import com.ruoyi.order.util.tencent.common.WXPayConstants;
import com.ruoyi.order.util.tencent.common.WXPayUtil;
import com.ruoyi.order.util.tencent.protocol.UnifiedorderReqData;
import com.ruoyi.order.util.tencent.protocol.WXPayRequest;
import com.ruoyi.order.util.tencent.service.RefundService;
import com.ruoyi.order.util.tencent.service.UnifiedorderService;
import com.tencent.protocol.RefundReqData;
import com.tencent.protocol.UnifiedorderReqData;
import com.tencent.protocol.WXPayRequest;
import com.tencent.service.RefundService;
import com.tencent.service.UnifiedorderService;
/**
 * SDK总入口
 */
@@ -21,7 +21,7 @@
public class WXPay {
    
      private WXPayConfig config;
        private SignType signType;
        private WXPayConstants.SignType signType;
        private boolean autoReport;
        private boolean useSandbox;
        private String notifyUrl;
@@ -54,10 +54,10 @@
            this.autoReport = autoReport;
            this.useSandbox = useSandbox;
            if (useSandbox) {
                this.signType = SignType.MD5; // 沙箱环境
                this.signType = WXPayConstants.SignType.MD5; // 沙箱环境
            }
            else {
                this.signType = SignType.HMACSHA256;
                this.signType = WXPayConstants.SignType.HMACSHA256;
            }
            this.wxPayRequest = new WXPayRequest(config);
        }
@@ -84,7 +84,7 @@
    /**
     * 请求统一下单服务
     */
    public static String requestUnifiedorderService(Integer apptype,UnifiedorderReqData unifiedorderReqData) throws Exception{
    public static String requestUnifiedorderService(Integer apptype, UnifiedorderReqData unifiedorderReqData) throws Exception{
        return new UnifiedorderService(apptype).request(unifiedorderReqData);
    }
    
@@ -219,12 +219,12 @@
     * @throws Exception
     */
    public Map<String, String> fillRequestData(Map<String, String> reqData) throws Exception {
        reqData.put("appid", Configure.getAppid());
        reqData.put("appid", Configure.getAppid());
        reqData.put("mch_id", Configure.getMchid());
        reqData.put("nonce_str", WXPayUtil.generateUUID());
        reqData.put("sign_type", WXPayConstants.MD5);
        //reqData.put("sign_type", WXPayConstants.HMACSHA256);
        reqData.put("sign", WXPayUtil.generateSignature(reqData, Configure.getKey(), SignType.MD5));
        reqData.put("sign", WXPayUtil.generateSignature(reqData, Configure.getKey(), WXPayConstants.SignType.MD5));
        return reqData;
    }