第9章 使用客戶端憑據保護API

来源:https://www.cnblogs.com/thinksjay/archive/2019/04/26/10774718.html
-Advertisement-
Play Games

快速入門介紹了使用IdentityServer保護API的最基本方案。 我們將定義一個API和一個想要訪問它的客戶端。 客戶端將通過提供 在IdentityServer請求訪問令牌, 充當客戶端和IdentityServer都知道的秘密,並且它將使用該令牌來訪問API。 9.1設置ASP.NET核心 ...


快速入門介紹了使用IdentityServer保護API的最基本方案。 我們將定義一個API和一個想要訪問它的客戶端。 客戶端將通過提供ClientCredentials在IdentityServer請求訪問令牌,ClientCredentials充當客戶端和IdentityServer都知道的秘密,並且它將使用該令牌來訪問API。

9.1設置ASP.NET核心應用程式

首先為應用程式創建一個目錄 - 然後使用我們的模板創建一個包含基本IdentityServer設置的ASP.NET Core應用程式,例如:

md quickstart
cd quickstart

md src
cd src

dotnet new is4empty -n IdentityServer

這將創建以下文件:

  • IdentityServer.csproj- 項目文件和Properties\launchSettings.json文件
  • Program.csStartup.cs- 主要的應用程式入口點
  • Config.cs - IdentityServer資源和客戶端配置文件

您現在可以使用自己喜歡的文本編輯器來編輯或查看文件。如果您希望獲得Visual Studio支持,可以添加如下解決方案文件:

cd ..
dotnet new sln -n Quickstart

然後讓它添加你的IdentityServer項目(記住這個命令,因為我們將在下麵創建其他項目):

dotnet sln add .\src\IdentityServer\IdentityServer.csproj

註意
此模板中使用的協議是http,當在Kestrel上運行時,埠設置為5000IISExpress上的隨機埠。您可以在Properties\launchSettings.json文件中更改它。但是,所有快速入門指令都假定您使用Kestrel上的預設埠以及http協議,該協議足以進行本地開發。

9.2 定義API資源

API是您要保護的系統中的資源。

資源定義可以通過多種方式載入,模板使用“代碼作為配置”appproach。在Config.cs文件中,您可以找到一個名為GetApisAPI 的方法,如下所示:

public static IEnumerable<ApiResource> GetApis()
{
    return new List<ApiResource>
    {
        new ApiResource("api1", "My API")
    };
}

9.3 定義客戶端

下一步是定義可以訪問此API的客戶端。

對於此方案,客戶端將不具有互動式用戶,並將使用IdentityServer的所謂客戶端密鑰進行身份驗證。將以下代碼添加到Config.cs文件中:

public static IEnumerable<Client> GetClients()
{
    return new List<Client>
    {
        new Client
        {
            ClientId = "client",

            // no interactive user, use the clientid/secret for authentication
            AllowedGrantTypes = GrantTypes.ClientCredentials,

            // secret for authentication
            ClientSecrets =
            {
                new Secret("secret".Sha256())
            },

            // scopes that client has access to
            AllowedScopes = { "api1" }
        }
    };
}

9.4 配置IdentityServer

Startup.cs載入資源和客戶端定義 - 模板已經為您執行此操作:

public void ConfigureServices(IServiceCollection services)
{
    var builder = services.AddIdentityServer()
        .AddInMemoryIdentityResources(Config.GetIdentityResources())
        .AddInMemoryApiResources(Config.GetApis())
        .AddInMemoryClients(Config.GetClients());

    // rest omitted
}

就是這樣 - 如果您運行伺服器並瀏覽瀏覽器 http://localhost:5000/.well-known/openid-configuration,您應該會看到所謂的發現文檔。客戶端和API將使用它來下載必要的配置數據。


首次啟動時,IdentityServer將為您創建一個開發人員簽名密鑰,它是一個名為的文件tempkey.rsa。您不必將該文件檢入源代碼管理中,如果該文件不存在,將重新創建該文件。

9.5 添加API

接下來,為您的解決方案添加API。

