.Net Core应用搭建的分布式邮件系统设计

本篇分享的是由NetCore搭建的分布式邮件系统,主要采用NetCore的Api控制台应用程序,由于此系统属于公司的所以这里只能分享设计图和一些单纯不设计业务的类或方法;


为什么要在公司中首例采用NetCore做开发

为什么要在公司中首例采用NetCore做开发,有些netcoreapi不是还不全面么,您都敢尝试?恐怕会有人这样问我,我只能告诉你NetCore现在出2.0版本了,很多Framwork的常用封装都已经有了,况且她主打的是MVC模式,能够高效的开发系统,也有很多Core的Nuget包支持了,已经到达了几乎可以放心大胆使用的地步,退一万不说有些东西不支持那这又如何,可以采用接口的方式从其他地方对接过来也是一种不错的处理方案。为了让C#这门优秀的语言被广泛应用,默默努力着。

 

正片环节 - 分布式邮件系统设计图

分布式邮件系统说明

其实由上图可以知晓这里我主要采用了Api+服务的模式,这也是现在互联网公司经常采用的一种搭配默认;利用api接受请求插入待发送邮件队列和入库,然后通过部署多个NetCore跨平台服务(这里服务指的是:控制台应用)来做分布式处理操作,跨平台服务主要操作有:

. 邮件发送

. 邮件发送状态的通知(如果需要通知子业务,那么需要通知业务方邮件发送的状态)

. 通知失败处理(自动往绑定的责任人发送一封邮件)

. 填充队列(如果待发邮件队列或者通知队列数据不完整,需要修复队列数据)

Api接口的统一验证入口

这里我用最简单的方式,继承Controller封装了一个父级的BaseController,来让各个api的Controller基础统一来做身份验证;来看看重写 public override void OnActionExecuting(ActionExecutingContext context) 的验证代码:

public override void OnActionExecuting(ActionExecutingContext context)

        {

            base.OnActionExecuting(context);


            var moResponse = new MoBaseRp();

            try

            {


                #region 安全性验证


                var key = "request";

                if (!context.ActionArguments.ContainsKey(key)) { moResponse.Msg = "请求方式不正确"; return; }

                var request = context.ActionArguments[key];

                var baseRq = request as MoBaseRq;

                //暂时不验证登录账号密码

                if (string.IsNullOrWhiteSpace(baseRq.UserName) || string.IsNullOrWhiteSpace(baseRq.UserPwd)) { moResponse.Msg = "登录账号或密码不能为空"; return; }

                else if (baseRq.AccId <= 0) { moResponse.Msg = "发送者Id无效"; return; }

                else if (string.IsNullOrWhiteSpace(baseRq.FuncName)) { moResponse.Msg = "业务方法名不正确"; return; }


                //token验证

                var strToken = PublicClass._Md5($"{baseRq.UserName}{baseRq.AccId}", "");

                if (!strToken.Equals(baseRq.Token, StringComparison.OrdinalIgnoreCase)) { moResponse.Msg = "Token验证失败"; return; }


                //验证发送者Id

                if (string.IsNullOrWhiteSpace(baseRq.Ip))

                {

                    var account = _db.EmailAccount.SingleOrDefault(b => b.Id == baseRq.AccId);

                    if (account == null) { moResponse.Msg = "发送者Id无效。"; return; }

                    else

                    {

                        if (account.Status != (int)EnumHelper.EmStatus.启用)

                        {

                            moResponse.Msg = "发送者Id已禁用"; return;

                        }


                        //验证ip

                        var ipArr = account.AllowIps.Split(new char[] { ',' }, StringSplitOptions.RemoveEmptyEntries);

                        //当前请求的Ip

                        var nowIp = this.GetUserIp();

                        baseRq.Ip = nowIp;

                        //默认*为所有ip , 匹配ip

                        if (!ipArr.Any(b => b.Equals("*")) && !ipArr.Any(b => b.Equals(nowIp)))

                        {

                            moResponse.Msg = "请求IP为授权"; return;

                        }

                    }

                }

                else

                {

                    var account = _db.EmailAccount.SingleOrDefault(b => b.Id == baseRq.AccId && b.AllowIps.Any(bb => bb.Equals(baseRq.Ip)));

                    if (account == null) { moResponse.Msg = "发送者未授权"; return; }

                    else if (account.Status != (int)EnumHelper.EmStatus.启用)

                    {

                        moResponse.Msg = "发送者Id已禁用"; return;

                    }

                }


                //内容非空,格式验证

                if (!context.ModelState.IsValid)

                {

                    var values = context.ModelState.Values.Where(b => b.Errors.Count > 0);

                    if (values.Count() > 0)

                    {

                        moResponse.Msg = values.First().Errors.First().ErrorMessage;

                        return;

                    }

                }


                #endregion


                moResponse.Status = 1;

            }

            catch (Exception ex)

            {

                moResponse.Msg = "O No请求信息错误";

            }

            finally

            {

                if (moResponse.Status == 0) { context.Result = Json(moResponse);             }

            }

        }

