微信公众号:Dotnet9,网站:Dotnet9,问题或建议:请网站留言

如果对您有所帮助:欢迎赞赏

使用SignalR从服务端主动推送警报日志到各种终端(桌面、移动、网页)

阅读导航

  1. 本文背景
  2. 代码实现
  3. 本文参考

1.本文背景

工作上有个业务,.Net Core WebAPI作为服务端,需要将运行过程中产生的日志分类,并实时推送到各种终端进行报警,终端有桌面(WPF)、移动(Xamarin.Forms)、网站(Angular.JS)等,使用SignalR进行警报日志推送。

下面是桌面端的测试效果:

2.代码实现

整个系统由服务端、桌面端、网站、移动端组成,结构如下:

2.1 服务端与客户端都使用的日志实体类

简单的日志定义,服务端会主动将最新日志通过AlarmLogItem实例推送到各个终端:

  1. /// <summary>
  2. /// 报警日志
  3. /// </summary>
  4. public class AlarmLogItem
  5. {
  6. public string Id { get; set; }
  7. /// <summary>
  8. /// 日志类型
  9. /// </summary>
  10. public AlarmLogType Type { get; set; }
  11. /// <summary>
  12. /// 日志名称
  13. /// </summary>
  14. public string Text { get; set; }
  15. /// <summary>
  16. /// 日志详细信息
  17. /// </summary>
  18. public string Description { get; set; }
  19. /// <summary>
  20. /// 日志更新时间
  21. /// </summary>
  22. public string UpdateTime { get; set; }
  23. }
  24. public enum AlarmLogType
  25. {
  26. Info,
  27. Warn,
  28. Error
  29. }

2.2 服务端

使用 .Net Core 2.2 搭建的Web API项目

2.2.1 集线器类AlarmLogHub.cs

定义集线器Hub类AlarmLogHub,继承自Hub,用于SignalR通信,看下面的代码,没加任何方法,您没看错:

  1. public class AlarmLogHub : Hub
  2. {}

2.2.2 Startup.cs

需要在此类中注册SignalR管道及服务,在下面两个关键方法中用到,B/S后端的朋友非常熟悉了。

  1. ConfigureServices方法

添加SignalR管道(是这个说法吧?):

  1. services.AddSignalR(options => { options.EnableDetailedErrors = true; });
  1. Configure方法注册SignalR服务地址

端口用的8022,客户端访问地址是:http://localhost:8022/alarmlog

  1. app.UseSignalR(routes =>
  2. {
  3. routes.MapHub<AlarmLogHub>("/alarmlog");
  4. });

2.2.3 SignalRTimedHostedService.cs

这是个关键类,用于服务端主动推送日志使用,Baidu、Google好久才找到,站长技术栈以C/S为主,B/S做的不多,没人指点,心酸,参考网址:How do I push data from hub to client every second using SignalR

该类继承自IHostedService,作为服务自启动(乱说的),通过SignalRTimedHostedService 的构造函数依赖注入得到IHubContext<AlarmLogHub>的实例,用于服务端向各客户端推送日志使用(在StartAsync方法中开启定时器,模拟服务端主动推送警报日志,见 DoWork 方法):

  1. internal class SignalRTimedHostedService : IHostedService, IDisposable
  2. {
  3. private readonly IHubContext<AlarmLogHub> _hub;
  4. private Timer _timer;
  5. //模拟发送报警日志
  6. List<AlarmLogItem> lstLogs = new List<AlarmLogItem> {
  7. new AlarmLogItem{ Type=AlarmLogType.Error,Text="OK WebSocket断连",Description="尝试连接50次,未成功重连!"},
  8. new AlarmLogItem{ Type=AlarmLogType.Warn,Text="OK WebSocket断开重连",Description="尝试连接5次,成功重连!"},
  9. new AlarmLogItem{ Type=AlarmLogType.Warn,Text="OK Restfull断连",Description="尝试连接30次,成功重连!"},
  10. new AlarmLogItem{ Type=AlarmLogType.Error,Text="OK WebSocket断连",Description="第一次断开链接!"},
  11. new AlarmLogItem{ Type=AlarmLogType.Info,Text="OK WebSocket连接成功",Description="首次成功连接!"},
  12. new AlarmLogItem{ Type=AlarmLogType.Error,Text="OK WebSocket断连",Description="尝试连接第7次,未成功重连!"}
  13. };
  14. Random rd = new Random(DateTime.Now.Millisecond);
  15. public SignalRTimedHostedService(IHubContext<AlarmLogHub> hub)
  16. {
  17. _hub = hub;
  18. }
  19. public Task StartAsync(CancellationToken cancellationToken)
  20. {
  21. _timer = new Timer(DoWork, null, TimeSpan.Zero,
  22. TimeSpan.FromSeconds(1));
  23. return Task.CompletedTask;
  24. }
  25. private void DoWork(object state)
  26. {
  27. if (DateTime.Now.Second % rd.Next(1, 3) == 0)
  28. {
  29. AlarmLogItem log = lstLogs[rd.Next(lstLogs.Count)];
  30. log.UpdateTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff");
  31. _hub.Clients.All.SendAsync("ReceiveAlarmLog", log);
  32. }
  33. }
  34. public Task StopAsync(CancellationToken cancellationToken)
  35. {
  36. _timer?.Change(Timeout.Infinite, 0);
  37. return Task.CompletedTask;
  38. }
  39. public void Dispose()
  40. {
  41. _timer?.Dispose();
  42. }
  43. }

