.NET6之MiniAPI(九):基于角色的身份验证和授权

身份验证是这样一个过程:由用户提供凭据,然后将其与存储在操作系统、数据库、应用或资源中的凭据进行比较。 在授权过程中,如果凭据匹配,则用户身份验证成功,可执行已向其授权的操作。 授权指判断允许用户执行的操作的过程。也可以将身份验证理解为进入空间(例如服务器、数据库、应用或资源)的一种方式,而授权是用户可以对该空间(服务器、数据库或应用)内的哪些对象执行哪些操作。

微软官方文档

asp.net core支持多种授权,本篇重点说明JWT的基于角色授权方式。

基于JWT角色身份验证和授权,思路是在登录时分发加密的Token,在访问资源时带有这个Token,服务端要验证这个Token是不是自己分发的,如果是,再验证访问范围是否正确,本篇的范围就是那些资源是那种角色访问,得到Token的用户当前是那种角色,也就是角色和资源的匹配。

用asp.net core实现步骤:

1、appsettings.json中配置JWT参

2、添加身份认证和授权服务和中间件

3、定义生成Token的方法和验证Toekn参数的方法

4、登录时验证身份并分发Toekn

using Microsoft.AspNetCore.Authentication.JwtBearer;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Identity;
using Microsoft.IdentityModel.Tokens;
using System.IdentityModel.Tokens.Jwt;
using System.Security.Claims;
using System.Text;var builder = WebApplication.CreateBuilder();
//获取JWT参数,并注入到服务容器
var jwtConfig = new JWTConfig();
builder.Configuration.GetSection("JWTConfig").Bind(jwtConfig);
builder.Services.AddSingleton(jwtConfig);
//添加JJWT方式的身份认证和授权,
builder.Services.AddAuthorization().AddAuthentication(options =>{options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme;}).AddJwtBearer(JwtBearerDefaults.AuthenticationScheme, opt =>{opt.RequireHttpsMetadata = false;opt.TokenValidationParameters = JwtToken.CreateTokenValidationParameters(jwtConfig);});var app = builder.Build();
//使用身份认证和授权的中间件
app.UseAuthentication();
app.UseAuthorization();app.MapGet("/hellosystem", (ILogger<Program> logger, HttpContext context) =>
{var message = $"hello,system,{context.User?.Identity?.Name}";logger.LogInformation(message);return message;
}).RequireAuthorization(new RoleData { Roles = "system" });app.MapGet("/helloadmin", (ILogger<Program> logger, HttpContext context) =>
{var message = $"hello,admin,{context.User?.Identity?.Name}";logger.LogInformation(message);return message;
}).RequireAuthorization(new RoleData { Roles = "admin" });app.MapGet("/helloall", (ILogger<Program> logger, HttpContext context) =>
{var message = $"hello,all roles,{context.User?.Identity?.Name}";logger.LogInformation(message);return message;
}).RequireAuthorization(new RoleData { Roles = "admin,system" });//登录成功,并分发Token
app.MapPost("/login", [AllowAnonymous] (ILogger<Program> logger, LoginModel login, JWTConfig jwtConfig) =>
{logger.LogInformation("login");if (login.UserName == "gsw" && login.Password == "111111"){var now = DateTime.UtcNow;var claims = new Claim[] {new Claim(ClaimTypes.Role, "admin"),new Claim(ClaimTypes.Name, "桂素伟"),new Claim(ClaimTypes.Sid, login.UserName),new Claim(ClaimTypes.Expiration, now.AddSeconds(jwtConfig.Expires).ToString())};var token = JwtToken.BuildJwtToken(claims, jwtConfig);return token;}else{return "username or password is error";}
});app.Run();
//登录实体
public class LoginModel
{public string? UserName { get; set; }public string? Password { get; set; }
}
//JWT配置
public class JWTConfig
{public string? Secret { get; set; }public string? Issuer { get; set; }public string? Audience { get; set; }public int Expires { get; set; }
}
//JWT操作类型
public class JwtToken
{//获取Tokenpublic static dynamic BuildJwtToken(Claim[] claims, JWTConfig jwtConfig){var now = DateTime.UtcNow;var jwt = new JwtSecurityToken(issuer: jwtConfig.Issuer,audience: jwtConfig.Audience,claims: claims,notBefore: now,expires: now.AddSeconds(jwtConfig.Expires),signingCredentials: GetSigningCredentials(jwtConfig));var encodedJwt = new JwtSecurityTokenHandler().WriteToken(jwt);var response = new{Status = true,AccessToken = encodedJwt,ExpiresIn = now.AddSeconds(jwtConfig.Expires),TokenType = "Bearer"};return response;}static SigningCredentials GetSigningCredentials(JWTConfig jwtConfig){var keyByteArray = Encoding.ASCII.GetBytes(jwtConfig?.Secret!);var signingKey = new SymmetricSecurityKey(keyByteArray);return new SigningCredentials(signingKey, SecurityAlgorithms.HmacSha256);}//验证Token的参数public static TokenValidationParameters CreateTokenValidationParameters(JWTConfig jwtConfig){var keyByteArray = Encoding.ASCII.GetBytes(jwtConfig?.Secret!);var signingKey = new SymmetricSecurityKey(keyByteArray);return new TokenValidationParameters{ValidateIssuerSigningKey = true,IssuerSigningKey = signingKey,ValidateIssuer = true,ValidIssuer = jwtConfig?.Issuer,ValidateAudience = true,ValidAudience = jwtConfig?.Audience,ClockSkew = TimeSpan.Zero,RequireExpirationTime = true,};}
}
//mini api添加验证授权的参数类型
public class RoleData : IAuthorizeData
{public string? Policy { get; set; }public string? Roles { get; set; }public string? AuthenticationSchemes { get; set; }
}

