微信支付商户系统Native支付

news/2025/1/13 7:58:27/文章来源:https://www.cnblogs.com/LiuFqiang/p/18519817

简易demo演示

demo演示

点击体验

### Native支付介绍 目前微信支付有以下几种场景 * JSAPi支付,适合微信公众号及微信小程序 * APP支付 * H5支付 * Native支付,适合PC网站页面支付 [微信支付商户平台](https://pay.weixin.qq.com/) [微信支付Native接口文档](https://pay.weixin.qq.com/doc/v3/merchant/4012075105) Native支付是指商户系统按照微信支付协议生成支付二维码,用户再用微信“扫一扫”实现支付 ### 前提准备 微信商家号、微信小程序或者微信公众号appId、商户证书 需要提前开启Native支付 ![](https://img2024.cnblogs.com/blog/1597479/202411/1597479-20241101102120164-1844181028.png) 需要将公众号或者小程序的appId在微信支付后台关联起来 ![](https://img2024.cnblogs.com/blog/1597479/202411/1597479-20241101102251617-1587831565.png) 在微信支付——账户中心——API安全,生成商户APIV3证书 ![](https://img2024.cnblogs.com/blog/1597479/202411/1597479-20241101102550425-1205822505.png)

业务流程图


主要步骤为,pc端生成订单调用Native下单接口生成微信的native 跳转链接,再生成二维码返回到页面,用户微信扫一扫完成支付,微信支付后台回调支付成功的链接。

支付接入

引入微信支付的SDK

<dependency><groupId>com.github.wechatpay-apiv3</groupId><artifactId>wechatpay-java</artifactId><version>0.2.14</version></dependency>

生成二维码的工具,这里我选择了zxing

<dependency><groupId>com.google.zxing</groupId><artifactId>core</artifactId><version>3.4.1</version></dependency><dependency><groupId>com.google.zxing</groupId><artifactId>javase</artifactId><version>3.4.1</version></dependency>

创建订单接口https://pay.weixin.qq.com/doc/v3/merchant/4012525095
需要商户号、商户APIV3密钥、商户证书序列号、商户API秘钥文件
正常秘钥文件放在服务器上读取,这里先直接放在项目文件resources下面