SignalRTimedHostedService 类作为Host服务(继承自 IHostedService),需要在Startup.cs的ConfigureServices方法中注册管道(是吧?各位有没有B/S比较好的书籍推荐,站长打算有空好好学学):

  1. services.AddHostedService<SignalRTimedHostedService>();

服务端关键代码已经全部奉上,下面主要说说桌面端和移动端代码,其实两者代码类似。

2.3 网站

参考 index.html

2.4 桌面端(WPF)

使用 .Net Core 3.0创建的WFP工程,需要引入Nuget包:Microsoft.AspNetCore.SignalR.Client

界面用一个ListView展示收到的日志:

  1. <Grid>
  2. <ListBox x:Name="messagesList" RenderTransformOrigin="-0.304,0.109" BorderThickness="1" BorderBrush="Gainsboro"/>
  3. </Grid>

后台写的简陋,直接在窗体构造函数中连接服务端SignalR地址:http://localhost:8022/alarmlog, 监听服务端警报日志推送:ReceiveAlarmLog。

  1. using AppClient.Models;
  2. using Microsoft.AspNetCore.SignalR.Client;
  3. using System;
  4. using System.Threading.Tasks;
  5. using System.Windows;
  6. namespace SignalRChatClientCore
  7. {
  8. /// <summary>
  9. /// Interaction logic for MainWindow.xaml
  10. /// </summary>
  11. public partial class MainWindow : Window
  12. {
  13. HubConnection connection;
  14. public MainWindow()
  15. {
  16. InitializeComponent();
  17. connection = new HubConnectionBuilder()
  18. .WithUrl("http://localhost:8022/alarmlog")
  19. .Build();
  20. connection.Closed += async (error) =>
  21. {
  22. await Task.Delay(new Random().Next(0, 5) * 1000);
  23. await connection.StartAsync();
  24. };
  25. connection.On<AlarmLogItem>("ReceiveAlarmLog", (message) =>
  26. {
  27. this.Dispatcher.Invoke(() =>
  28. {
  29. messagesList.Items.Add(message.Description);
  30. });
  31. });
  32. try
  33. {
  34. connection.StartAsync();
  35. messagesList.Items.Add("Connection started");
  36. }
  37. catch (Exception ex)
  38. {
  39. messagesList.Items.Add(ex.Message);
  40. }
  41. }
  42. }
  43. }

2.4 移动端

移动端其实和桌面端类似,因为桌面端使用的 .Net Core 3.0,移动端使用的 .NET Standard 2.0,都需要引入Nuget包:Microsoft.AspNetCore.SignalR.Client。

