Skip to content

Commit c94a8cf

Browse files
nadirvishunvishun
and
vishun
authored
增加v3版本部分配置和示例 (#132)
* 增加v3版本部分配置和示例。 * readme中增加查看多商户的提示说明 * 优化版本升级后原有v2部分方法抛出异常 --------- Co-authored-by: vishun <[email protected]>
1 parent 7cbea4c commit c94a8cf

File tree

7 files changed

+283
-13
lines changed

7 files changed

+283
-13
lines changed

README.md

+2
Original file line numberDiff line numberDiff line change
@@ -34,3 +34,5 @@
3434
1. 打开网页浏览器,输入想要访问的地址,比如`http://localhost:8080/pay/closeOrder/123`查看效果。
3535
1. 还可以访问 http://localhost:8080/swagger-ui.html 来查看所有可用接口,并进行在线调试;
3636
1. 更多接口使用说明,请参考 [wiki 文档](https://github.com/Wechat-Group/weixin-java-tools/wiki/%E5%BE%AE%E4%BF%A1%E6%94%AF%E4%BB%98)
37+
## 多商户:
38+
- 请切换分支`multiple`来查看多商户的示例代码。

pom.xml

-6
Original file line numberDiff line numberDiff line change
@@ -18,14 +18,8 @@
1818
<description>微信支付 Demo with Spring Boot</description>
1919

2020
<properties>
21-
<<<<<<< HEAD
22-
<weixin-java-pay.version>4.6.0</weixin-java-pay.version>
23-
<io.springfox.version>2.10.0</io.springfox.version>
24-
=======
2521
<weixin-java-pay.version>4.7.0</weixin-java-pay.version>
2622
<io.springfox.version>2.5.0</io.springfox.version>
27-
>>>>>>> 1d97864 (:arrow_up: 升级sdk版本为4.7.0)
28-
2923
<maven.compiler.source>1.8</maven.compiler.source>
3024
<maven.compiler.target>1.8</maven.compiler.target>
3125
<maven.compiler.encoding>UTF-8</maven.compiler.encoding>

src/main/java/com/github/binarywang/demo/wx/pay/config/WxPayConfiguration.java

+8-2
Original file line numberDiff line numberDiff line change
@@ -26,12 +26,18 @@ public class WxPayConfiguration {
2626
@ConditionalOnMissingBean
2727
public WxPayService wxService() {
2828
WxPayConfig payConfig = new WxPayConfig();
29-
payConfig.setAppId(StringUtils.trimToNull(this.properties.getAppId()));
30-
payConfig.setMchId(StringUtils.trimToNull(this.properties.getMchId()));
29+
payConfig.setAppId(StringUtils.trimToNull(this.properties.getAppId()));//V3商户模式需要
30+
payConfig.setMchId(StringUtils.trimToNull(this.properties.getMchId()));//V3商户模式需要
3131
payConfig.setMchKey(StringUtils.trimToNull(this.properties.getMchKey()));
3232
payConfig.setSubAppId(StringUtils.trimToNull(this.properties.getSubAppId()));
3333
payConfig.setSubMchId(StringUtils.trimToNull(this.properties.getSubMchId()));
3434
payConfig.setKeyPath(StringUtils.trimToNull(this.properties.getKeyPath()));
35+
payConfig.setApiV3Key(StringUtils.trimToNull(this.properties.getApiV3Key()));//V3商户模式需要
36+
payConfig.setCertSerialNo(StringUtils.trimToNull(this.properties.getCertSerialNo()));//V3商户模式需要
37+
payConfig.setPrivateCertPath(StringUtils.trimToNull(this.properties.getPrivateCertPath()));//V3商户模式需要
38+
payConfig.setPrivateKeyPath(StringUtils.trimToNull(this.properties.getPrivateKeyPath()));//V3商户模式需要
39+
payConfig.setPublicKeyPath(StringUtils.trimToNull(this.properties.getPublicKeyPath()));//V3商户模式需要
40+
payConfig.setPublicKeyId(StringUtils.trimToNull(this.properties.getPublicKeyId()));//V3商户模式需要
3541

3642
// 可以指定是否使用沙箱环境
3743
payConfig.setUseSandboxEnv(false);

src/main/java/com/github/binarywang/demo/wx/pay/config/WxPayProperties.java

+38
Original file line numberDiff line numberDiff line change
@@ -13,11 +13,13 @@
1313
public class WxPayProperties {
1414
/**
1515
* 设置微信公众号或者小程序等的appid
16+
* (V3商户模式需要)
1617
*/
1718
private String appId;
1819

1920
/**
2021
* 微信支付商户号
22+
* (V3商户模式需要)
2123
*/
2224
private String mchId;
2325

@@ -41,4 +43,40 @@ public class WxPayProperties {
4143
*/
4244
private String keyPath;
4345

46+
/**
47+
* apiV3 秘钥值
48+
* (V3商户模式需要)
49+
*/
50+
private String apiV3Key;
51+
52+
/**
53+
* apiV3 证书序列号值
54+
* (V3商户模式需要)
55+
*/
56+
private String certSerialNo;
57+
58+
/**
59+
* apiclient_cert.pem证书文件的绝对路径或者以classpath:开头的类路径.
60+
* (V3商户模式需要,这里用的是文件路径,可以用其它base64编码或字节数组参数替代)
61+
*/
62+
private String privateCertPath;
63+
64+
/**
65+
* apiclient_key.pem证书文件的绝对路径或者以classpath:开头的类路径.
66+
* (V3商户模式需要,这里用的是文件路径,可以用其它base64编码或字节数组参数替代)
67+
*/
68+
private String privateKeyPath;
69+
70+
/**
71+
* 微信支付公钥,pub_key.pem证书文件的绝对路径或者以classpath:开头的类路径.
72+
* (V3商户模式需要,这里用的是文件路径,可以用其它base64编码或字节数组参数替代,2024.08后的新商户验签需要用此公钥)
73+
*/
74+
private String publicKeyPath;
75+
76+
/**
77+
* 微信支付公钥ID
78+
* (V3商户模式需要,2024.08后的新商户验签需要用此公钥ID)
79+
*/
80+
private String publicKeyId;
81+
4482
}

src/main/java/com/github/binarywang/demo/wx/pay/controller/WxPayController.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -231,7 +231,7 @@ public WxPayRedpackQueryResult queryRedpack(@PathVariable String mchBillNo) thro
231231
* @param sideLength 要生成的二维码的边长,如果为空,则取默认值400
232232
* @return 生成的二维码的字节数组
233233
*/
234-
public byte[] createScanPayQrcodeMode1(String productId, File logoFile, Integer sideLength) {
234+
public byte[] createScanPayQrcodeMode1(String productId, File logoFile, Integer sideLength) throws Exception {
235235
return this.wxService.createScanPayQrcodeMode1(productId, logoFile, sideLength);
236236
}
237237

@@ -264,7 +264,7 @@ public String createScanPayQrcodeMode1(String productId) {
264264
* @param sideLength 要生成的二维码的边长,如果为空,则取默认值400
265265
* @return 生成的二维码的字节数组
266266
*/
267-
public byte[] createScanPayQrcodeMode2(String codeUrl, File logoFile, Integer sideLength) {
267+
public byte[] createScanPayQrcodeMode2(String codeUrl, File logoFile, Integer sideLength) throws Exception {
268268
return this.wxService.createScanPayQrcodeMode2(codeUrl, logoFile, sideLength);
269269
}
270270

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,224 @@
1+
package com.github.binarywang.demo.wx.pay.controller;
2+
3+
import com.github.binarywang.wxpay.bean.notify.SignatureHeader;
4+
import com.github.binarywang.wxpay.bean.notify.WxPayNotifyV3Response;
5+
import com.github.binarywang.wxpay.bean.notify.WxPayNotifyV3Result;
6+
import com.github.binarywang.wxpay.bean.notify.WxPayNotifyV3Result.DecryptNotifyResult;
7+
import com.github.binarywang.wxpay.bean.notify.WxPayRefundNotifyV3Result;
8+
import com.github.binarywang.wxpay.bean.request.*;
9+
import com.github.binarywang.wxpay.bean.result.WxPayOrderQueryV3Result;
10+
import com.github.binarywang.wxpay.bean.result.WxPayRefundQueryV3Result;
11+
import com.github.binarywang.wxpay.bean.result.WxPayRefundV3Result;
12+
import com.github.binarywang.wxpay.bean.result.WxPayUnifiedOrderV3Result.AppResult;
13+
import com.github.binarywang.wxpay.bean.result.WxPayUnifiedOrderV3Result.JsapiResult;
14+
import com.github.binarywang.wxpay.bean.result.enums.TradeTypeEnum;
15+
import com.github.binarywang.wxpay.constant.WxPayConstants;
16+
import com.github.binarywang.wxpay.exception.WxPayException;
17+
import com.github.binarywang.wxpay.service.WxPayService;
18+
import io.swagger.annotations.Api;
19+
import io.swagger.annotations.ApiOperation;
20+
import lombok.AllArgsConstructor;
21+
import org.springframework.http.ResponseEntity;
22+
import org.springframework.web.bind.annotation.*;
23+
24+
import javax.servlet.http.HttpServletRequest;
25+
26+
27+
/**
28+
* @author Binary Wang
29+
*/
30+
@Api("微信支付V3部分接口示例")
31+
@RestController
32+
@RequestMapping("/pay/v3")
33+
@AllArgsConstructor
34+
public class WxPayV3Controller {
35+
private WxPayService wxService;
36+
37+
/**
38+
* <pre>
39+
* 查询订单
40+
* 详见https://pay.weixin.qq.com/doc/v3/merchant/4012791858)
41+
*
42+
* </pre>
43+
*
44+
* @param transactionId 微信订单号
45+
* @param outTradeNo 商户系统内部的订单号,当没提供transactionId时需要传这个。
46+
*/
47+
@ApiOperation(value = "查询订单")
48+
@GetMapping("/queryOrder")
49+
public WxPayOrderQueryV3Result queryOrder(@RequestParam(required = false) String transactionId,
50+
@RequestParam(required = false) String outTradeNo)
51+
throws WxPayException {
52+
return this.wxService.queryOrderV3(transactionId, outTradeNo);
53+
}
54+
55+
@ApiOperation(value = "查询订单")
56+
@PostMapping("/queryOrder")
57+
public WxPayOrderQueryV3Result queryOrder(@RequestBody WxPayOrderQueryV3Request wxPayOrderQueryV3Request) throws WxPayException {
58+
return this.wxService.queryOrderV3(wxPayOrderQueryV3Request);
59+
}
60+
61+
/**
62+
* <pre>
63+
* 关闭订单
64+
* 详见https://pay.weixin.qq.com/doc/v3/merchant/4012791860
65+
*
66+
* </pre>
67+
*
68+
* @param outTradeNo 商户系统内部的订单号
69+
*/
70+
@ApiOperation(value = "关闭订单")
71+
@GetMapping("/closeOrder/{outTradeNo}")
72+
public void closeOrder(@PathVariable String outTradeNo) throws WxPayException {
73+
this.wxService.closeOrderV3(outTradeNo);
74+
}
75+
76+
@ApiOperation(value = "关闭订单")
77+
@PostMapping("/closeOrder")
78+
public void closeOrder(@RequestBody WxPayOrderCloseV3Request wxPayOrderCloseV3Request) throws WxPayException {
79+
this.wxService.closeOrderV3(wxPayOrderCloseV3Request);
80+
}
81+
82+
/**
83+
* 调用统一下单接口(JSAPI)
84+
* 详见:https://pay.weixin.qq.com/doc/v3/merchant/4012791856
85+
*/
86+
@ApiOperation(value = "统一下单,并组装所需支付参数")
87+
@PostMapping("/createOrder")
88+
public JsapiResult createOrder(@RequestBody WxPayUnifiedOrderV3Request request) throws WxPayException {
89+
return this.wxService.createOrderV3(TradeTypeEnum.JSAPI, request);
90+
}
91+
92+
/**
93+
* 调用统一下单接口(APP)
94+
*/
95+
@ApiOperation(value = "统一下单,并组装所需支付参数")
96+
@PostMapping("/createOrderApp")
97+
public AppResult createOrderApp(@RequestBody WxPayUnifiedOrderV3Request request) throws WxPayException {
98+
return this.wxService.createOrderV3(TradeTypeEnum.APP, request);
99+
}
100+
101+
/**
102+
* <pre>
103+
* 微信支付-申请退款
104+
* 详见 https://pay.weixin.qq.com/doc/v3/merchant/4012791862
105+
* </pre>
106+
*
107+
* @param request 请求对象
108+
* @return 退款操作结果
109+
*/
110+
@ApiOperation(value = "退款")
111+
@PostMapping("/refund")
112+
public WxPayRefundV3Result refund(@RequestBody WxPayRefundV3Request request) throws WxPayException {
113+
return this.wxService.refundV3(request);
114+
}
115+
116+
/**
117+
* <pre>
118+
* 微信支付-查询退款
119+
* 详见 https://pay.weixin.qq.com/doc/v3/merchant/4012791863
120+
* </pre>
121+
*
122+
* @param outRefundNo 商户退款单号
123+
* @return 退款信息
124+
*/
125+
@ApiOperation(value = "退款查询")
126+
@GetMapping("/refundQuery")
127+
public WxPayRefundQueryV3Result refundQuery(@RequestParam("outRefundNo") String outRefundNo)
128+
throws WxPayException {
129+
return this.wxService.refundQueryV3(outRefundNo);
130+
}
131+
132+
@ApiOperation(value = "退款查询")
133+
@PostMapping("/refundQuery")
134+
public WxPayRefundQueryV3Result refundQuery(@RequestBody WxPayRefundQueryV3Request wxPayRefundQueryV3Request) throws WxPayException {
135+
return this.wxService.refundQueryV3(wxPayRefundQueryV3Request);
136+
}
137+
138+
/**
139+
* <pre>
140+
* 支付成功回调
141+
* 详见 https://pay.weixin.qq.com/doc/v3/merchant/4012791861
142+
* </pre>
143+
*
144+
* @param notifyData
145+
* @param request
146+
* @return
147+
* @throws WxPayException
148+
*/
149+
@ApiOperation(value = "支付回调通知处理")
150+
@PostMapping("/notify/order")
151+
public ResponseEntity<String> parseOrderNotifyResult(@RequestBody String notifyData, HttpServletRequest request) {
152+
SignatureHeader header = getRequestHeader(request);
153+
try {
154+
WxPayNotifyV3Result res = this.wxService.parseOrderNotifyV3Result(notifyData, header);
155+
DecryptNotifyResult decryptRes = res.getResult();
156+
// TODO 根据自己业务场景需要构造返回对象
157+
if (WxPayConstants.WxpayTradeStatus.SUCCESS.equals(decryptRes.getTradeState())) {
158+
//成功返回200/204,body无需有内容
159+
return ResponseEntity.status(200).body("");
160+
} else {
161+
//失败返回4xx或5xx,且需要构造body信息
162+
return ResponseEntity.status(500).body(WxPayNotifyV3Response.fail("错误原因"));
163+
}
164+
} catch (WxPayException e) {
165+
//失败返回4xx或5xx,且需要构造body信息
166+
return ResponseEntity.status(500).body(WxPayNotifyV3Response.fail("错误原因"));
167+
}
168+
}
169+
170+
/**
171+
* <pre>
172+
* 支付成功回调
173+
* 详见 https://pay.weixin.qq.com/doc/v3/merchant/4012791865
174+
* </pre>
175+
*
176+
* @param notifyData
177+
* @param request
178+
* @return
179+
* @throws WxPayException
180+
*/
181+
@ApiOperation(value = "退款回调通知处理")
182+
@PostMapping("/notify/refund")
183+
public ResponseEntity<String> parseRefundNotifyResult(@RequestBody String notifyData, HttpServletRequest request) {
184+
SignatureHeader header = getRequestHeader(request);
185+
try {
186+
WxPayRefundNotifyV3Result res = this.wxService.parseRefundNotifyV3Result(notifyData, header);
187+
WxPayRefundNotifyV3Result.DecryptNotifyResult decryptRes = res.getResult();
188+
// TODO 根据自己业务场景需要构造返回对象
189+
if (WxPayConstants.RefundStatus.SUCCESS.equals(decryptRes.getRefundStatus())) {
190+
//成功返回200/204,body无需有内容
191+
return ResponseEntity.status(200).body("");
192+
} else {
193+
//失败返回4xx或5xx,且需要构造body信息
194+
return ResponseEntity.status(500).body(WxPayNotifyV3Response.fail("错误原因"));
195+
}
196+
} catch (WxPayException e) {
197+
//失败返回4xx或5xx,且需要构造body信息
198+
return ResponseEntity.status(500).body(WxPayNotifyV3Response.fail("错误原因"));
199+
}
200+
}
201+
202+
/**
203+
* 组装请求头重的前面信息
204+
*
205+
* @param request
206+
* @return
207+
*/
208+
private SignatureHeader getRequestHeader(HttpServletRequest request) {
209+
// 获取通知签名
210+
String signature = request.getHeader("Wechatpay-Signature");
211+
String nonce = request.getHeader("Wechatpay-Nonce");
212+
String serial = request.getHeader("Wechatpay-Serial");
213+
String timestamp = request.getHeader("Wechatpay-Timestamp");
214+
215+
SignatureHeader signatureHeader = new SignatureHeader();
216+
signatureHeader.setSignature(signature);
217+
signatureHeader.setNonce(nonce);
218+
signatureHeader.setSerial(serial);
219+
signatureHeader.setTimeStamp(timestamp);
220+
return signatureHeader;
221+
}
222+
223+
224+
}

src/main/resources/application.yml.template

+9-3
Original file line numberDiff line numberDiff line change
@@ -6,9 +6,15 @@ logging:
66

77
wx:
88
pay:
9-
appId: #微信公众号或者小程序等的appid
10-
mchId: #微信支付商户号
9+
appId: #微信公众号或者小程序等的appid (V3商户模式需要)
10+
mchId: #微信支付商户号 (V3商户模式需要)
1111
mchKey: #微信支付商户密钥
1212
subAppId: #服务商模式下的子商户公众账号ID
1313
subMchId: #服务商模式下的子商户号
14-
keyPath: # p12证书的位置,可以指定绝对路径,也可以指定类路径(以classpath:开头)
14+
keyPath: # p12证书的位置,可以指定绝对路径,也可以指定类路径(以classpath:开头)
15+
apiV3Key: # apiV3 秘钥值 (V3商户模式需要)
16+
certSerialNo: # apiV3 证书序列号值 (V3商户模式需要)
17+
privateCertPath: # apiclient_cert.pem证书文件的绝对路径或者以classpath:开头的类路径. (V3商户模式需要)
18+
privateKeyPath: # apiclient_key.pem证书文件的绝对路径或者以classpath:开头的类路径. (V3商户模式需要)
19+
publicKeyPath: # 微信支付公钥,pub_key.pem证书文件的绝对路径或者以classpath:开头的类路径. (V3商户模式需要)
20+
publicKeyId: # 微信支付公钥ID (V3商户模式需要)

0 commit comments

Comments
 (0)