@Slf4j
@Service
public class WechatPayService extends AbstractPayService implements InitializingBean {/*** 商户号*/public static String merchantId = "xxx";/*** 商户证书序列号*/public static String merchantSerialNumber = "xxxx";/*** 商户APIV3密钥*/public static String apiV3Key = "xxxx";private Config config;/*** 商户API私钥*/private String privateKeyPem;@Overridepublic void afterPropertiesSet() throws Exception {ClassPathResource resource = new ClassPathResource("wechat/apiclient_key.pem");privateKeyPem = IOUtils.toString(resource.getInputStream());config = new RSAAutoCertificateConfig.Builder().merchantId(merchantId).privateKey(privateKeyPem).merchantSerialNumber(merchantSerialNumber).apiV3Key(apiV3Key).build();}@Overridepublic String createPayOrder(String clientIp, OrderRequest orderRequest) {final String orderNo = OrderUtils.generateOrderNo();// 一个商户号只能初始化一个配置,否则会因为重复的下载任务报错// 构建serviceNativePayService service = new NativePayService.Builder().config(config).build();// request.setXxx(val)设置所需参数,具体参数可见Request定义PrepayRequest request = new PrepayRequest();Amount amount = new Amount();amount.setTotal(1);request.setAmount(amount);request.setAppid("xxxx");request.setMchid(merchantId);request.setDescription("VIP体验卡");request.setNotifyUrl("http://myrkfh.natappfree.cc/mark_day/wechat/pay/notify");request.setOutTradeNo(orderNo);// 调用下单方法,得到应答PrepayResponse response = service.prepay(request);// 使用微信扫描 code_url 对应的二维码,即可体验Native支付log.info("调用微信生成订单返回结果:", response.toString());if (StrUtil.isBlank(response.getCodeUrl())) {throw new BaseException("创建订单失败");}String codeUrl = response.getCodeUrl();// 生成二维码QrConfig qrConfig = QrConfig.create();String qrBase64Text = QrCodeUtil.generateAsBase64(codeUrl, qrConfig, "svg");// 这里可以记录订单信息return qrBase64Text;}

orderNo在本系统中要唯一,不能有重复的订单号,否则下单失败

private static final String DATE_FORMAT = "yyyyMMddHHmmss";public static String generateOrderNo() {LocalDateTime now = LocalDateTime.now();DateTimeFormatter formatter = DateTimeFormatter.ofPattern(DATE_FORMAT);String timestamp = now.format(formatter);return timestamp + RandomUtil.randomNumbers(18);}

其中notifyUrl为支付成功回调地址,本地调试可以借助内网穿透工具,其他参数接口文档https://pay.weixin.qq.com/doc/v3/merchant/4012525095
微信返回的codeUrl为weixin://wxpay/bizpayurl?pr=Vd8RvS7z4格式的,放在手机上可以直接打开,我们要做的就是将codeUrl放入二维码的内容,也可以直接 使用Hutool,返回的二维码可以保存图片,将图片地址返回前端,也可以直接返回base64的图片地址,这里一般会记录订单数据为待支付状态数据,最后根据自己需要生成二维码的内容如下,有效期为两个小时

将地址返回过后后面就是等待微信支付的回调,

微信回调通知

微信支付通知接口文档 https://pay.weixin.qq.com/doc/v3/merchant/4012084431
如果处理成功了,需要返回HTTP状态码为200或204,否则微信会根据一定的策略进行重试

/*** 微信支付回调实体** @author liufuqiang* @Date 2024-10-21 17:33:00*/
@Data
public class WechatPayNotifyDTO {private String id;/*** 通知创建的时间*/@JsonAlias("create_time")private String createTime;/*** 通知的资源数据类型,支付成功通知为encrypt-resource。*/@JsonAlias("resource_type")private String resourceType;/*** 通知的类型,支付成功通知的类型为TRANSACTION.SUCCESS*/@JsonAlias("event_type")private String eventType;/*** 通知资源数据*/private Resource resource;@Datapublic static class Resource {/*** 加密算法AEAD_AES_256_GCM*/private String algorithm;/*** Base64编码后的开启/停用结果数据密文。*/private String ciphertext;/*** 附加数据。*/@JsonAlias("associated_data")private String associatedData;/*** 原始回调类型,为transaction*/@JsonAlias("original_type")private String originalType;/*** 加密使用的随机串。*/private String nonce;}/*** 回调摘要*/private String summary;

接受回调

@PostMapping("/notify")public void payNotify(HttpServletResponse response,HttpServletRequest request,@RequestHeader(Constant.WECHAT_PAY_SIGNATURE) String signature,@RequestHeader(Constant.WECHAT_PAY_NONCE) String nonce,@RequestHeader(Constant.WECHAT_PAY_TIMESTAMP) String timestamp,@RequestHeader(Constant.WECHAT_PAY_SERIAL) String serialNo,@RequestBody String requestBody) {HttpHeaders httpHeaders = new HttpHeaders();httpHeaders.addHeader(Constant.WECHAT_PAY_SIGNATURE, signature);httpHeaders.addHeader(Constant.WECHAT_PAY_NONCE, nonce);httpHeaders.addHeader(Constant.WECHAT_PAY_TIMESTAMP, timestamp);httpHeaders.addHeader(Constant.WECHAT_PAY_SERIAL, serialNo);// 验证签名boolean isSignValid = wechatPayService.validSignature(httpHeaders, requestBody);if (!isSignValid) {response.setStatus(HttpServletResponse.SC_BAD_REQUEST);}log.info("签名验证通过");WechatPayNotifyDTO notifyDTO = JsonUtils.genBeanByJson(requestBody, WechatPayNotifyDTO.class);wechatPayService.paymentHandle(notifyDTO);response.setStatus(HttpServletResponse.SC_OK);}

验签

防止伪造支付的回调,需要会返回的内容进行签名验证,
非文件/下载验证签名文档https://pay.weixin.qq.com/doc/v3/merchant/4012365350
如果嫌弃麻烦,也可以直接使用WechatPay2Validator
验证签名的时候尽量不要全部取出请求头的内容,根据HttpServletRequest全部取出的HeaderMap里面的key值全部是小写,签名的时候避免取到空值
签名的第三个参数为全部requestBody的内容,所以这里不能用实体对象接受数据,直接改用字符串接受数据

解密

根据API商户秘钥以及接口返回的associatedData、nonce对密文ciphertext进行解密,是AES对称加密解密

static final int TAG_LENGTH_BIT = 128;
public static String decryptToString(byte[] associatedData, byte[] nonce, String ciphertext) {try {Cipher cipher = Cipher.getInstance("AES/GCM/NoPadding");SecretKeySpec key = new SecretKeySpec(apiV3Key.getBytes(), "AES");GCMParameterSpec spec = new GCMParameterSpec(TAG_LENGTH_BIT, nonce);cipher.init(Cipher.DECRYPT_MODE, key, spec);cipher.updateAAD(associatedData);return new String(cipher.doFinal(Base64.getDecoder().decode(ciphertext)), "utf-8");} catch (NoSuchAlgorithmException | NoSuchPaddingException | UnsupportedEncodingException e) {} catch (InvalidKeyException | InvalidAlgorithmParameterException | IllegalBlockSizeException |BadPaddingException e) {}return "";}

解析完成是json字符串,这俩使用jackjson转为对象

@Data
public class OutOrderDTO {private String appid;@JsonAlias("out_trade_no")private String outTradeNo;@JsonAlias("trade_type")private String tradeType;@JsonAlias("trade_state")private String tradeState;@JsonAlias("trade_state_desc")private String tradeStateDesc;@JsonAlias("bank_ype")private String bankType;@JsonAlias("success_time")private String payTime;private Payer payer;private Amount amount;@Datapublic static final class Payer {private String openid;}@Datapublic static final class Amount {private Integer total;@JsonAlias("payer_total")private String payerTotal;}

创建订单时的OutTradeNo订单号也会返回,为了避免重复处理,在订单表加状态判断或者加分布式锁

WechatPayNotifyDTO.Resource resource = wechatPayNotifyDTO.getResource();String jsonStr = decryptToString(resource.getAssociatedData().getBytes(StandardCharsets.UTF_8),resource.getNonce().getBytes(StandardCharsets.UTF_8), resource.getCiphertext());log.info("微信支付回调信息:{}", jsonStr);OutOrderDTO outOrderDTO = JsonUtils.genBeanByJson(jsonStr, OutOrderDTO.class);// 订单处理log.info("支付完成");

同一个二维码也就是生成的native链接如果已经被支付过会提示订单已经支付,请勿重复发起支付

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/825221.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

centos7下redis安装

第1:下载地址 网页地址:https://redis.io/download 下载链接:http://download.redis.io/redis-stable.tar.gz 版本:Linux版,支持CentOS等其他Linux操作系统 备注:也可以直接通过Linux去下载Redis安装包 下载命令:wget http://download.redis.io/releases/redis-5.0.5.ta…

从0搭建 Spring Cloud Alibaba 基础工程框架搭建

整个项目结构:技术栈:spring cloud alibaba、MySQL8、Mybatis-Plus、Nacos、knife4j 接口文档、Lombok 一. 开发环境安装JDK17安装 MySQL安装二. 工程搭建 2.1 构建父子工程 2.1.1 创建父工程创建⼀个空的 Maven 项目, 删除所有代码, 只保留 pom.xml 目录结构: 图二完善父工程…

项目经理在哪些方面需要与组织的其他部门合作

项目经理需要与组织的其他部门在以下方面紧密合作:资源分配、沟通协调、风险管理、知识共享。在这些方面,特别需要强调沟通协调。有效的沟通能够确保项目团队与组织内其他部门之间的信息流动畅通无阻,帮助识别和解决跨部门的问题,同时促进项目与组织战略目标的一致性。 一、…

实现文件目录结构功能

实现文件目录结构功能@目录说明:该文章用于目录结构递进显示NodeConstructTree 说明:该文章用于目录结构递进显示Node package com.geespace.microservices.directory.assets.entity;import java.util.ArrayList; import java.util.List;import lombok.Data;/*** @Author: wj…

juicefs元数据存储方式

环境 文件系统使用juicefs,元数据存储使用postgresql,数据存储使用minio 问题? 通过juicefs写入一个文件,元数据在postgresql中是如何存储的?数据在minio中又是如何存储的? 使用docker部署完测试环境后,新建 file1、dir1/file1、dir1/file2三个文件 在postgresql中 jfs_…

有什么好的开源自动化测试框架可以推荐

根据“有什么好的开源自动化测试框架可以推荐”这个标题,1、Selenium,2、Appium,3、Robot Framework,4、JUnit。 对Selenium进行展开详细描述,在开源自动化测试工具领域,Selenium以其强大的功能和广泛的应用背景成为了多数开发者及测试人员的首选。Selenium不仅支持多种浏…

如何确定项目计划的关键绩效指标(KPI)

确定项目计划的关键绩效指标(KPI)应该以项目目标、过程优化与最终结果的可衡量性为依据。首先,明确项目目标是确定KPI的基础。根据目标,选择能够量化项目进程和成果的指标、保证KPI具有实时性和可操作性是确保效果的关键、并且需要确保KPI与组织的整体战略目标一致。例如,…

c#表达式树入门,看这个就够了

题记: 由于反射需要大量的性能开销,所以推荐用表达式树或者emit,但是emit 如果不熟悉指令编程的话,使用成本很大,所以优先推荐表达式树,但是网上给出来的文档 都非常的复杂,只是带你使用,刚好我团队的小伙伴也不太理解,所以我来整理一篇简单入门版本的.问: 反射有3种方式,一个是…

黑马PM-电商项目-电商后台

电商后台的核心作用及架构电商后台基础支撑

MEAS-Measurement: Sensors

Measurement: Sensors是一本开放获取期刊,对来自这个高度热门和多学科学科学科的所有相关领域的原创、高质量贡献开放。邀请提交关于科学、工程和技术的理论、研究、开发、制造和应用的各个方面的投稿,这些领域包括当今传感器和传感器系统。我们鼓励作者提交有关该领域的新材…

TI-Trends in Immunotherapy

Trends in Immunotherapy 是一本开放获取的同行评审期刊,涵盖与所有基于免疫系统的领域相关的各个学科。TI 的目标受众包括来自学术界、医疗行业、教育界等的科研人员、专业从业人员和医学学者。它提供了一个论坛来分享学术著作,以科学与医学相结合的方式推进免疫疗法。 发表…

为什么 C++ 编译速度比 Java 慢得多

### 为什么 C++ 编译速度比 Java 慢得多 在探讨为什么 C++ 编译速度比 Java 慢得多时,我们可以归纳出几个核心原因:C++的编译模型更为复杂、模板元编程、宏处理以及链接时间。其中,C++的编译模型更为复杂这一点尤为突出。C++需要处理的细节更多,如模板实例化、头文件的重复…