界面使用ListView展示日志,这就不贴代码了,使用的MVVM方式,直接贴ViewModel代码吧,大家只看个大概,不要纠结具体代码,参照桌面.cs代码,是不是一样的?

  1. using AppClient.Models;
  2. using AppClient.Views;
  3. using Microsoft.AspNetCore.SignalR.Client;
  4. using System;
  5. using System.Collections.ObjectModel;
  6. using System.Diagnostics;
  7. using System.Threading.Tasks;
  8. using Xamarin.Forms;
  9. using System.Linq;
  10. namespace AppClient.ViewModels
  11. {
  12. /// <summary>
  13. /// 报警日志VM
  14. /// </summary>
  15. public class AlarmItemsViewModel : BaseViewModel
  16. {
  17. private ViewState _state = ViewState.Disconnected;
  18. /// <summary>
  19. /// 报警日志列表
  20. /// </summary>
  21. public ObservableCollection<AlarmLogItem> AlarmItems { get; set; }
  22. public Command LoadItemsCommand { get; set; }
  23. //连接报警服务端
  24. private HubConnection _connection;
  25. public AlarmItemsViewModel()
  26. {
  27. Title = "报警日志";
  28. AlarmItems = new ObservableCollection<AlarmLogItem>();
  29. LoadItemsCommand = new Command(async () => await ExecuteLoadItemsCommand());
  30. //收到登录成功通知
  31. MessagingCenter.Subscribe<LoginViewModel, LoginUser>(this, "LoginSuccess", async (sender, userInfo) =>
  32. {
  33. //DisplayAlert("登录成功", userInfo.UserName, "确定");
  34. });
  35. MessagingCenter.Subscribe<NewItemPage, AlarmLogItem>(this, "添加项", async (obj, item) =>
  36. {
  37. var newItem = item as AlarmLogItem;
  38. AlarmItems.Add(newItem);
  39. await DataStore.AddItemAsync(newItem);
  40. });
  41. ConnectAlarmServer();
  42. }
  43. async Task ExecuteLoadItemsCommand()
  44. {
  45. if (IsBusy)
  46. return;
  47. IsBusy = true;
  48. try
  49. {
  50. AlarmItems.Clear();
  51. var items = await DataStore.GetItemsAsync(true);
  52. foreach (var item in items)
  53. {
  54. AlarmItems.Add(item);
  55. }
  56. }
  57. catch (Exception ex)
  58. {
  59. Debug.WriteLine(ex);
  60. }
  61. finally
  62. {
  63. IsBusy = false;
  64. }
  65. }
  66. private async Task ConnectAlarmServer()
  67. {
  68. if (_state == ViewState.Connected)
  69. {
  70. try
  71. {
  72. await _connection.StopAsync();
  73. }
  74. catch (Exception ex)
  75. {
  76. return;
  77. }
  78. _state = ViewState.Disconnected;
  79. }
  80. else
  81. {
  82. try
  83. {
  84. _connection = new HubConnectionBuilder()
  85. .WithUrl(App.Setting.AlarmHost)
  86. .Build();
  87. _connection.On<AlarmLogItem>("ReceiveAlarmLog", async (newItem) =>
  88. {
  89. AlarmItems.Add(newItem);
  90. await DataStore.AddItemAsync(newItem);
  91. });
  92. _connection.Closed += async (error) =>
  93. {
  94. await Task.Delay(new Random().Next(0, 5) * 1000);
  95. await _connection.StartAsync();
  96. };
  97. await _connection.StartAsync();
  98. }
  99. catch (Exception ex)
  100. {
  101. return;
  102. }
  103. _state = ViewState.Connected;
  104. }
  105. }
  106. private enum ViewState
  107. {
  108. Disconnected,
  109. Connecting,
  110. Connected,
  111. Disconnecting
  112. }
  113. }
  114. }

关键代码已经贴完了,希望对大家能有所帮助。

3.参考

  1. .NET 客户端 SignalR ASP.NET Core
  2. SignalR-samples
  3. How do I push data from hub to client every second using SignalR

除非注明,文章均由 Dotnet9 整理发布,欢迎转载。


转载请注明本文地址:https://dotnet9.com/6913.html


欢迎扫描下方二维码关注 Dotnet9 的微信公众号,本站会及时推送最新技术文章


