正片环节 - 分布式邮件系统设计图
分布式邮件系统说明
其实由上图可以知晓这里我主要采用了Api+服务的模式,这也是现在互联网公司经常采用的一种搭配默认;利用api接受请求插入待发送邮件队列和入库,然后通过部署多个NetCore跨平台服务(这里服务指的是:控制台应用)来做分布式处理操作,跨平台服务主要操作有:
. 邮件发送
. 邮件发送状态的通知(如果需要通知子业务,那么需要通知业务方邮件发送的状态)
. 通知失败处理(自动往绑定的责任人发送一封邮件)
. 填充队列(如果待发邮件队列或者通知队列数据不完整,需要修复队列数据)
Api接口的统一验证入口
这里我用最简单的方式,继承Controller封装了一个父级的BaseController,来让各个api的Controller基础统一来做身份验证;来看看重写 public override voidOnActionExecuting(ActionExecutingContext context) 的验证代码:
1 public override void OnActionExecuting(ActionExecutingContext context) 2 { 3 base.OnActionExecuting(context); 4 5 var moResponse = new MoBaseRp(); 6 try 7 { 8 9 #region 安全性验证10 11 var key = "request";12 if (!context.ActionArguments.ContainsKey(key)) { moResponse.Msg = "请求方式不正确"; return; }13 var request = context.ActionArguments[key];14 var baseRq = request as MoBaseRq;15 //暂时不验证登录账号密码16 if (string.IsNullOrWhiteSpace(baseRq.UserName) || string.IsNullOrWhiteSpace(baseRq.UserPwd)) { moResponse.Msg = "登录账号或密码不能为空"; return; }17 else if (baseRq.AccId <= 0) { moResponse.Msg = "发送者Id无效"; return; }18 else if (string.IsNullOrWhiteSpace(baseRq.FuncName)) { moResponse.Msg = "业务方法名不正确"; return; }19 20 //token验证21 var strToken = PublicClass._Md5($"{baseRq.UserName}{baseRq.AccId}", "");22 if (!strToken.Equals(baseRq.Token, StringComparison.OrdinalIgnoreCase)) { moResponse.Msg = "Token验证失败"; return; }23 24 //验证发送者Id25 if (string.IsNullOrWhiteSpace(baseRq.Ip))26 {27 var account = _db.EmailAccount.SingleOrDefault(b => b.Id == baseRq.AccId);28 if (account == null) { moResponse.Msg = "发送者Id无效。"; return; }29 else30 {31 if (account.Status != (int)EnumHelper.EmStatus.启用)32 {33 moResponse.Msg = "发送者Id已禁用"; return;34 }35 36 //验证ip37 var ipArr = account.AllowIps.Split(new char[] { ',' }, StringSplitOptions.RemoveEmptyEntries);38 //当前请求的Ip39 var nowIp = this.GetUserIp();40 baseRq.Ip = nowIp;41 //默认*为所有ip , 匹配ip42 if (!ipArr.Any(b => b.Equals("*")) && !ipArr.Any(b => b.Equals(nowIp)))43 {44 moResponse.Msg = "请求IP为授权"; return;45 }46 }47 }48 else49 {50 var account = _db.EmailAccount.SingleOrDefault(b => b.Id == baseRq.AccId && b.AllowIps.Any(bb => bb.Equals(baseRq.Ip)));51 if (account == null) { moResponse.Msg = "发送者未授权"; return; }52 else if (account.Status != (int)EnumHelper.EmStatus.启用)53 {54 moResponse.Msg = "发送者Id已禁用"; return;55 }56 }57 58 //内容非空,格式验证59 if (!context.ModelState.IsValid)60 {61 var values = context.ModelState.Values.Where(b => b.Errors.Count > 0);62 if (values.Count() > 0)63 {64 moResponse.Msg = values.First().Errors.First().ErrorMessage;65 return;66 }67 }68 69 #endregion70 71 moResponse.Status = 1;72 }73 catch (Exception ex)74 {75 moResponse.Msg = "O No请求信息错误";76 }77 finally78 {79 if (moResponse.Status == 0) { context.Result = Json(moResponse); }80 }81 }
邮件请求父类实体:
1 ///2 /// 邮件请求父类 3 /// 4 public class MoBaseRq 5 { 6 7 public string UserName { get; set; } 8 9 public string UserPwd { get; set; }10 11 ///12 /// 验证token(Md5(账号+配置发送者账号信息的Id+Ip)) 必填13 /// 14 public string Token { get; set; }15 16 ///17 /// 配置发送者账号信息的Id 必填18 /// 19 public int AccId { get; set; }20 21 ///22 /// 业务方法名称23 /// 24 public string FuncName { get; set; }25 26 ///27 /// 请求者Ip,如果客户端没赋值,默认服务端获取28 /// 29 public string Ip { get; set; }30 31 }
第三方Nuget包的便利
此邮件系统使用到了第三方包,这也能够看出有很多朋友正为开源,便利,NetCore的推广努力着;
首先看看MailKit(邮件发送)包,通过安装下载命令: Install-Package MailKit 能够下载最新包,然后你不需要做太花哨的分装,只需要正对于邮件发送的服务器,端口,账号,密码做一些设置基本就行了,如果可以您可以直接使用我的代码:
1 ///2 /// 发送邮件 3 /// 4 /// 5 /// 6 /// 7 /// 8 /// 9 ///10 public static bool _SendEmail(11 Dictionary dicToEmail,12 string title, string content,13 string name = "爱留图网", string fromEmail = "[email protected]",14 string host = "smtp.qq.com", int port = 587,15 string userName = "[email protected]", string userPwd = "123123")16 {17 var isOk = false;18 try19 {20 if (string.IsNullOrWhiteSpace(title) || string.IsNullOrWhiteSpace(content)) { return isOk; }21 22 //设置基本信息23 var message = new MimeMessage();24 message.From.Add(new MailboxAddress(name, fromEmail));25 foreach (var item in dicToEmail.Keys)26 {27 message.To.Add(new MailboxAddress(item, dicToEmail[item]));28 }29 message.Subject = title;30 message.Body = new TextPart("html")31 {32 Text = content33 };34 35 //链接发送36 using (var client = new SmtpClient())37 {38 // For demo-purposes, accept all SSL certificates (in case the server supports STARTTLS)39 client.ServerCertificateValidationCallback = (s, c, h, e) => true;40 41 //采用qq邮箱服务器发送邮件42 client.Connect(host, port, false);43 44 // Note: since we don't have an OAuth2 token, disable45 // the XOAUTH2 authentication mechanism.46 client.AuthenticationMechanisms.Remove("XOAUTH2");47 48 //qq邮箱,密码(安全设置短信获取后的密码) ufiaszkkulbabejh49 client.Authenticate(userName, userPwd);50 51 client.Send(message);52 client.Disconnect(true);53 }54 isOk = true;55 }56 catch (Exception ex)57 {58 59 }60 return isOk;61 }
Redis方面的操作包StackExchange.Redis,现在NetCore支持很多数据库驱动(例如:Sqlserver,mysql,postgressql,db2等)这么用可以参考下这篇文章AspNetCore - MVC实战系列(一)之Sqlserver表映射实体模型,不仅如此还支持很多缓存服务(如:Memorycach,Redis),这里讲到的就是Redis,我利用Redis的list的队列特性来做分布式任务存储,尽管目前我用到的只有一个主Redis服务还没有业务场景需要用到主从复制等功能;这里分享的代码是基于StackExchange.Redis基础上封装对于string,list的操作:
1 public class StackRedis : IDisposable 2 { 3 #region 配置属性 基于 StackExchange.Redis 封装 4 //连接串 (注:IP:端口,属性=,属性=) 5 public string _ConnectionString = "127.0.0.1:6377,password=shenniubuxing3"; 6 //操作的库(注:默认0库) 7 public int _Db = 0; 8 #endregion 9 10 #region 管理器对象 11 12 ///13 /// 获取redis操作类对象 14 /// 15 private static StackRedis _StackRedis; 16 private static object _locker_StackRedis = new object(); 17 public static StackRedis Current 18 { 19 get 20 { 21 if (_StackRedis == null) 22 { 23 lock (_locker_StackRedis) 24 { 25 _StackRedis = _StackRedis ?? new StackRedis(); 26 return _StackRedis; 27 } 28 } 29 30 return _StackRedis; 31 } 32 } 33 34 ///35 /// 获取并发链接管理器对象 36 /// 37 private static ConnectionMultiplexer _redis; 38 private static object _locker = new object(); 39 public ConnectionMultiplexer Manager 40 { 41 get 42 { 43 if (_redis == null) 44 { 45 lock (_locker) 46 { 47 _redis = _redis ?? GetManager(this._ConnectionString); 48 return _redis; 49 } 50 } 51 52 return _redis; 53 } 54 } 55 56 ///57 /// 获取链接管理器 58 /// 59 /// 60 ///61 public ConnectionMultiplexer GetManager(string connectionString) 62 { 63 return ConnectionMultiplexer.Connect(connectionString); 64 } 65 66 /// 67 /// 获取操作数据库对象 68 /// 69 ///70 public IDatabase GetDb() 71 { 72 return Manager.GetDatabase(_Db); 73 } 74 #endregion 75 76 #region 操作方法 77 78 #region string 操作 79 80 /// 81 /// 根据Key移除 82 /// 83 /// 84 ///85 public async Task Remove(string key) 86 { 87 var db = this.GetDb(); 88 89 return await db.KeyDeleteAsync(key); 90 } 91 92 /// 93 /// 根据key获取string结果 94 /// 95 /// 96 ///97 public async Task Get(string key) 98 { 99 var db = this.GetDb();100 return await db.StringGetAsync(key);101 }102 103 /// 104 /// 根据key获取string中的对象105 /// 106 ///107 /// 108 /// 109 public async Task Get (string key)110 {111 var t = default(T);112 try113 {114 var _str = await this.Get(key);115 if (string.IsNullOrWhiteSpace(_str)) { return t; }116 117 t = JsonConvert.DeserializeObject (_str);118 }119 catch (Exception ex) { }120 return t;121 }122 123 /// 124 /// 存储string数据125 /// 126 /// 127 /// 128 /// 129 ///130 public async Task Set(string key, string value, int expireMinutes = 0)131 {132 var db = this.GetDb();133 if (expireMinutes > 0)134 {135 return db.StringSet(key, value, TimeSpan.FromMinutes(expireMinutes));136 }137 return await db.StringSetAsync(key, value);138 }139 140 /// 141 /// 存储对象数据到string142 /// 143 ///144 /// 145 /// 146 /// 147 /// 148 public async Task Set (string key, T value, int expireMinutes = 0)149 {150 try151 {152 var jsonOption = new JsonSerializerSettings()153 {154 ReferenceLoopHandling = ReferenceLoopHandling.Ignore155 };156 var _str = JsonConvert.SerializeObject(value, jsonOption);157 if (string.IsNullOrWhiteSpace(_str)) { return false; }158 159 return await this.Set(key, _str, expireMinutes);160 }161 catch (Exception ex) { }162 return false;163 }164 #endregion165 166 #region List操作(注:可以当做队列使用)167 168 /// 169 /// list长度170 /// 171 ///172 /// 173 /// 174 public async Task GetListLen (string key)175 {176 try177 {178 var db = this.GetDb();179 return await db.ListLengthAsync(key);180 }181 catch (Exception ex) { }182 return 0;183 }184 185 /// 186 /// 获取队列出口数据并移除187 /// 188 ///189 /// 190 /// 191 public async Task GetListAndPop (string key)192 {193 var t = default(T);194 try195 {196 var db = this.GetDb();197 var _str = await db.ListRightPopAsync(key);198 if (string.IsNullOrWhiteSpace(_str)) { return t; }199 t = JsonConvert.DeserializeObject (_str);200 }201 catch (Exception ex) { }202 return t;203 }204 205 /// 206 /// 集合对象添加到list左边207 /// 208 ///209 /// 210 /// 211 /// 212 public async Task SetLists (string key, List values)213 {214 var result = 0L;215 try216 {217 var jsonOption = new JsonSerializerSettings()218 {219 ReferenceLoopHandling = ReferenceLoopHandling.Ignore220 };221 var db = this.GetDb();222 foreach (var item in values)223 {224 var _str = JsonConvert.SerializeObject(item, jsonOption);225 result += await db.ListLeftPushAsync(key, _str);226 }227 return result;228 }229 catch (Exception ex) { }230 return result;231 }232 233 /// 234 /// 单个对象添加到list左边235 /// 236 ///237 /// 238 /// 239 /// 240 public async Task SetList (string key, T value)241 {242 var result = 0L;243 try244 {245 result = await this.SetLists(key, new List { value });246 }247 catch (Exception ex) { }248 return result;249 }250 251 252 #endregion253 254 #region 额外扩展255 256 /// 257 /// 手动回收管理器对象258 /// 259 public void Dispose()260 {261 this.Dispose(_redis);262 }263 264 public void Dispose(ConnectionMultiplexer con)265 {266 if (con != null)267 {268 con.Close();269 con.Dispose();270 }271 }272 273 #endregion274 275 #endregion276 }
用到Redis的那些操作就添加哪些就行了,也不用太花哨能用就行;