WebSocket长连接接入支付宝消息服务,实现消息通知

简介: 支付宝开放平台消息服务提供两种通讯协议来接收消息,一种是基于 HTTPS/HTTP,一种是基于 WebSocket长连接。相比之下,WebSocket长连接有更多的优势,所以一般选择使用WebSocket长连接来接收支付宝服务端发来的消息。

大家好,我是小悟

在对接支付宝开放平台的一些常用功能时,常常需要收到支付宝的回调通知结果,才能处理业务逻辑。此文介绍通过WebSocket长连接接入支付宝消息服务,实现消息通知。

包括五部分内容:问题、优势、配置、代码接入、总结。

image.png

问题
比如接入互联网平台直付通二级商户进件时,需要知道这个进件审核的结果,是审核通过还是审核拒绝,就要用到直付通商户进件审核通过消息接口和直付通二级商户进件拒绝消息接口。

再比如接入支付宝小程序模板开发时,需要知道第三应用授权取消结果、小程序审核结果、服务商代创建小程序结果等等,就要用到第三方应用授权取消消息接口、小程序审核通过通知接口、小程序审核驳回通知接口、商户确认服务商代创建小程序结果通知接口等等。

虽然可以通过对应的查询接口主动发起查询结果,但多个业务结果还需开发多个查询接口,体验终归不如由支付宝服务端侧直接通知开发者来的好。所以千万别干“脱裤子放pi,多此一举”的事情来,哈哈哈。

image.png

为了解决通知的问题,支付宝开放平台消息服务提供两种通讯协议来接收消息,一种是基于 HTTPS/HTTP,一种是基于 WebSocket 长连接。

优势
选择WebSocket 长连接有诸多优势:

官方提供封装好的SDK,开发者无需考虑通信、验签、组装报文协议,只要专心根据收到的消息处理自身的业务逻辑即可。

无需申请https证书,减少繁琐的证书申请工作,消息也能触达。

开发者无需额外开发一个服务来接收开放平台的消息。

相比之下,WebSocket 长连接有更多的优势,所以一般选择使用WebSocket 长连接来接收支付宝服务端发来的消息。

配置
创建好应用后,在产品绑定-绑定产品,添加对应的产品。

image.png

然后在开发设置-消息服务-FROM平台订阅所需监听的消息接口,接入方式选择WebSocket。

image.png

image.png

接入
以上操作是接入的前提条件,务必检查清楚再进行代码的开发。

可使用普通公钥方式和公钥证书方式接入,方式不同,SDK的使用也不同,这个取决于设置接口加签是何种方式,这边选择的是公钥证书方式。

image.png

在代码中引入依赖,这边有个注意点就是,如果选择的是公钥证书模式的话,SDK版本需要使用4.11.54.ALL 及以上版本。

<!-- https://mvnrepository.com/artifact/com.alipay.sdk/alipay-sdk-java -->
<dependency>
    <groupId>com.alipay.sdk</groupId>
    <artifactId>alipay-sdk-java</artifactId>
    <version>4.11.54.ALL</version>
</dependency>

这个是重点,开发一个支付宝消息配置类,支付宝服务端有消息通知时会自动触发。

@Component
@Configuration
@EnableConfigurationProperties({AliPayProperties.class})
public class AliPayMsgConfig {
    private static Logger logger = LoggerFactory.getLogger(AliPayMsgConfig.class);
    private AliPayProperties aliPay;
    public AliPayMsgConfig(AliPayProperties aliPay) {
        this.aliPay = aliPay;
    }
    @Bean
    public AlipayMsgClient alipayMsgClient() throws Exception {
        AlipayMsgClient alipayMsgClient = AlipayMsgClient.getInstance(aliPay.getAppId());
        alipayMsgClient.setConnector("openchannel.alipay.com");
        alipayMsgClient.setSecurityCertConfig(aliPay.getSignType(), FileUtil.readUtf8String(aliPay.getAppCertPrivateKeyPath()), aliPay.getAppCertPublicKeyPath(), aliPay.getAliPayCertPublicKeyPath(), aliPay.getAliPayRootCertPath());
        alipayMsgClient.setCharset(aliPay.getChartSet());
        alipayMsgClient.setMessageHandler(new MsgHandler() {
            /**
             * 客户端接收到消息后回调此方法
             *  @param  msgApi 接收到的消息的消息api名
             *  @param  msgId 接收到的消息的消息id
             *  @param  bizContent 接收到的消息的内容,json格式
             */
            @Override
            public void onMessage (String msgApi, String msgId, String bizContent) {
                logger.info("receive message. msgApi:{},msgId:{},bizContent:{}", msgApi, msgId, bizContent);
                if (StringUtils.equals("alipay.open.app.api.field.changed", msgApi)) {
                    logger.info("用户信息申请记录审核通知,接收到的消息内容:{}", bizContent);
                } else if (StringUtils.equals("alipay.open.auth.appauth.cancelled", msgApi)) {
                    logger.info("第三方应用授权取消消息,接收到的消息内容:{}", bizContent);
                } else if (StringUtils.equals("alipay.open.auth.userauth.cancelled", msgApi)) {
                    logger.info("用户授权取消消息,接收到的消息内容:{}", bizContent);
                } else if (StringUtils.equals("alipay.open.mini.version.audit.passed", msgApi)) {
                    logger.info("小程序审核通过通知,接收到的消息内容:{}", bizContent);
                } else if (StringUtils.equals("alipay.open.mini.version.audit.rejected", msgApi)) {
                    logger.info("小程序审核驳回通知,接收到的消息内容:{}", bizContent);
                } else if (StringUtils.equals("alipay.trade.refund.depositback.completed", msgApi)) {
                    logger.info("收单退款冲退完成通知,接收到的消息内容:{}", bizContent);
                } else if (StringUtils.equals("alipay.open.mini.merchant.confirmed", msgApi)) {
                    logger.info("商户确认服务商代创建小程序结果通知,接收到的消息内容:{}", bizContent);
                }
            }
        });
        alipayMsgClient.connect();
        return alipayMsgClient;
    }
}

