.net core使用ocelot---第二篇 身份驗證

来源:https://www.cnblogs.com/xlxr45/archive/2019/08/08/11321134.html
-Advertisement-
Play Games

簡介原文鏈接 .net core使用ocelot 第一篇 簡單使用 接上文,我將繼續介紹使用asp.net core 創建API網關,主要介紹身份驗證(authentication )相關內容。 API服務是需要保護的資源,我們應該儘可能的保證他們的安全。 通常,我們會使用aspnet securi ...


簡介原文鏈接

      .net core使用ocelot---第一篇 簡單使用

         接上文,我將繼續介紹使用asp.net core 創建API網關,主要介紹身份驗證(authentication )相關內容。

         API服務是需要保護的資源,我們應該儘可能的保證他們的安全。

         通常,我們會使用aspnet security 來保證我們項目的安全,aspnet security代碼庫是為asp.net core 設計的安全和授權中間件。已經讓事情變得簡單。

         那麼我們在項目中引入API網關會不會導致巨大的改變?答案是,不會。我們的修改微乎其微,但是卻讓安全和授權變得簡單。

         先看下麵的截圖。

  截圖顯示,當我們訪問我們的服務,API網關會讓我們首先訪問其授權模塊。我們必須訪問授權服務獲得訪問令牌(access token),然後用訪問令牌訪問受保護的服務。

         可能你傾向將授權服務獨立,這意味客戶端得先訪問授權服務獲得訪問令牌。然後攜帶令牌訪問API網關。毫無疑問這樣做沒問題,但是我建議將授權服務和其他服務放在一起。

         APIGateway是我們所有服務的入口,對於身份未驗證的客戶端僅可以訪問授權服務。

         OK,開始我們的表演。

         我會使用上一個demo的部分內容,便於理解。

Step1

項目名稱

項目類型

描述

APIGateway

ASP.NET Core Empty

示例的入口

CustomersAPIServices

ASP.NET Core Web API

API 服務處理消費者的操作

AuthServer

ASP.NET Core Web API

API 服務處理授權操作

ClientApp

Console App

控制台程式模擬客戶端

  APIGateway和CustomerAPIServices和上篇文章的例子一樣,你可以在APIGatewayBasicDemo獲得。

Step2

         創建AuthServer,AuthServer主要是為request請求生成訪問令牌(access token),我們需要添加一個方法處理。

[HttpGet]  
public IActionResult Get(string name, string pwd)  
{  
    //just hard code here.  
    if (name == "catcher" && pwd == "123")  
    {  
        var now = DateTime.UtcNow;  
  
        var claims = new Claim[]  
        {  
            new Claim(JwtRegisteredClaimNames.Sub, name),  
            new Claim(JwtRegisteredClaimNames.Jti, Guid.NewGuid().ToString()),  
            new Claim(JwtRegisteredClaimNames.Iat, now.ToUniversalTime().ToString(), ClaimValueTypes.Integer64)  
        };  
  
        var signingKey = new SymmetricSecurityKey(Encoding.ASCII.GetBytes(_settings.Value.Secret));  
        var tokenValidationParameters = new TokenValidationParameters  
        {  
            ValidateIssuerSigningKey = true,  
            IssuerSigningKey = signingKey,  
            ValidateIssuer = true,  
            ValidIssuer = _settings.Value.Iss,  
            ValidateAudience = true,  
            ValidAudience = _settings.Value.Aud,  
            ValidateLifetime = true,  
            ClockSkew = TimeSpan.Zero,  
            RequireExpirationTime = true,  
  
        };  
  
        var jwt = new JwtSecurityToken(  
            issuer: _settings.Value.Iss,  
            audience: _settings.Value.Aud,  
            claims: claims,  
            notBefore: now,  
            expires: now.Add(TimeSpan.FromMinutes(2)),  
            signingCredentials: new SigningCredentials(signingKey, SecurityAlgorithms.HmacSha256)  
        );  
        var encodedJwt = new JwtSecurityTokenHandler().WriteToken(jwt);  
        var responseJson = new  
        {  
            access_token = encodedJwt,  
            expires_in = (int)TimeSpan.FromMinutes(2).TotalSeconds  
        };  
  
        return Json(responseJson);  
    }  
    else  
    {  
        return Json("");  
    }  
}  

  在驗證用戶時。我使用硬編碼將用戶名寫死,因為對於本文這個不是那麼重要。

  這樣我們就完成了授權服務,現在跑起來。

