Magicodes.WeiChat——多租戶的設計與實現

来源:http://www.cnblogs.com/codelove/archive/2016/03/23/5313349.html
-Advertisement-
Play Games

多租戶(Multi Tenancy/Tenant)是一種軟體架構,其定義是:在一臺伺服器上運行單個應用實例,它為多個租戶提供服務。 本框架使用的是共用資料庫、共用 Schema、共用數據表的數據設計架構。 進入系統管理員界面,打開租戶管理界面,如下圖所示: 下麵是租戶管理界面: 這裡可以管理租戶成員 ...


概要

多租戶(Multi Tenancy/Tenant)是一種軟體架構,其定義是:在一臺伺服器上運行單個應用實例,它為多個租戶提供服務。

本框架使用的是共用資料庫、共用 Schema、共用數據表的數據設計架構。

操作說明

進入系統管理員界面,打開租戶管理界面,如下圖所示:

image

下麵是租戶管理界面:

image

這裡可以管理租戶成員,也可以讓管理員綁定微信。

下麵是公眾號配置界面:

image

這裡可以配置公眾號的信息。

系統管理員不僅可以管理自己的租戶,還可以管理其他租戶內容——公眾號管理。

下麵是公眾號管理界面:

image

架構實現

如上面所述,本框架使用的是共用資料庫、共用 Schema、共用數據表的數據設計架構。那麼,本框架是如何實現的呢?

主要是分為以下三步:

1. 建立TenantId

2. 擴展ASP.NET Indentity以支持多租戶

3. 註冊租戶篩選器

那麼首先,這裡需要介紹的是TenantId。

建立租戶Id(TenantId)

我們先來看看租戶表:

/// <summary>
    /// 租戶信息
    /// </summary>
    public class Account_Tenant
    {
        /// <summary>
        /// 多租戶Id
        /// </summary>
        public int Id { get; set; }
        /// <summary>
        /// 是否為系統租戶(僅支持一個)
        /// </summary>
        public bool IsSystemTenant { get; set; }
        /// <summary>
        /// 租戶名稱
        /// </summary>
        [Display(Name = "名稱")]
        [Required]
        [MaxLength(20)]
        public string Name { get; set; }
        [Display(Name = "備註")]
        [DataType(DataType.MultilineText)]
        [MaxLength(500)]
        public string Remark { get; set; }
}

如上所示,Id為主鍵,標識列,由資料庫自動生成(EF Code First模式下,預設Id為主鍵,int類型主鍵自動設置為標識列)。

那麼,租戶Id產生了之後,所有租戶共用數據表存放數據,不同租戶的數據需要通過 TenantId 欄位來區分。

我們來看一個基類的設計:

public abstract class WeiChat_TenantBase<TKey> : ITenantId, IAdminCreate<string>, IAdminUpdate<string>
    {
        [Key]
        public virtual TKey Id { get; set; }
        /// <summary>
        /// 創建時間
        /// </summary>
        [Display(Name = "創建時間")]
        public DateTime CreateTime { get; set; }
        /// <summary>
        /// 更新時間
        /// </summary>
        [Display(Name = "更新時間")]
        public DateTime? UpdateTime { get; set; }
        /// <summary>
        /// 創建者
        /// </summary>
        [MaxLength(128)]
        public string CreateBy { get; set; }

        /// <summary>
        /// 創建者
        /// </summary>
        [Display(Name = "創建者")]
        //[NotMapped]
        [ForeignKey("CreateBy")]
        public AppUser CreateUser { get; set; }

        /// <summary>
        /// 更新者
        /// </summary>
        [MaxLength(128)]
        public string UpdateBy { get; set; }
        /// <summary>
        /// 編輯者
        /// </summary>
        [MaxLength(256)]
        [Display(Name = "最後編輯")]
        //[NotMapped]
        public AppUser UpdateUser { get; set; }

        public int TenantId { get; set; }
}

如上所示,TenantId就是數據的分水嶺,不同數據的篩選需要根據其來篩選。

如下圖的設計:

image

從上圖可以看出,這塊錯綜複雜的類都缺不了TenantId,可能看類還是不太明白,我們來看表結構吧,比如說:

imageimageimageimage

等等。如上面表結構所示,TenantId為個表間必備欄位。

而在Code First模式下,使用繼承可以很方便的將所有的模型類加上相關欄位。

眾所周知,本框架使用了ASP.NET Indentity,那麼如何對ASP.NET Indentity實現多租戶的擴展呢?

擴展ASP.NET Indentity以支持多租戶

在本框架中,編寫了庫Magicodes.WeiChat.Data.Multitenant,用於擴展ASP.NET Indentity以支持多租戶。

使用過ASP.NET Indentity的朋友應該都知道Microsoft.AspNet.Identity.EntityFramework——ASP.NET Indentity使用EF作為其數據存儲的實現庫。通過對象瀏覽器查看,不難看出,其主要定義了以下對象:

image

其中,IdentityDbContext 繼承自System.Data.Entity.DbContext,具體定義如下所示:

public class IdentityDbContext<TUser, TRole, TKey, TUserLogin, TUserRole, TUserClaim> : System.Data.Entity.DbContext

where TUser : Microsoft.AspNet.Identity.EntityFramework.IdentityUser<TKey, TUserLogin, TUserRole, TUserClaim>

