.Net Core應用搭建的分散式郵件系統設計

来源:http://www.cnblogs.com/wangrudong003/archive/2017/05/24/6898386.html
-Advertisement-
Play Games

本篇分享的是由NetCore搭建的分散式郵件系統,主要採用NetCore的Api和控制台應用程式,由於此系統屬於公司的所以這裡只能分享設計圖和一些單純不設計業務的類或方法; 為什麼要在公司中首例採用NetCore做開發 為什麼要在公司中首例採用NetCore做開發,有些netcoreapi不是還不全 ...


本篇分享的是由NetCore搭建的分散式郵件系統,主要採用NetCore的Api控制台應用程式,由於此系統屬於公司的所以這裡只能分享設計圖和一些單純不設計業務的類或方法;

為什麼要在公司中首例採用NetCore做開發

為什麼要在公司中首例採用NetCore做開發,有些netcoreapi不是還不全面麽,您都敢嘗試?恐怕會有人這樣問我,我只能告訴你NetCore現在出2.0版本了,很多Framwork的常用封裝都已經有了,況且她主打的是MVC模式,能夠高效的開發系統,也有很多Core的Nuget包支持了,已經到達了幾乎可以放心大膽使用的地步,退一萬不說有些東西不支持那這又如何,可以採用介面的方式從其他地方對接過來也是一種不錯的處理方案。為了讓C#這門優秀的語言被廣泛應用,默默努力著。

目前我寫的NetCore方面的文章

AspNetCore - MVC實戰系列目錄

.NetCore上傳多文件的幾種示例

開源一個跨平臺運行的服務插件 - TaskCore.MainForm

NET Core-學習筆記

Asp.NetCore1.1版本沒了project.json,這樣來生成跨平臺包

 

正片環節 - 分散式郵件系統設計圖

分散式郵件系統說明

其實由上圖可以知曉這裡我主要採用了Api+服務的模式,這也是現在互聯網公司經常採用的一種搭配預設;利用api接受請求插入待發送郵件隊列和入庫,然後通過部署多個NetCore跨平臺服務(這裡服務指的是:控制台應用)來做分散式處理操作,跨平臺服務主要操作有:

. 郵件發送

. 郵件發送狀態的通知(如果需要通知子業務,那麼需要通知業務方郵件發送的狀態)

. 通知失敗處理(自動往綁定的責任人發送一封郵件)

. 填充隊列(如果待發郵件隊列或者通知隊列數據不完整,需要修複隊列數據)

Api介面的統一驗證入口

這裡我用最簡單的方式,繼承Controller封裝了一個父級的BaseController,來讓各個api的Controller基礎統一來做身份驗證;來看看重寫 public override void OnActionExecuting(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                 //驗證發送者Id
25                 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                     else
30                     {
31                         if (account.Status != (int)EnumHelper.EmStatus.啟用)
32                         {
33                             moResponse.Msg = "發送者Id已禁用"; return;
34                         }
35 
36                         //驗證ip
37                         var ipArr = account.AllowIps.Split(new char[] { ',' }, StringSplitOptions.RemoveEmptyEntries);
38                         //當前請求的Ip
39                         var nowIp = this.GetUserIp();
40                         baseRq.Ip = nowIp;
41                         //預設*為所有ip , 匹配ip
42                         if (!ipArr.Any(b => b.Equals("*")) && !ipArr.Any(b => b.Equals(nowIp)))
43                         {
44                             moResponse.Msg = "請求IP為授權"; return;
45                         }
46                     }
47                 }
48                 else
49                 {
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                 #endregion
70 
71                 moResponse.Status = 1;
72             }
73             catch (Exception ex)
74             {
75                 moResponse.Msg = "O No請求信息錯誤";
76             }
77             finally
78             {
79                 if (moResponse.Status == 0) { context.Result = Json(moResponse); }
80             }
81         }

