业务型网站使用过程中,消息通知是一个不可或缺的功能,采用站内通知、短信通知、邮件通知、微信通知等等各种方式都有,ABP框架对这部分工作已经封装的很好了,站在巨人的肩膀上,一览全貌,带来的就是心情舒畅。

  ABP官网地址:https://aspnetboilerplate.com/

一、明确概念及设计

  一次完整的消息发送/接收过程中,会存在几个必要的点,也正如同现实生活中的场景,有人发送,有人接收,发送的消息本身也有类型。

  

1、消息类型定义

  在ABP中已经提供了消息类型定义的相关类,但是需要我们去实现,在领域层新建一个Notifications文件夹并添加一个定义常用消息类型名称的静态类如AppNotificationNames(名称可随意),在其中开始定义消息类型名称。

/// <summary>
/// 设置应用程序中常用通知唯一的名称常量
/// </summary>
public static class AppNotificationNames
{
#region 新的任务
public const string NewTask = "App.NewTask";
#endregion #region 简单消息
public const string SimpleMessage = "App.SimpleMessage";
#endregion
}

  其次,开始消息类型定义,新建一个NotificationProvider的类,完成消息类型定义,管理消息类型时,可以针对消息设置权限,这样方便不同的角色拥有不同的消息通知。

/// <summary>
/// 消息类型定义
/// </summary>
public class AppNotificationProvider : NotificationProvider
{
/// <summary>
/// 设置消息类型
/// </summary>
/// <param name="context"></param>
public override void SetNotifications(INotificationDefinitionContext context)
{
#region 任务提醒
context.Manager.Add(
new NotificationDefinition(
AppNotificationNames.NewTask,
displayName: L("NewTask"),
permissionDependency: new SimplePermissionDependency(PermissionNames.Pages_TaskManage)
)
);
#endregion
} private static ILocalizableString L(string name)
{
return new LocalizableString(name, SurroundConsts.LocalizationSourceName);
}
}

  最后在领域层的Module.PreInitialize()中完成加入。

//通知定义
Configuration.Notifications.Providers.Add<AppNotificationProvider>();

2、消息订阅方式

  对于消息订阅环节,是存在两种情形的,第一种,发送方发出的消息,可以指定所有订阅了的人员接收,比如,只有拥有任务管理权限的人才会接收到新的任务下发的通知;第二种就是指定到具体人员接收,比如文件上传完毕的消息通知是通知到负责文件上传的人。

   

  消息订阅的管理,在ABP框架中已经封装好了,我们可以通过构造函数注入直接使用,通过获取系统内的提前定义的所有消息类型,我们可以在界面上完成订阅工作,为当前用户订阅在他角色权限内的消息通知及无权限限制的消息通知等。

private readonly INotificationDefinitionManager _notificationDefinitionManager;
private readonly INotificationSubscriptionManager _notificationSubscriptionManager; public NotificationAppService(
INotificationDefinitionManager notificationDefinitionManager,
INotificationSubscriptionManager notificationSubscriptionManager)
{
_notificationDefinitionManager = notificationDefinitionManager;
_notificationSubscriptionManager = notificationSubscriptionManager;
}

3、消息发布方式

  

  消息发布时,可以是嵌入在其它操作中,比如制定完成一个新的任务后,发送消息给相关人员,或是在文件上传完毕后,发送给文件上传人员,也可以是定时任务中,指定时间点发送消息,对于这部分来讲,在领域层建立一个消息通知的类,来负责处理消息通知,而具体实现消息通知的过程ABP框架中已经实现了,如同消息订阅一样,只需要在构造函数中注入即可,其次定义具体的消息发送方法,使得具体操作结束后调用具体的消息发送来发送消息。

/// <summary>
/// 消息通知发布实例
/// </summary>
public class AppNotifier : DomainService, IAppNotifier
{
#region 初始化
private readonly INotificationPublisher _notificationPublisher; public AppNotifier(INotificationPublisher notificationPublisher)
{
_notificationPublisher = notificationPublisher;
}
#endregion #region 消息发送
public async Task SendMessageAsync(UserIdentifier user, string message, NotificationSeverity severity = NotificationSeverity.Info)
{
await _notificationPublisher.PublishAsync(
AppNotificationNames.SimpleMessage,
new MessageNotificationData(message),
severity: severity,
userIds: new[] { user }
);
}
#endregion #region 任务提醒
public async Task NewTaskAsync(string message, NotificationSeverity severity = NotificationSeverity.Info)
{
await PublishMessage(AppNotificationNames.NewTask, message);
}
#endregion #region 辅助方法
private async Task PublishMessage(string appNotificationName, string message)
{
await _notificationPublisher.PublishAsync(appNotificationName, new MessageNotificationData(message));
}
#endregion
}

