0

0

C# .NET中基于JWT和外部授权服务器的REST API安全配置指南

霞舞

霞舞

发布时间:2025-08-31 12:56:36

|

799人浏览过

|

来源于php中文网

原创

C# .NET中基于JWT和外部授权服务器的REST API安全配置指南

本文旨在提供一个简洁明了的教程,指导开发者如何在C# .NET应用中,以纯资源服务器模式,通过外部授权服务器(如AWS Cognito或Asgardeo)实现REST API的安全保护。文章将重点介绍如何利用JWT Bearer认证机制,通过最小化配置,快速构建一个能够验证传入访问令牌的API服务,无需深度集成用户管理功能。

核心概念:JWT Bearer认证与资源服务器

在现代微服务架构中,api安全通常通过oauth 2.0和openid connect协议实现,其中json web token (jwt) 作为访问令牌的标准格式。资源服务器(resource server)是托管受保护资源的api服务,其主要职责是验证传入请求中的访问令牌,并根据令牌的有效性和权限信息决定是否授予访问。

对于从Java Spring Boot等其他平台迁移到C# .NET的开发者而言,如何以简洁高效的方式实现REST API的安全保护,尤其是作为纯资源服务器的角色,常常是一个挑战。许多教程倾向于将用户管理与API安全紧密结合,但这并非总是纯资源服务器所需的场景。本教程将展示如何在C# .NET中,通过最小化配置,实现一个专注于令牌验证的资源服务器。

配置步骤

在C# .NET(本示例基于.NET 6.0)中实现JWT Bearer认证的资源服务器,主要涉及以下三个步骤:

1. 添加认证服务

在应用程序的Program.cs文件中,需要配置认证服务,指定使用JWT Bearer认证方案。这包括设置默认的认证和挑战方案,并配置JWT Bearer选项,如受众(Audience)和OpenID Connect元数据地址。

using Microsoft.AspNetCore.Authentication.JwtBearer;
using Microsoft.IdentityModel.Tokens;

var builder = WebApplication.CreateBuilder(args);

// 添加认证服务
builder.Services.AddAuthentication(options =>
{
    // 设置默认的认证方案为JwtBearer
    options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;
    // 设置默认的挑战方案为JwtBearer
    options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme;
}).AddJwtBearer(options =>
{
    // 配置JWT Bearer选项
    // Audience (受众) 是验证令牌时检查的声明之一,通常是客户端ID或资源服务器的标识符。
    // 它确保令牌是为预期接收者颁发的。
    options.Audience = ""; 

    // MetadataAddress (元数据地址) 指向外部授权服务器的OpenID Connect配置发现文档。
    // .NET认证中间件将从该地址自动获取公钥(JWKS URI)、颁发者(Issuer)等信息,
    // 用于验证JWT令牌的签名和有效性。
    options.MetadataAddress = "https://api.asgardeo.io/t//oauth2/token/.well-known/openid-configuration";

    // 如果需要更精细的控制,例如自定义令牌验证参数,可以在这里进一步配置TokenValidationParameters。
    // options.TokenValidationParameters = new TokenValidationParameters
    // {
    //     ValidateIssuer = true,
    //     ValidIssuer = "https://api.asgardeo.io/t//oauth2/token",
    //     ValidateAudience = true,
    //     ValidAudience = "",
    //     ValidateLifetime = true,
    //     ClockSkew = TimeSpan.Zero // 允许的时间偏移量,建议设置为零或较小值
    // };
});

// 其他服务注册,例如添加控制器
builder.Services.AddControllers();
// Learn more about configuring Swagger/OpenAPI at https://aka.ms/aspnetcore/swashbuckle
builder.Services.AddEndpointsApiExplorer();
builder.Services.AddSwaggerGen();

var app = builder.Build();

// Configure the HTTP request pipeline.
if (app.Environment.IsDevelopment())
{
    app.UseSwagger();
    app.UseSwaggerUI();
}

app.UseHttpsRedirection();

请将替换为您的客户端ID,将https://api.asgardeo.io/t//oauth2/token/.well-known/openid-configuration替换为您的外部授权服务器的OpenID Connect发现文档地址。

2. 配置认证与授权中间件

在请求处理管道中,必须按照正确的顺序添加认证和授权中间件。UseAuthentication()负责验证请求的凭据(即JWT令牌),而UseAuthorization()则根据验证结果和端点上定义的授权策略来决定是否允许访问。