您可以使用Visual Studio中的ASP.NET Core Web API(或空)模板,也可以使用.NET CLI來創建API項目。從src文件夾中運行以下命令:

dotnet new web -n Api

然後通過運行以下命令將其添加到解決方案中:

cd ..
dotnet sln add .\src\Api\Api.csproj

將API應用程式配置為http://localhost:5001僅運行。您可以通過編輯Properties文件夾中的launchSettings.json文件來完成此操作。將應用程式URL設置更改為:

"applicationUrl": "http://localhost:5001"

9.6 控制器

在API項目中添加一個新文件夾Controllers和一個新控制器IdentityController

[Route("identity")]
[Authorize]
public class IdentityController : ControllerBase
{
    [HttpGet]
    public IActionResult Get()
    {
        return new JsonResult(from c in User.Claims select new { c.Type, c.Value });
    }
}

這個控制器將在後面被用於測試授權需求,同時通過API的眼睛(瀏覽工具)來可視化身份信息。

9.7 配置

最後一步是將身份驗證服務添加到DI和身份驗證中間件到管道。這些將:

  • 驗證輸入的令牌以確保它來自可信任的發佈者(IdentityServer)
  • 驗證令牌是否可用於該 api(也就是 Scope)。

Startup更新為如下所示:

public class Startup
{
    public void ConfigureServices(IServiceCollection services)
    {
        services.AddMvcCore()
            .AddAuthorization()
            .AddJsonFormatters();

        services.AddAuthentication("Bearer")
            .AddJwtBearer("Bearer", options =>
            {
                options.Authority = "http://localhost:5000";
                options.RequireHttpsMetadata = false;

                options.Audience = "api1";
            });
    }

    public void Configure(IApplicationBuilder app)
    {
        app.UseAuthentication();

        app.UseMvc();
    }
}

AddAuthentication將身份驗證服務添加到DI並配置"Bearer"為預設方案。 UseAuthentication將身份驗證中間件添加到管道中,以便在每次調用主機時自動執行身份驗證。

http://localhost:5001/identity在瀏覽器上導航到控制器應返回401狀態代碼。這意味著您的API需要憑證,現在受IdentityServer保護。

9.8 創建客戶端

最後一步是編寫請求訪問令牌的客戶端,然後使用此令牌訪問API。為此,在您的解決方案中添加一個控制台項目,請記住在以下位置創建它src:

dotnet new console -n Client

然後和以前一樣,使用以下方法將其添加到您的解

cd ..
dotnet sln add .\src\Client\Client.csproj

打開Program.cs並將內容從這裡複製到它。

客戶端程式非同步調用Main方法以運行非同步http調用。 從C#7.1開始,此功能可用,一旦您編輯Client.csproj以將以下行添加為PropertyGroup,它就可用:

<LangVersion>latest</LangVersion>

IdentityServer的令牌端點實現OAuth 2.0協議,您可以使用原始HTTP來訪問它。但是,我們有一個名為IdentityModel的客戶端庫,它將協議交互封裝在易於使用的API中。

IdentityModel NuGet包添加到您的客戶端。這可以通過Visual Studio的nuget對話框,手動添加到Client.csproj文件,或使用CLI來完成:

dotnet add package IdentityModel

IdentityModel包括用於發現端點的客戶端庫。這樣您只需要知道IdentityServer的基地址 - 可以從元數據中讀取實際的端點地址:

// discover endpoints from metadata
var client = new HttpClient();
var disco = await client.GetDiscoveryDocumentAsync("http://localhost:5000");
if (disco.IsError)
{
    Console.WriteLine(disco.Error);
    return;
}

接下來,您可以使用發現文檔中的信息向IdentityServer請求令牌以訪問api1:

// request token
var tokenResponse = await client.RequestClientCredentialsTokenAsync(new ClientCredentialsTokenRequest
{
    Address = disco.TokenEndpoint,

    ClientId = "client",
    ClientSecret = "secret",
    Scope = "api1"
});

if (tokenResponse.IsError)
{
    Console.WriteLine(tokenResponse.Error);
    return;
}

Console.WriteLine(tokenResponse.Json);