邮件请求父类实体:

/// <summary>

    /// 邮件请求父类

    /// </summary>

    public class MoBaseRq

    {


        public string UserName { get; set; }


        public string UserPwd { get; set; }


        /// <summary>

        /// 验证token(Md5(账号+配置发送者账号信息的Id+Ip))   必填

        /// </summary>

        public string Token { get; set; }


        /// <summary>

        /// 配置发送者账号信息的Id  必填

        /// </summary>

        public int AccId { get; set; }


        /// <summary>

        /// 业务方法名称

        /// </summary>

        public string FuncName { get; set; }


        /// <summary>

        /// 请求者Ip,如果客户端没赋值,默认服务端获取

        /// </summary>

        public string Ip { get; set; }


    }

第三方Nuget包的便利


此邮件系统使用到了第三方包,这也能够看出有很多朋友正为开源,便利,NetCore的推广努力着;

首先看看MailKit(邮件发送)包,通过安装下载命令: Install-Package MailKit 能够下载最新包,然后你不需要做太花哨的分装,只需要正对于邮件发送的服务器,端口,账号,密码做一些设置基本就行了,如果可以您可以直接使用我的代码:

/// <summary>

        /// 发送邮件

        /// </summary>

        /// <param name="dicToEmail"></param>

        /// <param name="title"></param>

        /// <param name="content"></param>

        /// <param name="name"></param>

        /// <param name="fromEmail"></param>

        /// <returns></returns>

        public static bool _SendEmail(

            Dictionary<string, string> dicToEmail,

            string title, string content,

            string name = "爱留图网", string fromEmail = "841202396@qq.com",

            string host = "smtp.qq.com", int port = 587,

            string userName = "841202396@qq.com", string userPwd = "123123")

        {

            var isOk = false;

            try

            {

                if (string.IsNullOrWhiteSpace(title) || string.IsNullOrWhiteSpace(content)) { return isOk; }


                //设置基本信息

                var message = new MimeMessage();

                message.From.Add(new MailboxAddress(name, fromEmail));

                foreach (var item in dicToEmail.Keys)

                {

                    message.To.Add(new MailboxAddress(item, dicToEmail[item]));

                }

                message.Subject = title;

                message.Body = new TextPart("html")

                {

                    Text = content

                };


                //链接发送

                using (var client = new SmtpClient())

                {

                    // For demo-purposes, accept all SSL certificates (in case the server supports STARTTLS)

                    client.ServerCertificateValidationCallback = (s, c, h, e) => true;


                    //采用qq邮箱服务器发送邮件

                    client.Connect(host, port, false);


                    // Note: since we don't have an OAuth2 token, disable

                    // the XOAUTH2 authentication mechanism.

                    client.AuthenticationMechanisms.Remove("XOAUTH2");


                    //qq邮箱,密码(安全设置短信获取后的密码)  ufiaszkkulbabejh

                    client.Authenticate(userName, userPwd);


                    client.Send(message);

                    client.Disconnect(true);

                }

                isOk = true;

            }

            catch (Exception ex)

            {


            }

            return isOk;

        }

Redis方面的操作包StackExchange.Redis,现在NetCore支持很多数据库驱动(例如:Sqlserver,mysql,postgressql,db2等)这么用可以参考下这篇文章AspNetCore - MVC实战系列(一)之Sqlserver表映射实体模型,不仅如此还支持很多缓存服务(如:Memorycach,Redis),这里讲到的就是Redis,我利用Redis的list的队列特性来做分布式任务存储,尽管目前我用到的只有一个主Redis服务还没有业务场景需要用到主从复制等功能;这里分享的代码是基于StackExchange.Redis基础上封装对于string,list的操作:

