ServiceStack 多租戶的實現方案

来源:https://www.cnblogs.com/joyswings/archive/2019/04/02/10644868.html
-Advertisement-
Play Games

以SqlServer為例子說明ServiceStack實現多租戶,在SqlServer中創建4個Database:TMaster、T1,T2,T3,為了安全起見 每個Database不用sa賬號,而是用獨立的資料庫的賬號和密碼,為了方便演示這密碼設置成一樣 租戶TMaster Database:TM ...


以SqlServer為例子說明ServiceStack實現多租戶,在SqlServer中創建4個Database:TMaster、T1,T2,T3,為了安全起見

每個Database不用sa賬號,而是用獨立的資料庫的賬號和密碼,為了方便演示這密碼設置成一樣

租戶TMaster Database:TMaster  賬號密碼: User Id=TMaster;Password=t123

租戶T1 Database:T1  賬號密碼: User Id=T1;Password=t123

租戶T2 Database:T2  賬號密碼: User Id=T2;Password=t123

租戶T3 Database:T3  賬號密碼: User Id=T3;Password=t123

創建資料庫的方法可以參見文章:  https://www.cnblogs.com/tonge/p/3791029.html

每個登陸用自己的賬號和密碼登陸,其它的資料庫是沒有訪問許可權的,這個各個租戶是完全隔離的。

假設Node和npm已經安裝

npm install -g @servicestack/cli

執行命令dotnet-new selfhost SSHost

這樣就創建了ServiceStack的控制台程式,用VS2017解決方案,在ServiceModel的Types文件夾添加TenantConfig類文件

代碼如下:

using System;
using System.Collections.Generic;
using System.Text;

namespace ssTest.ServiceModel.Types
{
    public interface IForTenant
    {
        string TenantId { get; }
    }

    public class TenantConfig
    {
        public string Id { get; set; }

        public string Company { get; set; }
    }
}

 

修改Hello.cs文件,代碼如下:

using ServiceStack;
using ssTest.ServiceModel.Types;
using System;

namespace ssTest.ServiceModel
{
    [Route("/hello")]
    [Route("/hello/{Name}")]
    public class Hello : IForTenant, IReturn<HelloResponse>
    {

        public string Name { get; set; }

        // 實現介面IForTenant(租戶標識Id)
        public string TenantId { get; set; }
}

    public class HelloResponse
    {
        public string Result { get; set; }

        public DateTime Date { get; set; }

        // 返回租戶公司信息
        public TenantConfig Config { get; set; } 
    }
}

 

主程式的Startup代碼如下

    public class Startup
    {
        public void ConfigureServices(IServiceCollection services)
        {
        }

        public void Configure(IApplicationBuilder app, IHostingEnvironment env)
        {
            // JsConfig.DateHandler = DateHandler.ISO8601;
            // 保證時間類型的欄位可以解析成js識別的時間類型
            JsConfig<DateTime>.SerializeFn = time => new DateTime(time.Ticks, DateTimeKind.Local).ToString("o");
            JsConfig<DateTime?>.SerializeFn =
                time => time != null ? new DateTime(time.Value.Ticks, DateTimeKind.Local).ToString("o") : null;
            JsConfig.DateHandler = DateHandler.ISO8601;

            app.UseServiceStack(new AppHost());

            app.Run(context =>
            {
                context.Response.Redirect("/metadata");
                return Task.FromResult(0);
            });
        }
    }

 

下麵就到核心代碼了,在主程式中建立多租戶Db工程類,讓程式可以自動的根據租戶Id訪問自己的資料庫

        public class MultiTenantDbFactory : IDbConnectionFactory
        {
            private readonly IDbConnectionFactory dbFactory;

            public MultiTenantDbFactory(IDbConnectionFactory dbFactory)
            {
                this.dbFactory = dbFactory;
            }

            public IDbConnection OpenDbConnection()
            {
                var tenantId = RequestContext.Instance.Items["TenantId"] as string;
                return OpenTenant(tenantId);
            }

            public IDbConnection OpenTenant(string tenantId = null)
            {
                return tenantId != null
                    ? dbFactory.OpenDbConnectionString($"Data Source=.; Initial Catalog={tenantId};User Id={tenantId};Password=t123;pooling=true;")
                    : dbFactory.OpenDbConnection();
            }

            public IDbConnection CreateDbConnection()
            {
                return dbFactory.CreateDbConnection();
            }
        }

 

AppHost中加入如下代碼,GlobalRequestFilters的作用,根據傳入的租戶Id來選擇相應的資料庫,如果租戶Id為null,系統自動使用TMaster資料庫