註意
將訪問令牌從控制台複製並粘貼到https://jwt.io/以檢查原始令牌。

9.9 調用

要將訪問令牌發送到API,通常使用HTTP Authorization標頭。這是使用SetBearerToken擴展方法完成的:

// call api
var client = new HttpClient();
client.SetBearerToken(tokenResponse.AccessToken);

var response = await client.GetAsync("http://localhost:5001/identity");
if (!response.IsSuccessStatusCode)
{
    Console.WriteLine(response.StatusCode);
}
else
{
    var content = await response.Content.ReadAsStringAsync();
    Console.WriteLine(JArray.Parse(content));
}

輸出應如下所示:

註意
預設情況下,訪問令牌將包含有關範圍(scope),生命周期(nbf和exp),客戶端ID(client_id)和頒發者名稱(iss)的聲明。

9.10 進一步的實驗

本演練重點介紹了迄今為止的成功之路

  • 客戶端能夠請求令牌
  • 客戶端可以使用令牌來訪問API

你現在可以嘗試引發一些錯誤來學習系統的相關行為,比如:

  • 嘗試在未運行時連接到IdentityServer(不可用)
  • 嘗試使用無效的客戶端ID或密碼來請求令牌
  • 嘗試在令牌請求期間請求無效範圍
  • 嘗試在API未運行時調用API(不可用)
  • 不要將令牌發送到API
  • 將API配置為需要與令牌中的範圍不同的範圍

github地址


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

-Advertisement-
Play Games
更多相關文章
  • 因為官網asp.net core webapi教程部分,給出的是使用記憶體中的數據即 UseInMemoryDatabase 的方式, 這裡記錄一下,使用SQL Server資料庫的方式即 UseSqlServer 的方式。 環境說明: 這裡使用的是win 7 下的 virtual studio 20 ...
  • ASP.NET Core在Azure Kubernetes Service中的部署和管理 [TOC] 目標 部署:掌握將aspnetcore程式成功發佈到Azure Kubernetes Service(AKS)上 管理:掌握將AKS上的aspnetcore程式擴容、更新版本 準備工作 註冊 Azu ...
  • cookie屬性: name欄位為一個cookie的名稱。 value欄位為一個cookie的值。 domain欄位為可以訪問此cookie的功能變數名稱。 path欄位為可以訪問此cookie的頁面路徑。 比如domain是abc.com,path是/test,那麼只有/test路徑下的頁面可以讀取此co ...
  • 註銷IdentityServer就像刪除身份驗證cookie一樣簡單,但是為了完成聯合註銷,我們必須考慮將用戶從客戶端應用程式(甚至可能是上游身份提供者)中簽名。 24.1 刪除認證 要刪除身份驗證cookie,只需使用 擴展方法 即可。您需要傳遞使用的方案( 除非您已更改,否則提供此方案): 或者 ...
  • demo地址: "BulkAll" 批量導入 實現目標:想要使用ElasticSearch的 .Net Api客戶端NEST批量導入數據,併發非同步高效的批量導入 NEST提供了BulkAll 不廢話,上代碼 如果想要對處理導入過程進行監控可以這麼替換 還可以使用C 的local function特性 ...
  • 因為需要把好多的文檔放在伺服器上訪問,最開始是使用第3方網站www.aconvert.com,這個其實也挺方便的, 最後由於一些需求原因,最終只有依靠代碼來進行,以下是簡化後的代碼 創建一個控制台應用 在Nuget中引用Aspose.Words(這邊是Core 2.2,引用的最新版本) 具體代碼: ...
  • Docker 發佈 Abp net core web 服務 準備工作:Abp 項目,這個是模板下載地址 https://aspnetboilerplate.com/Templates (本例使用的是Single Page Web Application) 安裝docker :參考:http://ww ...
  • IdentityServer旨在實現可擴展性,其中一個可擴展點是用於IdentityServer所需數據的存儲機制。本快速入門展示瞭如何配置IdentityServer以使用EntityFramework Core(EF)作為此數據的存儲機制(而不是使用我們迄今為止使用的記憶體中實現)。 註意 除了手 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...