public class StackRedis : IDisposable

    {

        #region 配置属性   基于 StackExchange.Redis 封装

        //连接串 (注:IP:端口,属性=,属性=)

        public string _ConnectionString = "127.0.0.1:6377,password=shenniubuxing3";

        //操作的库(注:默认0库)

        public int _Db = 0;

        #endregion


        #region 管理器对象


        /// <summary>

        /// 获取redis操作类对象

        /// </summary>

        private static StackRedis _StackRedis;

        private static object _locker_StackRedis = new object();

        public static StackRedis Current

        {

            get

            {

                if (_StackRedis == null)

                {

                    lock (_locker_StackRedis)

                    {

                        _StackRedis = _StackRedis ?? new StackRedis();

                        return _StackRedis;

                    }

                }


                return _StackRedis;

            }

        }


        /// <summary>

        /// 获取并发链接管理器对象

        /// </summary>

        private static ConnectionMultiplexer _redis;

        private static object _locker = new object();

        public ConnectionMultiplexer Manager

        {

            get

            {

                if (_redis == null)

                {

                    lock (_locker)

                    {

                        _redis = _redis ?? GetManager(this._ConnectionString);

                        return _redis;

                    }

                }


                return _redis;

            }

        }


        /// <summary>

        /// 获取链接管理器

        /// </summary>

        /// <param name="connectionString"></param>

        /// <returns></returns>

        public ConnectionMultiplexer GetManager(string connectionString)

        {

            return ConnectionMultiplexer.Connect(connectionString);

        }


        /// <summary>

        /// 获取操作数据库对象

        /// </summary>

        /// <returns></returns>

        public IDatabase GetDb()

        {

            return Manager.GetDatabase(_Db);

        }

        #endregion


        #region 操作方法


        #region string 操作


        /// <summary>

        /// 根据Key移除

        /// </summary>

        /// <param name="key"></param>

        /// <returns></returns>

        public async Task<bool> Remove(string key)

        {

            var db = this.GetDb();


            return await db.KeyDeleteAsync(key);

        }


        /// <summary>

        /// 根据key获取string结果

        /// </summary>

        /// <param name="key"></param>

        /// <returns></returns>

        public async Task<string> Get(string key)

        {

            var db = this.GetDb();

            return await db.StringGetAsync(key);

        }


        /// <summary>

        /// 根据key获取string中的对象

        /// </summary>

        /// <typeparam name="T"></typeparam>

        /// <param name="key"></param>

        /// <returns></returns>

        public async Task<T> Get<T>(string key)

        {

            var t = default(T);

            try

            {

                var _str = await this.Get(key);

                if (string.IsNullOrWhiteSpace(_str)) { return t; }


                t = JsonConvert.DeserializeObject<T>(_str);

            }

            catch (Exception ex) { }

            return t;

        }


        /// <summary>

        /// 存储string数据

        /// </summary>

        /// <param name="key"></param>

        /// <param name="value"></param>

        /// <param name="expireMinutes"></param>

        /// <returns></returns>

        public async Task<bool> Set(string key, string value, int expireMinutes = 0)

        {

            var db = this.GetDb();

            if (expireMinutes > 0)

            {

                return db.StringSet(key, value, TimeSpan.FromMinutes(expireMinutes));

            }

            return await db.StringSetAsync(key, value);

        }


        /// <summary>

        /// 存储对象数据到string

        /// </summary>

        /// <typeparam name="T"></typeparam>

        /// <param name="key"></param>

        /// <param name="value"></param>

        /// <param name="expireMinutes"></param>

        /// <returns></returns>

        public async Task<bool> Set<T>(string key, T value, int expireMinutes = 0)

        {

            try

            {

                var jsonOption = new JsonSerializerSettings()

                {

                    ReferenceLoopHandling = ReferenceLoopHandling.Ignore

                };

                var _str = JsonConvert.SerializeObject(value, jsonOption);

                if (string.IsNullOrWhiteSpace(_str)) { return false; }


                return await this.Set(key, _str, expireMinutes);

            }

            catch (Exception ex) { }

            return false;

        }

        #endregion


        #region List操作(注:可以当做队列使用)


        /// <summary>

        /// list长度

        /// </summary>

        /// <typeparam name="T"></typeparam>

        /// <param name="key"></param>

        /// <returns></returns>

        public async Task<long> GetListLen<T>(string key)

        {

            try

            {

                var db = this.GetDb();

                return await db.ListLengthAsync(key);

            }

            catch (Exception ex) { }

            return 0;

        }


        /// <summary>

        /// 获取队列出口数据并移除

        /// </summary>

        /// <typeparam name="T"></typeparam>

        /// <param name="key"></param>

        /// <returns></returns>

        public async Task<T> GetListAndPop<T>(string key)

        {

            var t = default(T);

            try

            {

                var db = this.GetDb();

                var _str = await db.ListRightPopAsync(key);

                if (string.IsNullOrWhiteSpace(_str)) { return t; }

                t = JsonConvert.DeserializeObject<T>(_str);

            }

            catch (Exception ex) { }

            return t;

        }


        /// <summary>

        /// 集合对象添加到list左边

        /// </summary>

        /// <typeparam name="T"></typeparam>

        /// <param name="key"></param>

        /// <param name="values"></param>

        /// <returns></returns>

        public async Task<long> SetLists<T>(string key, List<T> values)

        {

            var result = 0L;

            try

            {

                var jsonOption = new JsonSerializerSettings()

                {

                    ReferenceLoopHandling = ReferenceLoopHandling.Ignore

                };

                var db = this.GetDb();

                foreach (var item in values)

                {

                    var _str = JsonConvert.SerializeObject(item, jsonOption);

                    result += await db.ListLeftPushAsync(key, _str);

                }

                return result;

            }

            catch (Exception ex) { }

            return result;

        }


        /// <summary>

        /// 单个对象添加到list左边

        /// </summary>

        /// <typeparam name="T"></typeparam>

        /// <param name="key"></param>

        /// <param name="value"></param>

        /// <returns></returns>

        public async Task<long> SetList<T>(string key, T value)

        {

            var result = 0L;

            try

            {

                result = await this.SetLists(key, new List<T> { value });

            }

            catch (Exception ex) { }

            return result;

        }



        #endregion


        #region 额外扩展


        /// <summary>

        /// 手动回收管理器对象

        /// </summary>

        public void Dispose()

        {

            this.Dispose(_redis);

        }


        public void Dispose(ConnectionMultiplexer con)

        {

            if (con != null)

            {

                con.Close();

                con.Dispose();

            }

        }


        #endregion


        #endregion

    }