使用SignalR从服务端主动推送警报日志到各种终端(桌面、移动、网页)的更多相关文章

  1. 1.使用SignalR实现页面即时刷新(服务端主动推送)

    模块功能说明: 实现技术:sqlserver,MVC,WebAPI,ADO.NET,SignalR(服务器主动推送) 特殊车辆管理--->移动客户端采集数据存入数据库---->只要数据库数 ...

  2. 使用SignalR实现页面即时刷新(服务端主动推送)

    模块功能说明: 实现技术:sqlserver,MVC,WebAPI,ADO.NET,SignalR(服务器主动推送) 特殊车辆管理--->移动客户端采集数据存入数据库---->只要数据库数 ...

  3. 使用SignalR实现服务端消息推送

    概述 这篇文章参考的是Server Broadcast with SignalR 2这篇教程,很不错的一篇教程,如果有兴趣的话可以查看原文,今天记录下来作为一个学习笔记,这样今后翻阅会更方便一点. 这 ...

  4. Asp.net SignalR 实现服务端消息推送到Web端

              之前的文章介绍过Asp.net SignalR,  ASP .NET SignalR是一个ASP .NET 下的类库,可以在ASP .NET 的Web项目中实现实时通信.  今天我 ...

  5. SignalR 实现web浏览器客户端与服务端的推送功能

    SignalR 是一个集成的客户端与服务器库,基于浏览器的客户端和基于 ASP.NET 的服务器组件可以借助它来进行双向多步对话. 换句话说,该对话可不受限制地进行单个无状态请求/响应数据交换:它将继 ...

  6. ASP.NET SignalR 系列(七)之服务端触发推送

    前面几章讲的都是从客户端触发信息推送的,但在实际项目中,很多信息可能是由系统服务端推送的,下面2图分别展示两种通道 客户端触发推送 服务端推送 下面我们就重点介绍下服务端如何调用集线器的对象进行推送 ...

  7. Spring mvc服务端消息推送(SSE技术)

    SSE技术是基于单工通信模式,只是单纯的客户端向服务端发送请求,服务端不会主动发送给客户端.服务端采取的策略是抓住这个请求不放,等数据更新的时候才返回给客户端,当客户端接收到消息后,再向服务端发送请求 ...

  8. SSE技术详解:使用 HTTP 做服务端数据推送应用的技术

    SSE ( Server-sent Events )是 WebSocket 的一种轻量代替方案,使用 HTTP 协议. 严格地说,HTTP 协议是没有办法做服务器推送的,但是当服务器向客户端声明接下来 ...

  9. SSE(Server-sent events)技术在web端消息推送和实时聊天中的使用

    最近在公司闲着没事研究了几天,终于搞定了SSE从理论到实际应用,中间还是有一些坑的. 1.SSE简介 SSE(Server-sent events)翻译过来为:服务器发送事件.是基于http协议,和W ...

随机推荐

  1. AI: 如何用钢笔工具画曲线

    AI 可以用来绘制矢量图片. 点击钢笔工具,点击画图会画出直线,点击拖拉画图会画出曲线. 锚点的摆放位置在侧面而非顶端. 控制柄越长,图形越尖锐. 画圆时控制柄长度控制在两点之间1/3 长度. 使用的 ...

  2. Codeforces_816

    A.不断增加时间,直到符合要求. #include<bits/stdc++.h> using namespace std; int a,b; char c; int f(int x) { ...

  3. Codeforces 977B Two-gram(stl之string掉进坑)

    Two-gram is an ordered pair (i.e. string of length two) of capital Latin letters. For example, " ...

  4. React+wangeditor+node富文本处理带图片上传

    最近有个需求出现在我的视野中,因为我的另外的博客需要上传文章,但是我不想每次都在我的数据库中慢慢的修改格式,所以我另做了一个后台去编辑文本后发送给服务器,那么这里就涉及到两点,一个是富文本,一个是需要 ...

  5. Go语言实现:【剑指offer】二叉树中和为某一值的路径

    该题目来源于牛客网<剑指offer>专题. 输入一颗二叉树的跟节点和一个整数,打印出二叉树中结点值的和为输入整数的所有路径.路径定义为从树的根结点开始往下一直到叶结点所经过的结点形成一条路 ...

  6. Python 元类 - Metaclasses

    Python 元类 - Metaclasses 默认情况下儿, classes 是有 type() 构造的. 类的结构体在一个新的 namespace 被执行, 类的名字 class name 绑定( ...

  7. centos 7安装reids

    一.reids下载  下载地址: https://redis.io/ 二.解压安装 ① 解压:tar -zxvf redis-5.0.5.tar.gz ② 安装环境:yum install gcc-c ...

  8. PHP 安装扩展步骤

    一般来说php安装扩展需要几下几个步骤   1.下载扩展包    比如  pdo_mysql.tar.gz  (如果不想下载,可以到php安装目录,(类似php-5.3.3/ext/)的ext文件中找 ...

  9. sed知识及常用用法梳理

    1.sed命令简介及其参数说明 sed流编辑器,擅长对文本进行增删改查,过滤指定的字符串和取指定行,也可以在行中字符串前后插入内容,功能非常强大. 注意:sed默认只支持基本的正则表达式,如果要想支持 ...

  10. [Linux-CentOS7]安装Telnet

    # yum install telnet Loaded plugins: fastestmirror Loading mirror speeds from cached hostfile Resolv ...