Java支付宝沙箱环境配置(测试)
1. 沙箱配置环境
沙箱应用 - 开放平台 (alipay.com)
2. 需要用到的基本信息
3. Pom文件添加依赖
<!--支付宝依赖 --><dependency><groupId>com.alipay.sdk</groupId><artifactId>alipay-easysdk</artifactId><version>2.2.0</version></dependency><dependency><groupId>com.alipay.sdk</groupId><artifactId>alipay-sdk-java</artifactId><version>4.22.113.ALL</version></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId></dependency>
4. 添加配置文件
alipay:# 配置项1appId: # 配置项2 appPrivateKey: # 配置项3 alipayPublicKey: # 配置项4 notifyUrl:
点击查看可以看到私钥和支付宝公钥
关于配置项4, 这里需要使用到 “内网穿透”, 测试推荐: https://natapp.cn/
官方配置教程: https://natapp.cn/article/natapp_newbie
内网穿透说明: 因为电脑的ip是内网ip, 所以在公网没有办法直接访问, 这个时候需要用到内网穿透技术, 通过访问特定网址, 等同于访问本机服务, 启动上面配置之后可以看到如下信息
http://78b3ci.natappfree.cc -> 127.0.0.1:8880
即访问前面的网址, 等同于访问本机的8880服务
而配置项4, 使用的时机是当买方支付成功或失败后,支付宝官方需要通过这个接口告诉你结果 即 回调接口
5. 配置类
@Data
@Component
//读取yml文件中alipay 开头的配置
@ConfigurationProperties(prefix = "alipay")
public class AliPayConfig {private String appId;private String appPrivateKey;private String alipayPublicKey;private String notifyUrl;@PostConstructpublic void init() {// 设置参数(全局只需设置一次)Config config = new Config();config.protocol = "https";config.gatewayHost = "openapi.alipaydev.com";config.signType = "RSA2";config.appId = this.appId;config.merchantPrivateKey = this.appPrivateKey;config.alipayPublicKey = this.alipayPublicKey;config.notifyUrl = this.notifyUrl;Factory.setOptions(config);System.out.println("=======支付宝SDK初始化成功=======");}
}
消息实体类:
@Data
public class AliPay {/*** 商户订单号 */private String traceNo;/*** 订单金额*/private double totalAmount;/*** 订单标题*/private String subject;private String alipayTraceNo;
}
相关接口:
@RestController
@RequestMapping("alipay")
@Transactional(rollbackFor = Exception.class)
public class AliPayController {@ResourceAliPayConfig aliPayConfig;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";@GetMapping("test")@ApiOperation("测试")public String test() {return "ok";}@GetMapping("/pay") // &subject=xxx&traceNo=xxx&totalAmount=xxx@ApiOperation("支付宝支付")public void pay(AliPay aliPay, HttpServletResponse httpResponse) throws Exception {AlipayClient alipayClient = new DefaultAlipayClient(GATEWAY_URL, aliPayConfig.getAppId(),aliPayConfig.getAppPrivateKey(), FORMAT, CHARSET, aliPayConfig.getAlipayPublicKey(), SIGN_TYPE);AlipayTradePagePayRequest request = new AlipayTradePagePayRequest();request.setNotifyUrl(aliPayConfig.getNotifyUrl());request.setBizContent("{\"out_trade_no\":\"" + aliPay.getTraceNo() + "\","+ "\"total_amount\":\"" + aliPay.getTotalAmount() + "\","+ "\"subject\":\"" + aliPay.getSubject() + "\","+ "\"product_code\":\"FAST_INSTANT_TRADE_PAY\"}");String form = "";try {// 调用SDK生成表单form = alipayClient.pageExecute(request).getBody();} catch (AlipayApiException e) {e.printStackTrace();}httpResponse.setContentType("text/html;charset=" + CHARSET);// 直接将完整的表单html输出到页面httpResponse.getWriter().write(form);httpResponse.getWriter().flush();httpResponse.getWriter().close();}/*** 支付宝回调返回结果接口* */@PostMapping("/notify") // 注意这里必须是POST接口public String payNotify(HttpServletRequest request) throws Exception {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));}String tradeNo = params.get("out_trade_no");String gmtPayment = params.get("gmt_payment");String alipayTradeNo = params.get("trade_no");// 支付宝验签if (Factory.Payment.Common().verifyNotify(params)) {// 验签通过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";}
}
到这里基础的支付已经完成了, 具体的业务逻辑可以在这个框架基础上面修改, 比如更新订单状态.等, 下面我们开始测试
**注:**回调方法需要配置不用token
6. 测试
可以先调一下测试接口, 测试内网穿透是否成功
接口: http://78b3ci.natappfree.cc/alipay/test
没有问题, 下面在调用测试接口
付款接口: http://localhost:8880/alipay/pay?subject=iPhone15&totalAmount=6666&traceNo=100000000001
需要注意: traceNo系统唯一 不能重复 不能重复 不能重复!!!
调用接口后, 跳转返回界面如下:
在这里支付宝也贴心的给了一个买方账户:
沙箱账号: https://open.alipay.com/develop/sandbox/account
输入买家账号密码之后跳转新的界面
输入密码确认支付
查看控制台:
=========支付宝异步回调========
交易名称: iPhone15
交易状态: TRADE_SUCCESS
支付宝交易凭证号: 2024061922001400840503224710
商户订单号: 100000000001
交易金额: 6666.00
买家在支付宝唯一id: 2088722033400843
买家付款时间: 2024-06-19 14:09:20
买家付款金额: 6666.00
到此支付完成
代码参考: 2024年最新版 springboot+vue整合支付宝沙箱支付功能,一步一步带您实现完整的支付宝支付功能_vue支付宝支付-CSDN博客