用到Redis的那些操作就添加哪些就行了,也不用太花哨能用就行;

如何生成跨平台的api服务和应用程序服务

这小节的内容最重要,由于之前有相关的文章,这里就不用再赘述了,来这里看看:Asp.NetCore1.1版本没了project.json,这样来生成跨平台包

原文地址:http://www.cnblogs.com/wangrudong003/p/6898386.html


.NET社区新闻,深度好文,微信中搜索dotNET跨平台或扫描二维码关注

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

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

相关文章

JavaFX UI控件教程(五)之Radio Button

翻译自 Radio Button 本章讨论单选按钮控件和RadioButton类&#xff0c;该类的一个专门实现ToggleButton。 可以选择或取消选择单选按钮控件。通常&#xff0c;单选按钮组合成一个组&#xff0c;其中一次只能选择一个按钮。此行为将它们与切换按钮区分开来&#xff0c;因为组…

.NET开源MSSQL、Redis监控产品Opserver之Redis配置

Redis监控数据实例的加载可以查看Opserver.Core项目data/Redis文件夹下的RedisModule.cs,我加了点注释 /// <summary>/// 加载Redis连接 /// </summary>/// <returns></returns>private static List<RedisConnectionInfo> LoadRedisConn…

使用变量注意事项

Java中每个变量必须先声明&#xff0c;后使用(否则会编译失败) 使用变量名来访问这块区域的数据 变量的作用域&#xff1a;其定义所在的一对{ }内 变量只有在其作用域内才有效 同一个作用域内&#xff0c;不能定义重名的变量

JavaFX UI控件教程(六)之Toggle Button

翻译自 Toggle Button 在本章中&#xff0c;您将了解ToggleButton该类&#xff0c;这是通过JavaFX API提供的另一种类型的按钮。 可以将两个或多个切换按钮组合成一个组&#xff0c;其中一次只能选择一个按钮&#xff0c;或者不需要选择。图5-1是组合了三个切换按钮的应用程…

窥探ASP.Net MVC底层原理 实现跨越Session的分布式TempData

Hi,guys!Long time no see! 1、问题的引出 我相信大家在项目中都使用过TempData&#xff0c;TempData是一个字典集合&#xff0c;一般用于两个请求之间临时缓存数据或者页面之间传递消息。也都知道TempData是用Session来实现的&#xff0c;既然是用Session来实现的&#xff0…

某同学正为自己安装不上sqlserver数据库而愁眉苦脸,使用朋友给的方法顿时喜笑颜开,那么朋友到底出了个什么样的方法呢?...

小故事引入&#xff1a;张同学下一节课就要开始学习使用jdbc访问Sql Server数据库了&#xff0c;部分同学由于刚换电脑&#xff0c;导致没有安装数据库&#xff0c;于是同学们按照老师给的方法&#xff08;文末提供老师的方法&#xff09;&#xff0c;将5G多的Sqlserver的安装包…

JavaFX UI控件教程(七)之Checkbox