验证结果:

1、没有登录,返回401

9213adda90241e8f1d00bf17c18e9629.png

2、登录,取token

c43c3a15d4c1d190de273c99fd8271df.png

3、正确访问

1ea82ada9ab0760ad7ace02639bd1c74.png

4、没有授权访问,返回403

e52f05bfe0f017d9b352336b25d7597a.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/292430.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

flex容器属性(一)

一&#xff0c;概念 flexible box ,意为“弹性布局”&#xff0c;用来为盒状模型提供最大的灵活性。 块级布局更侧重于垂直方向&#xff0c;行内布局更侧重于水平方向&#xff0c;于此相对的&#xff0c;弹性盒子布局算法是方向无关的。 块级flex布局&#xff1a; .box{display…

关于最近打的几题斜率优化的总结。加几AC代码。

斜率优化错误总结 网上说很多OJ桑的斜率优化大多都是模板题- -&#xff0c;结果每次都跪Orz。。。在此总结一些常见错误&#xff1a; 1&#xff1a;不得不说斜率优化很多时候计算式很长- -&#xff0c;代码容易错细节- -。 2&#xff1a;其次就是弹队头以及弹队尾的时候大小关系…

字符串之统计字符串

题目: 给一个str = "aaabbbcccddee"然后返回字符串“a_3_b_3_c_3_d_2_e_2” 分析: 给一个str = "aaabbbcccddee"然后返回字符串“a_3_b_3_c_3_d_2_e_2”,我们从结果可知道有一定规律,我们可以先得到a_3_b,然后把a_3_b保存起来,然后得到a_3_b_3_c 以此…

iio Engine logoHTML5 应用框架 iio Engine

iio Engine 是一个新的 HTML5 应用开源框架&#xff0c;基于 JavaScript 和 Canvas 开发&#xff0c;集成了 Box2D 在线演示&#xff1a;http://www.huiyi8.com/divcss/转载于:https://www.cnblogs.com/lhrs/p/4138106.html

带圈汉字 在线生成_手写签名在线生成器-手写签名在线生成器可复制

签名设计地址&#xff1a;www.mgs2s.com&#xff08;复制到浏览器打开&#xff09;工具集成签名设计免费版下载&#xff0c;签名设计免费版在线立即生成&#xff0c;简单简体签名设计免费版。最新方便设计公文签名设计颜色保存分享免费版1、所以大家最好写签名的时候&#xff0…

真快!10秒内将k8s集群运行起来

大家好&#xff0c;我是小碗汤&#xff0c;今天演示一个项目&#xff0c;可以在一分钟内用容器将k8s集群运行起来&#xff0c;真的很方便。您可能已经知道&#xff0c;将 Kubernetes 集群安装在 VM 上。但在 Docker 容器中安装一个 Kubernetes 集群&#xff0c;还没有太多的实践…

上公厕上一半突然被拽出来......

1 哈哈哈哈&#xff0c;太惨了&#xff08;素材来源网络&#xff0c;侵删&#xff09;▼2 糟糕&#xff0c;忘了切换账号了&#xff08;via.豆瓣史上写字最烂小组&#xff09;▼3 南北方酒局区别▼4 奇奇怪怪的东西又增加了&#xff08;素材来源网络&#xff0c;侵删&#x…

关于F5 排错的简单介绍之一

