.net core 3.0 Signalr - 08 业务实现-客户端demo
由于signalr作为一个单独的推送系统,跟业务系统是分离开的,所以此处模拟一个业务系统,新建一个.net core app项目
模拟实现一个登录功能
我们的登录很简单,当进入系统,如果检测到用户未登录则跳转到登录页面,用户只需要输入用户名点击登录即算登录成功
-
配置configservices方法 查看代码
services.addauthentication(cookieauthenticationdefaults.authenticationscheme) .addcookie(cookieauthenticationdefaults.authenticationscheme, cookieoption => { cookieoption.loginpath = "/account/login"; cookieoption.accessdeniedpath = "/account/login"; });
-
配置config方法,配置认证、授权的请求管道 查看代码
app.userouting(); app.useauthentication(); app.useauthorization();
接收登录的post请求,写cookie,跳转 查看代码
前端页面实现
首先在layout页面引入需要的js文件(vue、signalr、msgpack5、signalr-protocol-msgpack) 查看代码
封装signalr连接相关js
signalr客户端js的操作就是,创建连接、监听推送,封装后端js如下 查看代码
/** * 初始化连接 * @param {object} option 参数 * @param {string} option.url 连接的url地址 * @param {string} option.logginglevel 日志级别,默认为 error * @param {number} option.delay 延迟连接 默认为3000毫秒 * @param {function} option.onstarted 启动时触发 * @param {function} option.online 启动时触发 * @param {function} option.offline 启动时触发 * @returns {object} 连接的实例 */ function initsignalr(option) { var config = object.assign(true, { logginglevel: signalr.loglevel.error, delay: 3000, url: '' }, option); var connection = new signalr.hubconnectionbuilder() .configurelogging(config.logginglevel) .withurl(config.url, { accesstokenfactory: option.accesstokenfactory }) .withhubprotocol(new signalr.protocols.msgpack.messagepackhubprotocol()) .withautomaticreconnect([0, 2000, 5000, 10000, 20000]) .build(); connection.onreconnecting(function (info) { console.info('----------------------------------signalr-- onreconnecting', info); }); connection.onclose(function (err) { console.info('--------------------------------signalr-- onclose', err); }); connection.on('onnotify', config.onnotify); connection.on('online', config.online); connection.on('offline', config.offline); settimeout(function () { connection.start().then(function (data) { option.onstarted && option.onstarted(data); }).catch(function (error) { console.error(error.tostring()); }); }, option.delay); return connection; }
调用封装的js初始化连接 查看代码
- 然后在home/index.cshtml中引入上面的js
-
在页面加载完后,调用初始化(案例中使用了vue)
在进入页面后会弹窗让用户输入加入的组,可以不输入也可以多个function initconnect() { $("#collectionuserinfo").modal({ keyboard: false, show: true, backdrop: 'static' }) $('#collectionuserinfo').on('hidden.bs.modal', function () { var groups = $("#groups").val()||''; connect=initsignalr({ delay: 0, url:`${notifyurl}notify-hub?userid=${vm.userinfo.username}&group=${groups}`, logginglevel: signalr.loglevel.error, onnotify: dealnotify, online: function (data) { if (data.isfirst) { getonlineusers(); } getonlinegroups(); vm.logs.push(`新连接上线:${json.stringify(data)}`); }, offline: function (data) { if (data.islast) { getonlineusers(); } getonlinegroups(); vm.logs.push(`连接下线:${json.stringify(data)}`); }, onstarted: function () { getonlineusers(); getonlinegroups(); vm.$set(vm.userinfo, 'connectionid', connect.connectionid); vm.$set(vm.userinfo, 'groups', groups); vm.logs.push('连接成功'); } }); }) }
onnotify方法,如果仔细的话会看到里面的onnotify方法,所有的推送最终都会调用到该方法来进行分发。查看代码
offline,当有客户端下线的时候会触发,data里面包含有用户id、连接id、是否该用户的最后一个连接,可根据需要使用查看代码
online,当用户连接的时候会触发,data里面包含有用户id、连接id、是否该用户的第一个连接(用于用户上线后的逻辑处理),可根据需要使用 查看代码
onstarted,当成功连接后触发,可用于做一些连接后的业务逻辑处理,可根据需要使用 查看代码
获取当前用户信息、在线列表
在用户连接成功后,获取当前在线用户、用户组、当前用户信息,并设置到vue的data中 查看代码
模拟一个任务分配
在项目中心中,点击"模拟推送待办"按钮,将会向当前用户所在组中推送一条代码消息,可以登录不同账号、开多个tab页体验
点击事件代码位置 查看代码
assigntasktouser: function () { var that = this; $.ajax({ type: 'post', url: '/api/serverproxy/assigntasktouser', data: { groups:that.userinfo.groups } }) },
对应的推送解析代码
首先当有推送过来的时候,会首先进到onnotify方法,然后根据不同类型在分配到不同的js方法中
查看代码
效果图
模拟发送消息
登录互斥
登录互斥是指,当一个账号在a电脑登录,然后再在b电脑登录,最后的登录会排斥掉开始的登录,即,将a上的挤下线
首先用谷歌浏览器登录,输入用户名:xiexingen,然后连接
接着使用360急速浏览器登录,输入用户名:xiexingen 这个时候会发现谷歌浏览器中的登录已经退出,如图
必要条件: 不同浏览器、同一用户,比如:同一个浏览器,不同tab就不算(能共用cookie)
文件下载(指定连接推送)
文件下载的场景,用户在操作页面上选择了上千个文件,然后点击打包下载,这个时候可能需要很久时间才反应回来,那么这段时间如果让用户一直等待显然不妥,所以,当用户点击打包下载的时候,后端启用一个后台线程去打包、压缩,然后立即返回;用户可以继续操作,当服务器端打包好后推送给用户端,用户点击下载即可。
此处分两种情况
- 单连接推送
用户开了多个tab页,在其中一个上下载文件,如果后端推送的时候,直接给该用户推,显然不妥;正确的做法一个是只给操作的那个tab页推,这就需要,调用服务器端业务api的时候,需要把当前tab页对应的连接id发送到服务器端,服务器端处理完业务后,调用推送服务器,告诉推送服务器只推我给你的这个连接的客户端,这样就能指定连接推送。 - 单用户排除某个连接的其他连接推送
这种情况比较少见,告诉推送服务器,给这个用户,除了某个连接外的其他所有连接推送
模拟操作
点击第一个图中的"打包下载文件" 按钮,当前页面会受到文件下载的推送
在点击图二中的"推送当前用户其他页面更新操作"按钮,会发现出了当前tab页外,其他tab也都收到了推送消息,如下图
至此,signalr相关文章算是到此结束了,下一篇谈谈个人的一点心得以及里面存着的一些问题。
快速导航
标题 | 内容 |
---|---|
索引 | .net core 3.0 signalr - 实现一个业务推送系统 |
上一篇 | .net core 3.0 signalr - 07 业务实现-服务端 自定义管理组、用户、连接 |
下一篇 | .net core 3.0 signalr - 09 待改进&交流 |
源码地址 | 源码 |
官方文档 | 官方文档 |
推荐阅读
-
.net core Redis分布式缓存客户端实现逻辑分析及示例demo
-
.net core 3.0 Signalr - 06 业务实现-业务分析
-
.net core 3.0 Signalr - 07 业务实现-服务端 自定义管理组、用户、连接
-
.net core 3.0 Signalr - 08 业务实现-客户端demo
-
.net core 3.0 Signalr - 实现一个业务推送系统
-
.net core Redis分布式缓存客户端实现逻辑分析及示例demo
-
.net core 3.0 Signalr - 实现一个业务推送系统
-
.net core 3.0 Signalr - 06 业务实现-业务分析
-
.net core 3.0 Signalr - 07 业务实现-服务端 自定义管理组、用户、连接
-
.net core 3.0 Signalr - 08 业务实现-客户端demo