翻译自 Checkbox 本章教授如何向JavaFX应用程序添加复选框。 虽然复选框看起来类似于单选按钮&#xff0c;但它们不能组合到切换组中以便一次选择多个选项。有关详细信息&#xff0c;请参阅单选按钮和切换按钮章节。 图6-1显示了一个应用程序的屏幕截图&#xff0c;其中三…

.net Kafka.Client多个Consumer Group对Topic消费不能完全覆盖研究总结(一)

我们知道Kafka支持Consumer Group的功能&#xff0c;但是最近在应用Consumer Group时发现了一个Topic 的Partition不能100%覆盖的问题。 程序部署后&#xff0c;发现Kafka在pdb组的consumer消费topic时存在问题&#xff0c;consumer无法完全覆盖Topic的各个partition。如下图&…

JavaFX UI控件教程(八)之Choice Box

翻译自 Choice Box 本章介绍了选项框&#xff0c;这些UI控件提供了在几个选项之间快速选择的支持。 使用ChoiceBox该类将选择框添加到JavaFX应用程序。其简单的实现如图7-1所示。 图7-1创建包含三个项目的选择框 创建一个选择框 例7-1创建了一个包含三个项目的选择框。 例…

View Components as Tag Helpers,离在线模板编辑又进一步

在asp.net core mvc中增加了ViewComponent&#xff08;视图组件&#xff09;的概念&#xff0c;视图组件有点类似部分视图&#xff0c;但是比部分视图功能更加强大&#xff0c;它更有点像一个控制器。 使用方法 1&#xff0c;定义类派生自ViewComponent类 2&#xff0c;增加…

JavaFX UI控件教程(九)之Text Field

翻译自 Text Field 本章讨论文本字段控件的功能。 的TextField类实现接受并显示文本输入的UI控制。它提供了从用户接收文本输入的功能。与另一个文本输入控件一起&#xff0c;PasswordField此类扩展了TextInput类&#xff0c;它是通过JavaFX API提供的所有文本控件的超类。…

VS

&—逻辑与 | —逻辑或 &#xff01;—逻辑非 && —短路与 || —短路或 ^ —逻辑异或

ssl1643-最小乘车费用【dp练习】

最小乘车费用 题目 假设某条街上每一公里就有一个公共汽车站&#xff0c;并且乘车费用如下表&#xff1a;      而任意一辆汽车从不行驶超过10公里。某人想行驶n公里&#xff0c;假设他可以任意次换车&#xff0c;请你帮他找到一种乘车方案&#xff0c;使得总费用最小 …

JavaFX UI控件教程(十)之Scroll Bar

翻译自 Scroll Bar 本章介绍如何使用滚动条控件创建可滚动窗格。 本ScrollBar类可以在应用程序中创建滚动窗格和意见。图9-1显示了滚动条的三个区域&#xff1a;拇指&#xff0c;右侧和左侧按钮&#xff08;或向下和向上按钮&#xff09;以及轨道。 图9-1滚动条的元素 创建…

a+=b不一定等于a=a+b

说明不会改变本身变量的数据类型与&#xff0c;–运算符一样

Jexus部署.Net Core项目

Jexus Jexus 即 Jexus Web Server&#xff0c;简称JWS&#xff0c;是Linux平台上 的一款ASP.NET WEB服务器。它是 Linux、Unix、FreeBSD 等非Windows系统架设 ASP.NET WEB 服务器的核心程序。 将HTTP自宿主应用程序&#xff08;如Asp.net Core应用程序、Node.js应用程序等&…

JavaFX UI控件教程(十一)之Scroll Pane

翻译自 Scroll Pane 在本章中&#xff0c;您将学习如何在JavaFX应用程序中构建滚动窗格。 滚动窗格提供UI元素的可滚动视图。此控件使用户可以通过平移视口或使用滚动条来滚动内容。具有默认设置和添加的图像的滚动窗格如图10-1所示。 图10-1滚动窗格 创建滚动窗格 示例10…

Docker Machine 简介

Docker Machine 是什么&#xff1f; Docker Machine 是 Docker 官方提供的一个工具&#xff0c;它可以帮助我们在远程的机器上安装 Docker&#xff0c;或者在虚拟机 host 上直接安装虚拟机并在虚拟机中安装 Docker。我们还可以通过 docker-machine 命令来管理这些虚拟机和 Doc…

C#基础知识详解之【字段与属性】

讲理论知识之前&#xff0c;先看一段代码&#xff1a;public class Emp {//字段private int age;//属性public int Age{get { return age; }set { age value; }} }大家可以看到上面实例代码中&#xff0c;声明了一个名为age的字段&#xff0c;还有一个名为Age的属性&#xff0…