where TRole : Microsoft.AspNet.Identity.EntityFramework.IdentityRole<TKey, TUserRole>

where TUserLogin : Microsoft.AspNet.Identity.EntityFramework.IdentityUserLogin<TKey>

where TUserRole : Microsoft.AspNet.Identity.EntityFramework.IdentityUserRole<TKey>

where TUserClaim : Microsoft.AspNet.Identity.EntityFramework.IdentityUserClaim<TKey>

Microsoft.AspNet.Identity.EntityFramework 的成員

那麼,我們現在的主要對象就是搞定她們了。

image

一一對應關係如下所示:

image

如上所示,通過擴展ASP.NET Identity的IUser、IdentityUser、IdentityDbContext、IdentityUserLogin、UserStore來完成了對多租戶的支持,同時需要註意的是,還要重寫方法FindByNameAsync、AddLoginAsync、FindAsync、FindByEmailAsync、CreateAsync,以支持多租戶。

完成了對ASP.NET Identity的多租戶的支持,我們還需要對數據進行篩選,但是所有地方都添加篩選代碼是一件很麻煩的事情,而且在編寫邏輯的時候還很容易健忘,那麼有什麼好的方式呢?是時候祭出我們的神器了——EntityFramework.DynamicFilters。

註冊租戶篩選器

篩選器依賴ENTITYFRAMEWORK.DYNAMICFILTERS,這是一個開源項目,相關介紹可以訪問以下鏈接:

https://github.com/jcachat/EntityFramework.DynamicFilters

這裡,我們定義瞭如下租戶篩選器:

modelBuilder.Filter("TenantEntryFilter", (ITenantId app, int tenantId) => (app.TenantId == tenantId), 0);

然後我們可以使用以下代碼來啟用篩選器:

db.EnableFilter(tenantFilterName);

//設置多租戶過濾

db.SetFilterScopedParameterValue(tenantFilterName, "tenantId", TenantId);

以上代碼大家可以寫到通用的地方進行封裝,比如控制器基類的OnActionExecuting方法中。

尾聲

至此,整個多租戶的架構就基本完成了。當然我們還可以進行擴展,比如實現租戶緩存、租戶資源管理等等,這是後續的話題了。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 抽象工廠模式(Abstract Factory) 類圖 描述 抽象工廠: 多個抽象產品類,每個抽象產品類可以派生出多個具體產品類; 一個抽象工廠類,可以派生出多個具體工廠類; 每個具體工廠可以創建多個具體產品,即每個工廠可以生產一個產品集合。 應用場景 就拿生產轎車來說,轎車是由發動機、車輪、車體結 ...
  • 工廠方法模式(Factory Method) 類圖 描述 工廠方法: 一個抽象產品類,可以派生多個具體產品類; 一個抽象工廠類,可以派生多個具體工廠類; 每個具體工廠只能創建一個具體產品。 應用場景 汽車介面 汽車類 汽車工廠介面 汽車工廠類 調用 ...
  • 簡單工廠模式(Simple Factory) 類圖 描述 簡單工廠: 一個抽象產品類,可以派生多個具體產品類; 一個具體工廠類; 工廠只能創建一個具體產品。 應用場景 汽車介面 汽車類 汽車工廠類 調用,從配置文件中讀取操作符 ...
  • 無論做什麼事情呢,都要善始善終呢。前邊連續發表了5篇關於重構的博客,其中分門別類的介紹了一些重構手法。今天的這篇博客就使用一個完整的示例來總結一下之前的重構規則,也算給之前的關於重構的博客畫一個句號。今天的示例借鑒於《重構,改善既有代碼的設計》這本書中的第一章的示例,在其基礎上做了一些修改。今天博客 ...
  • 對第三方介面的調用我們需要對GET和POST進行監控,看一些請求的執行是否成功,如A調用B,B調用C,C調用D,這一連串的東西需要我們使用cat進行記錄,進行記錄之後,我們可以很容易的發現請求響應的時間及是否出錯,下麵是我對這兩種請求的封裝。 在程式中使用非常方便,如下代碼,一看便知 而它產生的消息 ...
  • 本系統旨在提供業務監控實時數據和歷史數據以及報表、閾值報警、同比增長分析等一體化的歷史業務數據解決方案。 技術選型 sdk部門有C#版和java版,api和website採用golang語音開發,資料庫採用mysql,數據傳輸採用socket+http 架構設計 本系統主要分3個部分:即sdk(@2 ...
  • 8. 告別Lock 不是一直說Lock比較麻煩危險嗎,那就不要好了。其實有一個Lock free的方法。 首先引入一個概念——原子變數。在這種變數上的操作是原子操作(atomic operation)。原子操作就是說這個操作要麼都完成,要麼都不完成,部分完成是不行的。就像物理化學中的原子一樣,借用不 ...
  • 一、現狀說明: 就在這金三銀四的求職黃金時期,我有幸作為公司的獨立技術面試官,擁有最終決定錄用權,在倍受上級領導的充分信任下,我也向上級保證,一定要為公司找到合適的人才,就在我滿懷信心的情況下麵試了一個又一個的求職者,發現了大多數求職者共同的問題,一是:眼高手低,即工作年限雖長,但受工作內容及個人原 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...