Step3

         回到CustomersAPIServices項目,我們應該保護這個服務。

         修改Startup,我們可以使用授權。在這我用JwtBearer進行授權,我會給TestKey設置預設的授權方案。TestKey我將在下麵提到。

public void ConfigureServices(IServiceCollection services)  
{  
    var audienceConfig = Configuration.GetSection("Audience");  
  
    var signingKey = new SymmetricSecurityKey(Encoding.ASCII.GetBytes(audienceConfig["Secret"]));  
    var tokenValidationParameters = new TokenValidationParameters  
    {  
        ValidateIssuerSigningKey = true,  
        IssuerSigningKey = signingKey,  
        ValidateIssuer = true,  
        ValidIssuer = audienceConfig["Iss"],  
        ValidateAudience = true,  
        ValidAudience = audienceConfig["Aud"],  
        ValidateLifetime = true,  
        ClockSkew = TimeSpan.Zero,  
        RequireExpirationTime = true,  
    };  
  
    services.AddAuthentication()  
            .AddJwtBearer("TestKey", x =>  
             {  
                 x.RequireHttpsMetadata = false;  
                 x.TokenValidationParameters = tokenValidationParameters;  
             });  
  
    services.AddMvc();  
}  
public void Configure(IApplicationBuilder app)  
{              
    app.UseAuthentication();  
    app.UseMvc();  
} 

  接下來,對需要保護的方法,添加Authorize。

[Authorize]  
[HttpGet]  
public IEnumerable<string> Get()  
{  
    return new string[] { "Catcher Wong", "James Li" };  
}  

註意

         該項目基於asp.net core 2.0. 如果你的項目是1.X,可能有些不同,建議用遷移遷移到2.0.以上。接下來就是見證奇跡的時候了。

Step4

         最重要的步驟來了,在APIGateway中配置授權。

public void ConfigureServices(IServiceCollection services)  
{  
    var audienceConfig = Configuration.GetSection("Audience");  
  
    var signingKey = new SymmetricSecurityKey(Encoding.ASCII.GetBytes(audienceConfig["Secret"]));  
    var tokenValidationParameters = new TokenValidationParameters  
    {  
        ValidateIssuerSigningKey = true,  
        IssuerSigningKey = signingKey,  
        ValidateIssuer = true,  
        ValidIssuer = audienceConfig["Iss"],  
        ValidateAudience = true,  
        ValidAudience = audienceConfig["Aud"],  
        ValidateLifetime = true,  
        ClockSkew = TimeSpan.Zero,  
        RequireExpirationTime = true,  
    };  
  
    services.AddAuthentication()  
            .AddJwtBearer("TestKey", x =>  
             {  
                 x.RequireHttpsMetadata = false;  
                 x.TokenValidationParameters = tokenValidationParameters;  
             });  
  
    services.AddOcelot(Configuration);  
}  

 

  這個配置的大部分代碼和Customer Service一樣。

  當Ocelot啟動,它會查看ReRoutes 》AuthenticationOptions 》AuthenticationProviderKey 的值,

檢查該值是否被授權服務註冊,如果沒有,Ocelot不會啟動,如果有,Ocelot在執行時使用授權服務。

         所以,我們得修改configuration.json,我們需要添加新的節點,並將其值賦為在Startup 類中定義的一樣。

{  
    "DownstreamPathTemplate": "/api/customers",  
    "DownstreamScheme": "http",  
    "DownstreamHost": "localhost",  
    "DownstreamPort": 9001,  
    "UpstreamPathTemplate": "/customers",  
    "UpstreamHttpMethod": [ "Get" ],  
    "AuthenticationOptions": {  
        "AuthenticationProviderKey": "TestKey",  
        "AllowedScopes": []  
    }  
}  

  啟動服務。

 

註意

         當你啟動項目時遇到下麵的錯誤,你應該檢查你的代碼,查看AddJwtBearer 方法是否指明授權方案。

        Unhandled Exception: System.InvalidOperationException: Scheme already exists: Bearer

Step5

         我們已經準備完畢,我們用我們的客戶端模擬APIGateway的某些請求。

 

         我們先添加獲得訪問令牌的方法。