别嫌if else长,这是demo,怎么方便怎么来。上面代码的AliPayProperties类是配置了应用的一些参数信息,包括应用id、加签类型、应用私钥证书路径、应用公钥证书路径、支付宝公钥证书路径、支付宝根证书路径、编码格式。

@Data
@ConfigurationProperties(prefix = "alipay")
public class AliPayProperties {
    /**
     * 加签类型
     */
    private String signType;
    /**
     * 应用id
     */
    private String appId;
    /**
     * 请求使用的编码格式,如utf-8,gbk,gb2312等
     */
    private String chartSet;
    /**
     * 应用私钥证书路径
     */
    private String appCertPrivateKeyPath;
    /**
     * 应用公钥证书路径
     */
    private String appCertPublicKeyPath;
    /**
     * 支付宝公钥证书路径
     */
    private String aliPayCertPublicKeyPath;
    /**
     * 支付宝根证书路径
     */
    private String aliPayRootCertPath;
}

总结
当支付宝服务端有给客户端发送消息时,会回调MsgHandler里面onMessage的实现,因此可以在onMessage中处理接收到的消息,根据msgApi消息名判断是哪种接口来处理对应的业务逻辑。

但需要注意的是,因各种异常原因,支付宝服务端可能会重复通知,因此为了业务不重复被处理,需要做幂等性的控制,可根据每条消息的唯一msgId或者自身系统的业务id比如uuid来事先查询。

image.png

您的一键三连,是我更新的最大动力,谢谢

山水有相逢,来日皆可期,谢谢阅读,我们再会

我手中的金箍棒,上能通天,下能探海

上一篇:一不小心,登上支付宝开发者社区热文榜单Top3

相关实践学习
消息队列+Serverless+Tablestore:实现高弹性的电商订单系统
基于消息队列以及函数计算,快速部署一个高弹性的商品订单系统,能够应对抢购场景下的高并发情况。
消息队列 MNS 入门课程
1、消息队列MNS简介 本节课介绍消息队列的MNS的基础概念 2、消息队列MNS特性 本节课介绍消息队列的MNS的主要特性 3、MNS的最佳实践及场景应用 本节课介绍消息队列的MNS的最佳实践及场景应用案例 4、手把手系列:消息队列MNS实操讲 本节课介绍消息队列的MNS的实际操作演示 5、动手实验:基于MNS,0基础轻松构建 Web Client 本节课带您一起基于MNS,0基础轻松构建 Web Client
相关文章
|
Web App开发 前端开发 Java
SpringBoot默认200个线程对于Websocket长连接够用吗?(一)
上篇推文从源码剖析SpringBoot中Tomcat的默认最大连接数中我们知道,SpringBoot的内嵌Tomcat默认的最大连接数为200。那么,这个默认值对于项目中引入了WebSocket使用长连接后,是否足够用呢?今天强哥就带大家一起从源码的角度来分析一下。
SpringBoot默认200个线程对于Websocket长连接够用吗?(一)
|
6月前
|
前端开发 JavaScript Java
Springboot+Netty+WebSocket搭建简单的消息通知
这样,你就建立了一个简单的消息通知系统,使用Spring Boot、Netty和WebSocket实现实时消息传递。你可以根据具体需求扩展和改进该系统。
154 1
|
关系型数据库 MySQL PHP
php使用webSocket实现Echarts长连接自动刷新的解决方案(3):获取读取数据库数据队列进行实时刷新
php使用webSocket实现Echarts长连接自动刷新的解决方案(3):获取读取数据库数据队列进行实时刷新
162 0
|
JSON 关系型数据库 MySQL
php使用webSocket实现Echarts长连接自动刷新的解决方案(2):后端服务端代码返回json数据
php使用webSocket实现Echarts长连接自动刷新的解决方案(2):后端服务端代码返回json数据
165 0
|
JSON 前端开发 JavaScript
php使用webSocket实现Echarts长连接自动刷新的解决方案(1):前端获取后端JSON数据
php使用webSocket实现Echarts长连接自动刷新的解决方案(1):前端获取后端JSON数据
158 0
|
网络协议 前端开发
WebSocket实现长连接实时消息推送
WebSocket实现长连接实时消息推送
WebSocket实现长连接实时消息推送
|
网络协议 算法
websocket 的实现原理 使用node搭建长连接
HTTP/1.1为了尽可能的提高HTTP性能,1.1规定所有连接必须是持久的,已经不需要在头部加上Connection:Keep-alive了。
|
网络协议 前端开发 JavaScript
Netty进阶 -- WebSocket长连接开发
Netty进阶 -- WebSocket长连接开发
748 0
Netty进阶 -- WebSocket长连接开发
|
移动开发 网络协议 物联网
用OkHttp实现WebSocket长连接
今天给大家带来一篇老文章,介绍WebSocket,大家可以了解了解。
632 0
用OkHttp实现WebSocket长连接