使用客户端认证控制API访问(客户端授权模式)
场景描述
使用IdentityServer保护API的最基本场景。
我们定义一个API和要访问API的客户端。客户端从IdentityServer请求AccessToken,然后访问对应的API。
创建项目
-
IdentityServer的ASP.NET Core Web空项目,端口5000
-
Api的ASP.NET Core Web API项目,端口5001
-
Client的控制台项目
data:image/s3,"s3://crabby-images/3bf63/3bf632cc7e6a8b0a2ba68c20d7148e9486889a6d" alt=""
IdentityServer准备工作
定义API资源
data:image/s3,"s3://crabby-images/e13d9/e13d91727b76431b71f81307d7c607f27b01c7d5" alt=""
定义客户端
data:image/s3,"s3://crabby-images/4eeaf/4eeafac9e3f1a219c908ebfd0cf819142167d73b" alt=""
配置IdentityServer
data:image/s3,"s3://crabby-images/a624e/a624ea8b686d6e11478fa25ca347263cdeec2496" alt=""
运行IdentityServer项目(自宿主)并在浏览器中输入地址:http://localhost:5000/.well-known/openid-configuration既可以看到IdentityServer的各种元数据信息,可以使用在线json格式化工具显示如下:
data:image/s3,"s3://crabby-images/c2442/c24420a8aeb6b6fe2f24caa9f89287dfda4402d4" alt=""
API准备
在API控制器上,增加[Authorize]特性(授权)
data:image/s3,"s3://crabby-images/b871e/b871ebb2c99c167ee673636035d5e6f8d9d7b2b3" alt=""
startup增加如下代码:
data:image/s3,"s3://crabby-images/92597/925978ea260858ae083cf1f11208140ee3bff7a4" alt=""
AddAuthentication将身份认证添加到DI,使用"Bearer"方案。AddIdentityServerAuthentication将IdentityServer Token认证处理程序添加到DI,供身份认证服务使用。配置提供Token的基地址为:"http://localhost:5000",不使用https。
如果此时使用postman访问:http://localhost:5001/api/values,则会得到401的结果
data:image/s3,"s3://crabby-images/524ea/524ea54b4929208ed94fd8b9a6ed1f805da3ca36" alt=""
创建客户端
IdentityModel 包括用于发现 IdentityServer 各个终结点(EndPoint)的客户端库。这样只需要知道 IdentityServer 的地址 - 可以从元数据中读取实际的各个终结点地址:
var client = new HttpClient();
var disdoc = client.GetDiscoveryDocumentAsync("http://localhost:5000").Result;
if (disdoc.IsError)
{
Console.WriteLine(disdoc.Error);
}
获取token
var tokenResponse = client.RequestClientCredentialsTokenAsync(new ClientCredentialsTokenRequest
{
Address = disdoc.TokenEndpoint,
ClientId = "client",
ClientSecret = "secret",
Scope = "api1"
}).Result;
if (tokenResponse.IsError)
{
Console.WriteLine(tokenResponse.Error);
}
else
{
Console.WriteLine(tokenResponse.Json);
}
调用API
HttpClient httpClient = new HttpClient();
httpClient.SetBearerToken(tokenResponse.AccessToken);
var response = httpClient.GetAsync("http://localhost:5001/api/values").Result;
if (response.IsSuccessStatusCode)
{
Console.WriteLine(response.Content.ReadAsStringAsync().Result);
}
使用postman调用
data:image/s3,"s3://crabby-images/28b57/28b57e16ea8f9ea6a108549e8fcfa6cecd85b311" alt=""
网友评论