private static string GetJwt()  
{  
    HttpClient client = new HttpClient();  
    client.BaseAddress = new Uri( "http://localhost:9000");  
    client.DefaultRequestHeaders.Clear();  
    var res2 = client.GetAsync("/api/auth?name=catcher&pwd=123").Result;  
    dynamic jwt = JsonConvert.DeserializeObject(res2.Content.ReadAsStringAsync().Result);  
    return jwt.access_token;  
} 

 

 接下來,編寫通過APIGateway訪問Customer Service方法的代碼。

static void Main(string[] args)  
{  
    HttpClient client = new HttpClient();  
      
    client.DefaultRequestHeaders.Clear();  
    client.BaseAddress = new Uri("http://localhost:9000");  
      
    // 1. without access_token will not access the service  
    //    and return 401 .  
    var resWithoutToken = client.GetAsync("/customers").Result;  
      
    //print something here   
      
    //2. with access_token will access the service  
    //   and return result.  
    client.DefaultRequestHeaders.Clear();  
    var jwt = GetJwt();  
      
    client.DefaultRequestHeaders.Add("Authorization", $"Bearer {jwt}");  
    var resWithToken = client.GetAsync("/customers").Result;  
      
    //print something here   
      
    //3. visit no auth service   
    client.DefaultRequestHeaders.Clear();  
    var res = client.GetAsync("/customers/1").Result;  
      
    //print something here   
      
    Console.Read();  
}  

 

運行結果。

 

  完工。

  源碼在此

總結

         沒啥。

 


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

-Advertisement-
Play Games
更多相關文章
  • 簡介 .net core使用ocelot 第一篇 簡單使用 .net core使用ocelot 第二篇 身份驗證使用 .net core使用ocelot 第三篇 日誌記錄 .net core使用ocelot 第四篇 限流熔斷 .net core使用ocelot 第五篇 服務質量 .net core使 ...
  • 代碼有時跨線程訪問UI,修改按鈕Enable屬性不報異常。調試發現修改按鈕屬性的線程是Background,執行不報異常。 在窗體構造中添加 Control.CheckForIllegalCrossThreadCalls = true; 後,調試執行報跨線程訪問異常。 暫未探究原因。 編寫代碼時,還 ...
  • 本文只配置了簡單文件存儲 1.添加nuget包 2.添加日誌配置文件nlog.config 這裡配置了三個target區分不同的日誌,具體配置需要自己研究,推薦鏈接https://www.cnblogs.com/TianFang/p/4003749.html 3.配置Startup 4.日誌使用 5 ...
  • 一、註冊公眾平臺 1.入口 瀏覽器搜索“微信公眾平臺”,進入官網,點右上角立即註冊。 2.選擇賬號類型 註冊前需要選擇一個賬號類型,共有4個賬號類型可以選擇,每種類型能提供不同的功能,功能區別見下圖。選擇一個適合的類型開始註冊(樓主大學期間寫微信文案已註冊了一個訂閱號,前期自學小程式,又註冊了一個小 ...
  • 簡介 .net core使用ocelot 第一篇 簡單使用 .net core使用ocelot 第二篇 身份驗證使用 .net core使用ocelot 第三篇 日誌記錄 .net core使用ocelot 第四篇 限流熔斷 .net core使用ocelot 第五篇 服務質量 .net core使 ...
  • 簡介 .net core使用ocelot 第一篇 簡單使用 .net core使用ocelot 第二篇 身份驗證 .net core使用ocelot 第三篇 日誌記錄 .net core使用ocelot 第四篇 限流熔斷 .net core使用ocelot 第五篇 服務質量 本文介紹Ocelot的負 ...
  • 簡介 .net core使用ocelot 第一篇 簡單使用 .net core使用ocelot 第二篇 身份驗證使用 .net core使用ocelot 第三篇 日誌記錄 .net core使用ocelot 第四篇 限流熔斷 本文介紹Ocelot的Qos(服務質量)模塊。 什麼是QoS 服務質量是指 ...
  • 簡介 .net core使用ocelot 第一篇 簡單使用 .net core使用ocelot 第二篇 身份驗證 .net core使用ocelot 第三篇 日誌記錄 前幾篇文章我們陸續介紹如何用ASP.NET Core 創建API網關。 這次我們討論Ocelot的流量限制模塊。 什麼是流量控制(R ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...