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:
# 配置项1
appId:
# 配置项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;
@PostConstruct
public 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 {
@Resource
AliPayConfig 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博客