郵件請求父類實體:

 1 /// <summary>
 2     /// 郵件請求父類
 3     /// </summary>
 4     public class MoBaseRq
 5     {
 6 
 7         public string UserName { get; set; }
 8 
 9         public string UserPwd { get; set; }
10 
11         /// <summary>
12         /// 驗證token(Md5(賬號+配置發送者賬號信息的Id+Ip))   必填
13         /// </summary>
14         public string Token { get; set; }
15 
16         /// <summary>
17         /// 配置發送者賬號信息的Id  必填
18         /// </summary>
19         public int AccId { get; set; }
20 
21         /// <summary>
22         /// 業務方法名稱
23         /// </summary>
24         public string FuncName { get; set; }
25 
26         /// <summary>
27         /// 請求者Ip,如果客戶端沒賦值,預設服務端獲取
28         /// </summary>
29         public string Ip { get; set; }
30 
31     }

第三方Nuget包的便利

此郵件系統使用到了第三方包,這也能夠看出有很多朋友正為開源,便利,NetCore的推廣努力著;

首先看看MailKit(郵件發送)包,通過安裝下載命令: Install-Package MailKit 能夠下載最新包,然後你不需要做太花哨的分裝,只需要正對於郵件發送的伺服器,埠,賬號,密碼做一些設置基本就行了,如果可以您可以直接使用我的代碼:

 1 /// <summary>
 2         /// 發送郵件
 3         /// </summary>
 4         /// <param name="dicToEmail"></param>
 5         /// <param name="title"></param>
 6         /// <param name="content"></param>
 7         /// <param name="name"></param>
 8         /// <param name="fromEmail"></param>
 9         /// <returns></returns>