二、完成用户订阅消息

  在MVC层,在用户控制器中完成对消息通知设置,消息本身是独立的,只应有了用户才活跃起来,因此把消息的设定挂钩在用户身上,应该算是合理的,并同时完成页面的设计。

/// <summary>
/// 通知设置
/// </summary>
/// <returns></returns>
public async Task<ActionResult> NotificationSetting()
{
var notificationSettings = await _notificationAppService.GetNotificationSettings();
return View(notificationSettings);
} /// <summary>
/// 更新通知订阅
/// </summary>
/// <returns></returns>
[HttpPost]
public async Task<JsonResult> UpdateNotificationSetting([FromBody]UpdateNotificationSettingsInput input)
{
await _notificationAppService.UpdateNotificationSettings(input);
return Json(new ResponseParamViewModel("通知设置已更新"));
}

  页面设计只需要对提前在领域层中定义好的消息类型展示出来即可,用户自主勾选哪些想要接收的消息提示。

  

三、发布消息通知用户

  在指定操作完毕,可以调用消息服务来推送消息到目标接收人员,此处,更改了原先数据字典创建处的代码,当创建数据字典成功后,给与一个消息提示来模拟发布消息通知用户的过程。

var existedDataDictionary = await _dataDictionaryRepository.GetAll().Where(d => d.TypeName == input.DataDictionary.TypeName).FirstOrDefaultAsync();

if (existedDataDictionary != null)
{
throw new UserFriendlyException(L("该字典类型已存在,无法添加"));
} var dataDictionary = ObjectMapper.Map<DataDictionary>(input.DataDictionary);
await _dataDictionaryRepository.InsertAsync(dataDictionary); //模拟测试消息通知
await _appNotifier.NewTaskAsync("字典类型已完成添加,可以开始使用了");

  该消息触发的前提是,需要订阅该消息,不然有人发,无人收,总是很难为情。因此提前在通知设置中开启类型为“新的任务”的开关。

  

  对于消息发送到了浏览器中呈现,这个过程采用了SignalR,ABP框架也完成了前后端的消息通知,因此可以直接使用即可。

  至此,站内消息通知的设计完毕,对于业务需求中可能面对的更为丰富的功能就得花更多时间来完成了。 

  代码地址:https://gitee.com/530521314/Partner.Surround.git

2019-07-27,望技术有成后能回来看见自己的脚步

