支付宝沙箱的应用

支付宝沙箱提供了一套模拟真实环境下的支付流程。大家想需要在项目中开发支付模块,可以采用支付宝沙箱进行开发是一个不错的选择

1.登录支付宝开放平台,然后开启沙箱功能

地址:支付宝开放平台 (alipay.com)

2.进入控制台,找到沙箱后获取或是查看密钥。密钥就是你进行沙箱开发的凭证

3.通过springboot开发支付沙箱,则加入支付宝沙箱的依赖

<!--支付宝SDK-->
<dependency>
  <groupId>com.alipay.sdk</groupId>
  <artifactId>alipay-sdk-java</artifactId>
  <version>4.22.110.ALL</version>
</dependency>

4.在application.yml文件中配置支付宝沙箱密钥

alipay:
  appId: 90x546
  appPrivateKey: MIIEvQH7i68TO1liXx448udTaH72+JgyBxfB4EkpIPREjZIc7h+Cxsc0POtOc1eLgOgkxYbM/01iaLPJ8aXh9RoFgwMwycD1rIuwTt1zpOShcrNk=
  alipayPublicKey: MIIBIGtkJ6YEgFO+lG2GIAOnkJYOZxpFfcjLpHb8ICejV0J6/pn+j9H3gaQWVcb1yTFHNByrT6heFFJmdbLcJsffqqbYr8T2hbtQIDAQAB
  notifyUrl: http://localhost:8080/bookorder/notify
  returnUrl: http://localhost:8080/index-user-order.html

appId:开发账号

appPrivateKey:系统私钥

alipayPublicKey:系统公钥

notifyUrl:支付成功或失败后的回调接口,这个接口一般需要进行内网穿透 

returnUrl:支付成功后,返回订单页面的地址

5.创建配置类,用于读取沙箱信息的配置

package com.liuyang.common;

import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;

@Component
@Data
@ConfigurationProperties(prefix = "alipay")
public class AlipayConfig {
    private String appId; // 支付宝应用Id,需要自行申请
    private String appPrivateKey; // 支付宝私钥
    private String alipayPublicKey; // 支付宝公钥
    private String notifyUrl; // 内网穿透地址, 支付接口的响应地址
    private String returnUrl; // 返回地址,本地应用返回订单页面地址
}

6.开发支付接口

(1) 接口中先定义支付的网关

// 支付宝网关
private static final String GATEWAY_URL = "https://openapi-sandbox.dl.alipaydev.com/gateway.do";
// 数据格式
private static final String FORMAT = "JSON";
// 中文编码
private static final String CHARSET = "UTF-8";
//签名方式
private static final String SIGN_TYPE = "RSA2";

(2)我们以图书商品为例,进行支付接口调用

//支付宝接口
    @GetMapping("/pay")
    public void pay(Bookorder bookorder, HttpServletResponse response) throws IOException {
        // 1. 创建Client,通用SDK提供的Client,负责调用支付宝的API
        AlipayClient alipayClient = new DefaultAlipayClient(GATEWAY_URL, alipayConfig.getAppId(), alipayConfig.getAppPrivateKey(), FORMAT, CHARSET, alipayConfig.getAlipayPublicKey(), SIGN_TYPE);
        // 2. 创建 Request并设置Request参数
        AlipayTradePagePayRequest request = new AlipayTradePagePayRequest();  // 发送请求的 Request类
        request.setNotifyUrl(alipayConfig.getNotifyUrl());
        request.setReturnUrl(alipayConfig.getReturnUrl());
        // 3. 创建JSONObject对象,用于封装JSON对象,并将JSON对象转为字符串并存储到request
        JSONObject bizContent = new JSONObject();
        bizContent.set("out_trade_no", bookorder.getId()); // 订单ID
        bizContent.set("total_amount", bookorder.getOrderPrice().doubleValue()); // 订单总金额
        bizContent.set("subject", bookorder.getBookId()); // 商品的ISBN编号, 支付名称
        bizContent.set("product_code", "FAST_INSTANT_TRADE_PAY");  // 固定配置
        request.setBizContent(bizContent.toString()); // 转为字符串进行存储

        // 执行请求,拿到响应的结果,返回给浏览器
        String form = "";
        try {
            form = alipayClient.pageExecute(request).getBody(); // 调用SDK生成表单
        } catch (AlipayApiException e) {
            e.printStackTrace();
        }
        response.setContentType("text/html;charset=" + CHARSET);
        PrintWriter out = response.getWriter();
        out.write(form);// 直接将完整的表单html输出到页面
        out.flush();
        out.close();

        // 这段代码可以省略,只是在无内网穿透的情况模拟更新数据库
        bookorder.setOrderStatus("1");
        bookorder.setOrderPayTime(LocalDateTime.now());
        bookorderService.saveOrUpdate(bookorder);
    }

7.支付成功或失败的回调接口

// 支付接口完成后的响应地址
    // 需要做内网穿透,参考地址 https://natapp.cn/login
    @PostMapping("/notify")
    public String payNotify(HttpServletRequest request) throws AlipayApiException {
        System.out.println("=========支付宝异步回调准备========");
        if (request.getParameter("trade_status").equals("TRADE_SUCCESS")) {
            System.out.println("=========支付宝异步回调========");
            Map<String, String> params = new HashMap<>();
            Map<String, String[]> requestParams = request.getParameterMap();
            for (String name : requestParams.keySet()) {
                params.put(name, request.getParameter(name));
                // System.out.println(name + " = " + request.getParameter(name));
            }

            // 返回订单编号
            String outTradeNo = params.get("out_trade_no");
            // 交易号
            String alipayTradeNo = params.get("trade_no");

            String sign = params.get("sign");
            String content = AlipaySignature.getSignCheckContentV1(params);
            boolean checkSignature = AlipaySignature.rsa256CheckContent(content, sign, alipayConfig.getAlipayPublicKey(), "UTF-8"); // 验证签名
            // 支付宝验签
            if (checkSignature) {
                // 验签通过
                System.out.println("交易名称: " + params.get("subject"));
                System.out.println("交易状态: " + params.get("trade_status"));
                System.out.println("支付宝交易凭证号: " + params.get("trade_no"));
                System.out.println("商户订单号: " + params.get("out_trade_no"));
                System.out.println("交易金额: " + params.get("total_amount"));
                System.out.println("买家在支付宝唯一id: " + params.get("buyer_id"));
                System.out.println("买家付款时间: " + params.get("gmt_payment"));
                System.out.println("买家付款金额: " + params.get("buyer_pay_amount"));
            }
        }
        return "success";
    }

8.前端调用支付请求的代码

window.open("http://localhost:8080/bookorder/pay?bookId=" + data.field.bookId + "&id=" + data.field.id + "&orderPrice=" + data.field.orderPrice)

 

posted @ 2023-11-07 10:31  子墨老师  阅读(137)  评论(0编辑  收藏  举报