10         public static bool _SendEmail(
11             Dictionary<string, string> 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             try
19             {
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 = content
33                 };
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, disable
45                     // the XOAUTH2 authentication mechanism.
46                     client.AuthenticationMechanisms.Remove("XOAUTH2");
47 
48                     //qq郵箱,密碼(安全設置簡訊獲取後的密碼)  ufiaszkkulbabejh
49                     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         /// <summary>
 13         /// 獲取redis操作類對象
 14         /// </summary>
 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         /// <summary>
 35         /// 獲取併發鏈接管理器對象
 36         /// </summary>
 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         /// <summary>
 57         /// 獲取鏈接管理器
 58         /// </summary>
 59         /// <param name="connectionString"></param>
 60         /// <returns></returns>
 61         public ConnectionMultiplexer GetManager(string connectionString)
 62         {
 63             return ConnectionMultiplexer.Connect(connectionString);
 64         }
 65 
 66         /// <summary>
 67         /// 獲取操作資料庫對象
 68         /// </summary>
 69         /// <returns></returns>
 70         public IDatabase GetDb()
 71         {
 72             return Manager.GetDatabase(_Db);
 73         }
 74         #endregion
 75 
 76         #region 操作方法
 77 
 78         #region string 操作
 79 
 80         /// <summary>
 81         /// 根據Key移除
 82         /// </summary>
 83         /// <param name="key"></param>
 84         /// <returns></returns>
 85         public async Task<bool> Remove(string key)
 86         {
 87             var db = this.GetDb();
 88 
 89             return await db.KeyDeleteAsync(key);
 90         }
 91 
 92         /// <summary>
 93         /// 根據key獲取string結果
 94         /// </summary>
 95         /// <param name="key"></param>
 96         /// <returns></returns>
 97         public async Task<string> Get(string key)
 98         {
 99             var db = this.GetDb();
100             return await db.StringGetAsync(key);
101         }
102 
103         /// <summary>
104         /// 根據key獲取string中的對象
105         /// </summary>
106         /// <typeparam name="T"></typeparam>
107         /// <param name="key"></param>
108         /// <returns></returns>
109         public async Task<T> Get<T>(string key)
110         {
111             var t = default(T);
112             try
113             {
114                 var _str = await this.Get(key);
115                 if (string.IsNullOrWhiteSpace(_str)) { return t; }
116 
117                 t = JsonConvert.DeserializeObject<T>(_str);
118             }
119             catch (Exception ex) { }
120             return t;
121         }
122 
123         /// <summary>
124         /// 存儲string數據
125         /// </summary>
126         /// <param name="key"></param>
127         /// <param name="value"></param>
128         /// <param name="expireMinutes"></param>
129         /// <returns></returns>
130         public async Task<bool> 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         /// <summary>
141         /// 存儲對象數據到string
142         /// </summary>
143         /// <typeparam name="T"></typeparam>
144         /// <param name="key"></param>
145         /// <param name="value"></param>
146         /// <param name="expireMinutes"></param>
147         /// <returns></returns>
148         public async Task<bool> Set<T>(string key, T value, int expireMinutes = 0)
149         {
150             try
151             {
152                 var jsonOption = new JsonSerializerSettings()
153                 {
154                     ReferenceLoopHandling = ReferenceLoopHandling.Ignore
155                 };
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         #endregion
165 
166         #region List操作(註:可以當做隊列使用)
167 
168         /// <summary>
169         /// list長度
170         /// </summary>
171         /// <typeparam name="T"></typeparam>
172         /// <param name="key"></param>
173         /// <returns></returns>
174         public async Task<long> GetListLen<T>(string key)
175         {
176             try
177             {
178                 var db = this.GetDb();
179                 return await db.ListLengthAsync(key);
180             }
181             catch (Exception ex) { }
182             return 0;
183         }
184 
185         /// <summary>
186         /// 獲取隊列出口數據並移除
187         /// </summary>
188         /// <typeparam name="T"></typeparam>
189         /// <param name="key"></param>
190         /// <returns></returns>
191         public async Task<T> GetListAndPop<T>(string key)
192         {
193             var t = default(T);
194             try
195             {
196                 var db = this.GetDb();
197                 var _str = await db.ListRightPopAsync(key);
198                 if (string.IsNullOrWhiteSpace(_str)) { return t; }
199                 t = JsonConvert.DeserializeObject<T>(_str);
200             }
201             catch (Exception ex) { }
202             return t;
203         }
204 
205         /// <summary>
206         /// 集合對象添加到list左邊
207         /// </summary>
208         /// <typeparam name="T"></typeparam>
209         /// <param name="key"></param>
210         /// <param name="values"></param>
211         /// <returns></returns>
212         public async Task<long> SetLists<T>(string key, List<T> values)
213         {
214             var result = 0L;
215             try
216             {
217                 var jsonOption = new JsonSerializerSettings()
218                 {
219                     ReferenceLoopHandling = ReferenceLoopHandling.Ignore
220                 };
221                 var db = this.GetDb();
222                 
              
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在後臺編寫Linq查詢的時候,有時我們寫的查詢語句,系統會給出一個警告:possible multiple enumeration of ienumerable. 出現這個警告的原因是: 如果該查詢是對資料庫的查詢,那麼該查詢語句會從資料庫執行兩次及以上,我們可以對這個查詢進行優化,優化的方法是 將 ...
  • 首先從資料庫讀取數據到DataTable,這我就不提了,大家都明白。下麵直接介紹如何從DataTable高效率導出數據到Excel中的方法,代碼如下: 說明: 1)上述方法中,將DataTable單元格內容寫入數組後一次性賦值給Excel的Range,效率非常高,比之迴圈DataTable單元格逐個 ...
  • 最新的NOPI應該是2.3了,但在官網上還是2.2.1。 也是第一次使用NPOI來導出Excel文件。 在寫的時候搜不到2.2.1的教程,搜了一個2.2.0的教程。 不過也沒什麼問題,NPOI是真的方便簡單。 不多說,放代碼 因為我的第一列是一個內碼值,要將他屏蔽掉, 所以在迴圈填充數據的時候,初始 ...
  • using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; namespace HellowWorld { class Prog ...
  • https://www.luogu.org/problem/show?pid=3518 問題描述: 輸入格式:n,k k個非負整數 輸出格式:一個數 樣例:入 42 5 28 31 10 38 24 出 14 k<=250 000 ,k<=n<=10^14; 分析;當x為密碼時,x 的因數和倍數都是 ...
  • 解決從Excel導入資料庫,導入到DataTable時數據類型發生變化的問題(如數字類型變成科學計數法,百分數變成小數) ...
  • NullReferenceException可能是.Net程式員遇到最多的例外了, 這個例外發生的如此頻繁, 以至於人們付出了巨大的努力來使用各種特性和約束試圖防止它發生, 但時至今日它仍然讓很多程式員頭痛, 今天我將講解這個令人頭痛的例外是如何發生的. 可以導致NullReferenceExcep ...
  • 我們在調試代碼的時候經常遇到DataTable的數據類型錯誤,這個類可以幫助我們很快查看DataTable的結構信息. ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...