InitDb的作用就是初始化三個資料庫,創建表TenantConfig並插入一條記錄。

        public override void Configure(Container container)
        {
            ConigureSqlserver(container);
        }

        private void ConigureSqlserver(Container container)
        {
            var dbFactory = new OrmLiteConnectionFactory(
                "Data Source=.; Initial Catalog=TMaster;User Id=TMaster;Password=t123;pooling=true;", SqlServerDialect.Provider);

            const int noOfTennants = 3;

            container.Register<IDbConnectionFactory>(c =>new MultiTenantDbFactory(dbFactory));

            var multiDbFactory = (MultiTenantDbFactory)container.Resolve<IDbConnectionFactory>();

            using (var db = multiDbFactory.OpenTenant())
                InitDb(db, "TMaster", "Masters inc.");

            for(int i=1; i<= noOfTennants; i++)
            {
                var tenantId = $"T{i}";

                using (var db = multiDbFactory.OpenTenant(tenantId))
                    InitDb(db, tenantId,  $"ACME {tenantId} inc.");
            }

            GlobalRequestFilters.Add((req, res, dto) =>
            {
                var forTennant = dto as IForTenant;
                if (forTennant != null)
                    RequestContext.Instance.Items.Add("TenantId", forTennant.TenantId);
            });
        }
        public void InitDb(IDbConnection db, string tenantId, string company)
        {
            db.DropAndCreateTable<TenantConfig>();
            db.Insert(new TenantConfig { Id = tenantId, Company = company });
        }

這樣核心代碼就完成了,我們用postman調用試試看,是不是達到了預期的效果

body為空,租戶Id沒有設置,系統認為是預設的資料庫TMaster,返回的是Master資料庫中的config表信息

body中設置json參數{"name":"joy", "tenantId":"t1"},可以看到查詢返回的是資料庫T1的信息

 

我們再試驗一下T2,body中設置json參數{"name":"peter", "tenantId":"t2"}

可以很驚喜的看到,查詢的是資料庫T2的信息

ServiceStack解決方案真是強大,本來一個複雜的多租戶問題就這樣輕易解決了,是不是很簡單。這裡例子用的都是sqlserver資料庫,實際上每個租戶可以使用不同的資料庫。

最近一年很流行ABP解決方案,我想說的是ServiceStack解決方案也很優秀,甚至更加優秀,當你瞭解越多你就會驚嘆當初作者的設計思路是多麼的優秀,有興趣的小伙伴可以一起挖掘和分享啊!

 


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

-Advertisement-
Play Games
更多相關文章
  • 1、JAVA中,char占2位元組,16位。可在存放漢字 2、char賦值 char a='a'; //任意單個字元,加單引號。 char a='中';//任意單個中文字,加單引號。 char a=111;//整數。0~65535。十進位、八進位、十六進位均可。輸出字元編碼表中對應的字元。 註:只能放 ...
  • 一,java介面 二,介面與類的區別 三,介面特性 不能使用new運算符實例化一個介面 //錯誤的 x = new Comparable(.....); //Comparable是一個介面 介面內能聲明變數 可以使用instanceof檢查一個對象是否屬於某個特定類 介面可擴展 介面不能包含實例域或 ...
  • 登錄事先準備好的 Jenkins 1 新建任務 2 源碼管理 git 輸入正確地址 3 構建環境:Delete workspace before build startsAbort the build if it's stuck 打勾 4 安裝事先準備好的 pm2打開shell安裝參考 https: ...
  • TomatoLog 對日誌的處理不敢說強大,但可能是 .NETCore 平臺上最簡單易用的日誌集成組件,具有高度靈活的使用方式,完全可定義配置的可擴展性,使用非同步寫入,業務完全解耦,客戶端的一鍵安裝、一行代碼上傳日誌 ...
  • Visual Studio 2019 的正式版現在可以[下載](https://visualstudio.microsoft.com/downloads/)了。藉助 Visual Studio 2019, 您和您的團隊將在構建當前和未來項目時變得更加高效, 因為您可以從 IDE 的創新中獲益, 從而... ...
  • 原文地址:http://www.entityframeworktutorial.net/code-first/dataannotation-in-code-first.aspx 數據註解特性是.NET特性,可以在EF或者EF Core中,應用於實體類上或者屬性上,以重寫預設的約定規則。 在EF 6和 ...
  • 上一篇文章我帶著大家體驗了一把《 "ASP.NET Core 3.0 上的gRPC服務模板初體驗(多圖)" 》,如果有興趣的可以點擊鏈接進行查看,相信跟著做的你,也是可以跑起來的。這篇文章我們將一起來探討下gRPC服務如何與HTTP APIs進行比較。用於為應用程式提供API的技術是一個重要的選擇, ...
  • 原文地址:http://www.entityframeworktutorial.net/code-first/configure-classes-in-code-first.aspx 在前面的章節中,我們學習了Code-First預設的約定。Code-First使用預設的約定,從你的領域類中生成概念 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...