目录
前提
使用支付宝沙箱模拟支付需要使用到公网IP,没有的可以使用内网穿透,我这里演示使用的是cpolar生成的IP地址,cpolar软件的下载、使用方式我在Windows上实现内网穿透这一片文章中介绍了,非常简洁。
得到账号信息
首先进入支付宝开放平台
登录之后,跟着下面图示的步骤进行,得到支付宝沙箱账号信息
进入支付宝开放平台,搜索沙箱,选择沙箱环境。
然后在沙箱环境中可以查看自己的 APPID、应用公钥、应用私钥、支付宝公钥。 将这些关键信息配置在工程中。
在代码中实现
第一步
首先将我们从上面得到的信息封装到一个bean实例中,并在其中实现pay(方法名可以不同)方法,方法中的基本内容大致不变。
@Component
@Data
public class AlipayTemplate {
//在支付宝创建的应用的id
private String app_id = "自己的id";
// 商户私钥,您的PKCS8格式RSA2私钥
private String merchant_private_key = "自己的私钥";
// 支付宝公钥,查看地址:https://openhome.alipay.com/platform/keyManage.htm 对应APPID下的支付宝公钥。
private String alipay_public_key = "自己的公钥";
// 服务器[异步通知]页面路径 需http://格式的完整路径,不能加?id=123这类自定义参数,必须外网可以正常访问
// 支付宝会悄悄的给我们发送一个请求,告诉我们支付成功的信息 post请求方式
// 这里需要公网IP,如果没有的可以使用内网穿透获得
private String notify_url="https://24434cel.r3.cpolar.cn/payed/notify";
// 页面跳转同步通知页面路径 需http://格式的完整路径,不能加?id=123这类自定义参数
//同步通知,支付成功,一般跳转到成功页
private String return_url="http://localhost:8081/payed/success";
// 签名方式
private String sign_type = "RSA2";
// 字符编码格式
private String charset = "utf-8";
// 支付宝网关; https://openapi.alipaydev.com/gateway.do
private String gatewayUrl = "https://openapi-sandbox.dl.alipaydev.com/gateway.do";
public String pay(Pay vo) throws AlipayApiException {
//AlipayClient alipayClient = new DefaultAlipayClient(AlipayTemplate.gatewayUrl, AlipayTemplate.app_id, AlipayTemplate.merchant_private_key, "json", AlipayTemplate.charset, AlipayTemplate.alipay_public_key, AlipayTemplate.sign_type);
//1、根据支付宝的配置生成一个支付客户端
AlipayClient alipayClient = new DefaultAlipayClient(gatewayUrl,
app_id, merchant_private_key, "json",
charset, alipay_public_key, sign_type);
//2、创建一个支付请求 //设置请求参数
AlipayTradePagePayRequest alipayRequest = new AlipayTradePagePayRequest();
alipayRequest.setReturnUrl(return_url);
alipayRequest.setNotifyUrl(notify_url);
//商户订单号,商户网站订单系统中唯一订单号,必填
String out_trade_no = vo.getTradeNo();
//付款金额,必填
String total_amount = vo.getAmount();
//订单名称,必填
String subject = vo.getSubject();
//商品描述,可空
String body = vo.getScript();
alipayRequest.setBizContent("{\"out_trade_no\":\""+ out_trade_no +"\","
+ "\"total_amount\":\""+ total_amount +"\","
+ "\"subject\":\""+ subject +"\","
+ "\"body\":\""+ body +"\","
+ "\"timeout_express\":\"1m\","
+ "\"product_code\":\"FAST_INSTANT_TRADE_PAY\"}");
String result = alipayClient.pageExecute(alipayRequest).getBody();
//会收到支付宝的响应,响应的是一个页面,只要浏览器显示这个页面,就会自动来到支付宝的收银台页面
System.out.println("支付宝的响应:"+result);
return result;
}
}
第二步
在pay方法中,我们使用内网穿透得到的一个ip地址异步请求后台的/payed/notify接口,在这个接口中接收到支付宝的异步通知,最简单的就是直接返回success即可
@ResponseBody
@RequestMapping("/payed/notify")
public String handleAlipayed(PayAsyncVo asyncVo, HttpServletRequest request) throws AlipayApiException, UnsupportedEncodingException {
// 只要收到支付宝的异步通知,返回 success 支付宝便不再通知
// 可以在这里面进行一些 验证逻辑,比如验证签名是否正确,解决乱码问题等
return "success";
}
也可以在里面进行一些验证操作
@ResponseBody
@RequestMapping("/payed/notify")
public String handleAlipayed(PayAsyncVo asyncVo, HttpServletRequest request) throws AlipayApiException, UnsupportedEncodingException {
// 只要收到支付宝的异步通知,返回 success 支付宝便不再通知
// 获取支付宝POST过来反馈信息
//TODO 验签
Map<String, String> params = new HashMap<>();
Map<String, String[]> requestParams = request.getParameterMap();
for (String name : requestParams.keySet()) {
String[] values = requestParams.get(name);
String valueStr = "";
for (int i = 0; i < values.length; i++) {
valueStr = (i == values.length - 1) ? valueStr + values[i]
: valueStr + values[i] + ",";
}
//乱码解决,这段代码在出现乱码时使用
// valueStr = new String(valueStr.getBytes("ISO-8859-1"), "utf-8");
params.put(name, valueStr);
}
boolean signVerified = AlipaySignature.rsaCheckV1(params, alipayTemplate.getAlipay_public_key(),
alipayTemplate.getCharset(), alipayTemplate.getSign_type()); //调用SDK验证签名
//如果验签无误
if (signVerified) {
System.out.println("签名验证成功...");
//去修改订单状态
// String result = orderService.handlePayResult(asyncVo);
// return result;
return "success";
} else {
System.out.println("签名验证失败...");
return "error";
}
}
@ToString
@Data
public class PayAsyncVo {
private String gmt_create;
private String charset;
private String gmt_payment;
private Date notify_time;
private String subject;
private String sign;
private String buyer_id;//支付者的id
private String body;//订单的信息
private String invoice_amount;//支付金额
private String version;
private String notify_id;//通知id
private String fund_bill_list;
private String notify_type;//通知类型; trade_status_sync
private String out_trade_no;//订单号
private String total_amount;//支付的总额
private String trade_status;//交易状态 TRADE_SUCCESS
private String trade_no;//流水号
private String auth_app_id;//
private String receipt_amount;//商家收到的款
private String point_amount;//
private String app_id;//应用id
private String buyer_pay_amount;//最终支付的金额
private String sign_type;//签名类型
private String seller_id;//商家的id
}
反正看个人需要,业务有要求的话就在其中加一些逻辑处理,没有要求的话就怎么简单怎么来。
第三步
在模拟支付成功之后,支付宝会调用后台的一个接口,这个接口就不需要内网传递,直接使用自己电脑的地址(localhost)就可以。
@ResponseBody
@RequestMapping("/payed/success")
public String paySucess(){
return "支付成功";
}
第四步
然后在需要支付服务的地方调用 pay 方法就可以了,pay 方法会返回一个页面的 前端代码,我们只需要把这些前端代码以 text/html 的形式返回给浏览器页面, 浏览器界面就可以生成前端代码对应的前端界面,就可以完成沙箱支付的流程。
@Resource
private AlipayTemplate alipayTemplate;
@ResponseBody
@GetMapping(value = "/payOrder",produces = "text/html")
public String payOrder() throws AlipayApiException {
//这些信息应该是根据自己业务的订单信息中得到的,这里为了简单,就直接设置了
//pay这个类是我自己定义的,只要包含下面这几个必要信息就行,对名字没有要求,是这个意思就行
Pay pay = new Pay();
pay.setAmount("1000");//金额-必须
pay.setScript("这是一个测试用例");//描述-可有可无
pay.setSubject("测试");//标题-必须
pay.setTradeNo("121212121212");//订单号-必须,并且每个订单号只能支付成功一次,每一次支付成功之后,下一次再以这个订单号进行提交,会显示当前订单已支付。
return alipayTemplate.pay(pay);
}
@Data
public class Pay {
private String tradeNo;
private String amount;
private String subject;
private String script;
}
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<form method="get" action="http://localhost:8081/payOrder">
<button type="submit">支付</button>
</form>
</body>
</html>
演示
在依次实现了上面的1~4步之后,我们就可以使用支付宝的模拟支付了
点击前端页面的支付
输入自己在支付宝开放平台设置的账户名、密码,点击下一步
点击确认付款
到这里支付宝模拟支付就成功了。