X-Admin&ABP框架开发-消息通知的更多相关文章

  1. iOS开发-消息通知机制(NSNotification和NSNotificationCenter)

    iOS中委托模式和消息机制基本上开发中用到的比较多,一般最开始页面传值通过委托实现的比较多,类之间的传值用到的比较多,不过委托相对来说只能是一对一,比如说页面A跳转到页面B,页面的B的值改变要映射到页 ...

  2. X-Admin&ABP框架开发-系统日志

    网站正常运行中有时出现异常在所难免,查看系统运行日志分析问题并能够根据错误信息快速解决问题尤为重要,ABP对于系统运行日志这块已经做了很好的处理,默认采用的Log4Net已经足够满足开发过程中的需要了 ...

  3. X-Admin&ABP框架开发-版本管理

    多租户系统中,针对于不同租户开放不同功能,或是按照不同功能进行收费管理,需要从宿主本身去管理租户的版本信息,如同酒店人员对不同房间收取不同费用,依据房间内部设施,房间大小等设置不同收费标准.Abp系统 ...

  4. X-Admin&ABP框架开发-设置管理

    在网站开发中,设置是不可缺少的一环,如用户设置.系统设置.甚至是租户设置等.ABP对于设置的管理已经做了很好的处理,我们可以借助巨人的力量来完成我们的冒险. ABP官网地址:https://aspne ...

  5. X-Admin&ABP框架开发-代码生成器

    在日常开发中,有时会遇到一些相似的代码,甚至是只要CV一次,改几个名称,就可以实现功能了,而且总归起来,都可以由一些公用的页面更改而来,因此,结合我日常开发中使用到的页面,封装一个适合自己的代码生成器 ...

  6. X-Admin&ABP框架开发-RBAC

    在业务系统需求规划过程中,通常对于诸如组织机构.用户和角色等这种基础功能,通常是将这部分功能规划到通用子域中,这也说明了,对于这部分功能来讲,是系统的基石,整个业务体系是建立于这部分基石之上的,当然, ...

  7. X-Admin&ABP框架开发-数据字典

    在业务型的系统开发中,我们需要维护各种个样的类型,比如客户类型.客户行业.商品类型等等,这些类型往往信息量不多,并且相似度极高,如果采用一类型一表去设计,将会造成极大的工作量,通过将这部分类型的信息进 ...

  8. X-Admin&ABP框架开发-租户管理

    软件即服务概念的推动,定制化到通用化的发展,用一套代码完成适应不同企业的需求,利用多租户技术可以去做到这一点.ABP里提供了多租户这一概念并且也在Zero模块中实现了这一概念. 一.多租户的概念 单部 ...

  9. ABP框架个人开发实战(1)_环境搭建

    前言 之前关注ABP框架有一阵子了,一直没有潜下心来实际研究一下.最近想自己建站,以后有自己的功能开发项目,可以在自己的站点上开发,并一步步的完善,所以找个比较好用的框架迫在眉睫,选来选去,决定用AB ...

随机推荐

  1. Spring Boot从入门到实战:集成AOPLog来记录接口访问日志

    日志是一个Web项目中必不可少的部分,借助它我们可以做许多事情,比如问题排查.访问统计.监控告警等.一般通过引入slf4j的一些实现框架来做日志功能,如log4j,logback,log4j2,其性能 ...

  2. 原创-使用pywinauto进行dotnet的winform程序控制(一)

    pywinauto自动化控制win32的ui的程序,网上有好多的教程.但是操作dotnet写的winform教程,就少之又少.今天我就来分享我的pywinauto操作dotnet的winform的研究 ...

  3. spring 5.x 系列第1篇 —— springmvc基础 (xml配置方式)

    文章目录 一.搭建hello spring工程 1.1 项目搭建 1.2 相关配置讲解 二.配置自定义拦截器 三.全局异常处理 四.参数绑定 4.1 参数绑定 4.2 关于日期格式转换的三种方法 五. ...

  4. jQuery中ajax-$.getJSON,$.getScript

    jQuery提供了两个特定异步加载的方法$.getJSON()方法和$.getScript()方法 $.getJSON()来加载特定的json文件 $.getScript()来加载特定的javascr ...

  5. Python开发【第九篇】: 并发编程

    内容概要 操作系统介绍 进程 线程 协程 二. 进程 python并发编程之多进程理论部分 在python程序中的进程操作 运行中的程序就是一个进程.所有的进程都是通过它的父进程来创建的.因此,运行起 ...

  6. 风险:隐蔽权限修改导致rgw服务中断

    上午正在开会,突然收到rgw服务异常的告警(503 Service Unavailable),立马停下来处理告警,避免影响到用户~   我们的rgw frontend用的是apache,之前也遇到过5 ...

  7. RQNOJ193 造路行动

    题目转移 详见最小生成树讲解 Kruskal #include<cstdio> #include<algorithm> using namespace std; ; int n ...

  8. BZOJ 3289:Mato的文件管理(莫队算法+树状数组)

    http://www.lydsy.com/JudgeOnline/problem.php?id=3289 题意:…… 思路:求交换次数即求逆序对数.确定了这个之后,先离散化数组.然后在后面插入元素的话 ...

  9. HDU 3068:最长回文(Manacher算法)

    http://acm.hdu.edu.cn/showproblem.php?pid=3068 最长回文 Problem Description   给出一个只由小写英文字符a,b,c...y,z组成的 ...

  10. 从零开始基于go-thrift创建一个RPC服务

    Thrift 是一种被广泛使用的 rpc 框架,可以比较灵活的定义数据结构和函数输入输出参数,并且可以跨语言调用.为了保证服务接口的统一性和可维护性,我们需要在最开始就制定一系列规范并严格遵守,降低后 ...