图星人
图星人

好用的AI生图工具,百万免费商用图库

下载
// ... (之前的代码)

app.UseHttpsRedirection();

// 启用认证中间件,它会尝试解析并验证传入请求中的JWT令牌。
app.UseAuthentication();
// 启用授权中间件,它会根据已认证用户的身份和策略来决定访问权限。
app.UseAuthorization();

// 映射控制器路由,确保在认证和授权之后执行。
app.MapControllers();

app.Run();

重要提示: UseAuthentication()、UseAuthorization()和MapControllers()的顺序至关重要。认证必须在授权之前发生,而路由映射则应在两者之后,以确保所有请求都能经过认证和授权检查。

3. 保护API端点

最后,通过在控制器或特定的API方法上应用[Authorize]属性,即可轻松保护您的API端点。当请求到达带有此属性的端点时,如果没有有效的JWT令牌或令牌验证失败,ASP.NET Core将返回401 Unauthorized响应。

using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;

namespace YourAppName.Controllers
{
    [ApiController]
    [Route("[controller]")]
    public class WeatherForecastController : ControllerBase
    {
        private static readonly string[] Summaries = new[]
        {
            "Freezing", "Bracing", "Chilly", "Cool", "Mild", "Warm", "Balmy", "Hot", "Sweltering", "Scorching"
        };

        private readonly ILogger _logger;

        public WeatherForecastController(ILogger logger)
        {
            _logger = logger;
        }

        [HttpGet(Name = "GetWeatherForecast")]
        public IEnumerable Get()
        {
            return Enumerable.Range(1, 5).Select(index => new WeatherForecast
            {
                Date = DateTime.Now.AddDays(index),
                TemperatureC = Random.Shared.Next(-20, 55),
                Summary = Summaries[Random.Shared.Next(Summaries.Length)]
            })
            .ToArray();
        }

        // 添加一个受保护的私有端点
        [HttpGet]
        [Route("Private")] // 定义该私有端点的路由
        [Authorize] // 应用Authorize属性,表示此端点需要认证
        public IActionResult Private()
        {
            // 只有经过认证的用户才能访问此端点
            return Ok(new
            {
                Message = "Hello from a private endpoint. You are authorized!"
            });
        }
    }
}

现在,当客户端向/WeatherForecast/Private端点发送请求时,必须在Authorization请求头中包含一个有效的Bearer JWT令牌,否则请求将被拒绝。

注意事项

  • 外部授权服务器配置: 确保您的外部授权服务器(如AWS Cognito、Asgardeo、Auth0等)已正确配置,并且其OpenID Connect发现文档(/.well-known/openid-configuration)可公开访问。
  • 客户端类型: 本教程提供的解决方案适用于那些不涉及客户端密钥(Client Secret)的应用场景,例如单页应用(SPA)或移动应用。对于需要客户端密钥的保密客户端(Confidential Client),AddJwtBearer的配置可能需要额外的参数,例如通过Authority直接指定颁发者,或者通过HttpClient配置来处理密钥交换。
  • 令牌验证参数: MetadataAddress会自动获取大部分必要的验证参数。然而,如果您的需求特殊,可以通过TokenValidationParameters属性进行更细粒度的控制,例如强制验证颁发者(Issuer)、自定义时钟偏移(Clock Skew)等。
  • 错误处理: 对于认证失败的情况,ASP.NET Core默认会返回401 Unauthorized。您可以根据需要实现自定义的错误处理逻辑,例如通过options.Events来捕获认证失败事件。
  • 策略授权: [Authorize]属性除了可以简单地要求认证外,还可以结合策略(Policies)进行更复杂的授权控制,例如要求用户拥有特定的角色或声明。

总结

通过上述简洁的配置,您可以在C# .NET应用程序中快速构建一个纯资源服务器,实现基于JWT Bearer令牌的REST API安全。这种方法与Spring Boot中spring-boot-starter-oauth2-resource-server的理念高度一致,允许开发者专注于业务逻辑,而将令牌验证的复杂性交由框架和外部授权服务器处理。这极大地简化了API安全实现的流程,提高了开发效率和系统的可维护性。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

845

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

745

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

740

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

420

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

447

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

431

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16947

2023.08.03

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

25

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.8万人学习

C# 教程
C# 教程

共94课时 | 7.5万人学习

Java 教程
Java 教程

共578课时 | 50.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号