我只是每天抽出几分钟&#xff0c;写写自己想到的东西。有时间会进行修正。F5 是一直专注于4到7层&#xff0c;但是ping F5上某个IP 总丢包&#xff0c;总和F5 有点关系吧&#xff0c;哈哈~&#xff0c;说这话&#xff0c;想必做过F5的人&#xff0c;都有点苦逼了吧。所以&…

java之socket的OOBInline和UrgentData和发送心跳包研究

UrgentData可以理解为紧急发送数据方式&#xff0c;如果我们客户端先用write方法写入数据&#xff0c;再用UrgentData发送数据&#xff0c;再去执行flush操作&#xff0c;我们可以得到服务端先打印UrgentData发送的数据&#xff0c;然后再打印write写入的数据。 客户端代码实现…

context:annotation-config 跟 context:component-scan诠释及区别

<context:annotation-config> 是用于激活那些已经在spring容器里注册过的bean&#xff08;无论是通过xml的方式还是通过package sanning的方式&#xff09;上面的注解。 <context:component-scan>除了具有<context:annotation-config>的功能之外&#xff0c;…

python 读行为数组_python将多列文件读入数组

我在读一个文件&#xff0c;它看起来像&#xff1a;Protein in water5826300LEU N 2945 7.972 16.153 13.055 -0.0183 0.4861 -0.4376300LEU H 2946 8.006 16.194 13.139 1.5894 1.3176 -1.4422300LEU CA 2947 8.017 16.020 13.016 0.1247 0.7136 -0.1096300LEU CB 2948 8.157 …

01背包初始化的细节问题与循环下限的改进

转自&#xff1a;背包久讲 Tianyi Cui 初始化的细节问题 我们看到的求最优解的背包问题题目中&#xff0c;事实上有两种不太相同的问法。有的题目要求“恰好装满背包”时的最优解&#xff0c;有的题目则并没有要求必须把背包装满。一种区别这两种问法的实现方法是在初始化的时候…

为什么中国这么多高薪程序员,开发不出Java, Typescript, Python, Rust, Node.js这些基础设施?...

近日&#xff0c;有人在网上问了这个问题&#xff0c;引起了网友热议&#xff1a;为什么中国这么多高薪程序员&#xff0c;开发不出Java&#xff0c;Typescript, Python, Rust, Node.js这些基础设施&#xff1f;对这个问题&#xff0c;大家从不同角度发表了自己的看法。有人说&…

字符串之从规定的字符串中根据下标得到字符

题目: 例如给规定字符串str = "a_3_b_4_c_5",给定下标4,会得到字符b str = "a_3_b_4_c_5" 就意味着字符串是str = "aaabbbbccccc"; 然后下标4,就会得到字符‘b’ 代码实现: package com.chenyu.string.cn;import java.util.Arrays;public…

当你女朋友第一次来你家会发生什么?

1 五一假期你都怎么过&#xff1f;▼2 万一实现了呢&#xff08;via.月半土荅塔&#xff09;▼3 你下周三就见不到李总了&#xff01;▼4 对不起&#xff0c;但真的很好笑▼5 女朋友第一次来我家做饭▼6 还我美女&#xff01;&#xff01;&#xff01;▼7 走他们的路&…

Linux常用C函数-接口处理篇(网络通信函数)

接口处理篇accept&#xff0c;bind&#xff0c;connect&#xff0c;endprotoent&#xff0c;endservent&#xff0c;getsockopt&#xff0c;htonl&#xff0c;htons&#xff0c;inet_addr&#xff0c;inet_aton&#xff0c;inet_ntoa&#xff0c;listen&#xff0c;ntohl&#…

hdu 5108

//题意是给一个数N,然后让你求M,使得N/M为素数,并且M的值最小//思路呢,大概有两种,一个是遍历素数求解的,不过数据太大不现实//另外一种就是质因数求解,for循环是遍历质因数,然后while循环是剔除相同的质因数//最后你可以判定剩下来的如果非1,则是最大素数....#include <ios…

跟随大数据旅行

2019独角兽企业重金招聘Python工程师标准>>> 跟随大数据旅行 这是一本短小而精悍的书&#xff0c;不需要花费太多时间就能够让读者弄清 楚大数据到底是什么&#xff0c;还能帮助读者了解大数据的来龙去脉以及未来大 数据对各行各业带来的影响与作用。大数据将带来新…

利用for循环调用插入方法批量插入 一条失败_算法与数据结构(1):基础部分——以插入排序为例...

本文将会以插入排序为例&#xff0c;介绍算法与数据结构的基础部分。插入排序排序可以说是整个算法中最为基础&#xff0c;最为重要的一部分&#xff0c;而插入排序正是排序算法中最简单的一种解决办法。什么是排序问题&#xff1f;输入&#xff1a;n个数的一个序列 。输出&…