无法使用来自 Asp.Net Web API 的代理流调用图 API
Unable to call graphAPI using On-behalf-of flow from Asp.Net Web API
我正在尝试在 Asp.net Web API (.net 5) 中实现代表用户。我从移动应用程序收到 access_token,将其发送到我的网站 API。 Web API 使用此令牌调用 GRAPH API 以获取用户的个人资料详细信息。
下面是我的代码
Startup.cs 文件
services.AddAuthentication("JwtBearer")
.AddJwtBearer("JwtBearer", options =>
{
options.MetadataAddress = $"https://login.microsoftonline.com/{Configuration["b2bAzureAppIdentity:TenantId"]}/v2.0/.well-known/openid-configuration";
options.TokenValidationParameters = new TokenValidationParameters()
{
ValidIssuer = $"https://sts.windows.net/{Configuration["b2bAzureAppIdentity:TenantId"]}/",
// as audience, both the client id and the identifierUri are allowed (sematically equivalent)
ValidAudiences = new[] { Configuration["b2bAzureAppIdentity:AppIdUri"], Configuration["b2bAzureAppIdentity:ClientId"] }
};
}).AddMicrosoftIdentityWebApi(Configuration, "b2bAzureAppIdentity")
.EnableTokenAcquisitionToCallDownstreamApi()
.AddMicrosoftGraph(Configuration.GetSection("DownstreamApi"))
.AddInMemoryTokenCaches();
controller.cs
[HttpGet("GetMyDetails")]
[AuthorizeForScopes(Scopes = new string[] { "user.read" })]
public async Task<IActionResult> GetMyDetails()
{
var user = await _graphServiceClient.Me.Request().GetAsync();
return new OkObjectResult(user.Photo);
}
Appsettings 格式如下
"b2bAzureAppIdentity": {
"Instance": "https://login.microsoftonline.com/",
"Domain": "",
"TenantId": "",
"ClientId": "",
"ClientSecret": "",
"AppIdUri": ""},
"DownstreamApi": {
"BaseUrl": "https://graph.microsoft.com/v1.0",
"Scopes": "user.read"},
在 Azure 中,API 权限和范围设置正确,这很明显,因为当我从邮递员那里拨打电话时,我能够获得 on_behalf_of 的访问令牌并将其用于通过调用 https://graph.microsoft.com/v1.0/me
获取用户的个人资料详细信息
在该行的控制器中
var user = await _graphServiceClient.Me.Request().GetAsync();
我收到一个错误:“没有帐户或登录提示被传递给 AcquireTokenSilent 调用。”
我用谷歌搜索了这个错误,解决方案说用户应该同意这个范围,但是它已经在 Azure 门户中得到了管理员的同意。此外,这在 Postman 中起作用的事实让人相信 APP 和 API 的配置是正确的。
有没有人遇到过类似的问题?
发生这种情况是因为收到的 access_token 没有随获取用户详细信息的请求一起发送。下面是如何实现 on behalf of provider:
的示例
// Create a client application.
IConfidentialClientApplication confidentialClientApplication = ConfidentialClientApplicationBuilder
.Create(clientId)
.WithTenantId(tenantID)
// The Authority is a required parameter when your application is configured
// to accept authentications only from the tenant where it is registered.
.WithAuthority(authority)
.WithClientSecret(clientSecret)
.Build();
// Use the API reference to determine which scopes are appropriate for your API request.
// e.g. - https://docs.microsoft.com/en-us/graph/api/user-get?view=graph-rest-1.0&tabs=http
var scopes = new string[] { "User.Read" };
// Create an authentication provider.
ClientCredentialProvider authenticationProvider = new OnBehalfOfProvider(confidentialClientApplication, scopes);
var jsonWebToken = actionContext.Request.Headers.Authorization.Parameter;
var userAssertion = new UserAssertion(jsonWebToken);
// Configure GraphServiceClient with provider.
GraphServiceClient graphServiceClient = new GraphServiceClient(authenticationProvider);
// Make a request
var me = await graphServiceClient.Me.Request().WithUserAssertion(userAssertion).GetAsync();
在这种情况下,令牌被添加到对 WithUserAssertion
的调用中的请求。
如果这对您有帮助,或者您还有其他问题,请告诉我。
我正在尝试在 Asp.net Web API (.net 5) 中实现代表用户。我从移动应用程序收到 access_token,将其发送到我的网站 API。 Web API 使用此令牌调用 GRAPH API 以获取用户的个人资料详细信息。 下面是我的代码 Startup.cs 文件
services.AddAuthentication("JwtBearer")
.AddJwtBearer("JwtBearer", options =>
{
options.MetadataAddress = $"https://login.microsoftonline.com/{Configuration["b2bAzureAppIdentity:TenantId"]}/v2.0/.well-known/openid-configuration";
options.TokenValidationParameters = new TokenValidationParameters()
{
ValidIssuer = $"https://sts.windows.net/{Configuration["b2bAzureAppIdentity:TenantId"]}/",
// as audience, both the client id and the identifierUri are allowed (sematically equivalent)
ValidAudiences = new[] { Configuration["b2bAzureAppIdentity:AppIdUri"], Configuration["b2bAzureAppIdentity:ClientId"] }
};
}).AddMicrosoftIdentityWebApi(Configuration, "b2bAzureAppIdentity")
.EnableTokenAcquisitionToCallDownstreamApi()
.AddMicrosoftGraph(Configuration.GetSection("DownstreamApi"))
.AddInMemoryTokenCaches();
controller.cs
[HttpGet("GetMyDetails")]
[AuthorizeForScopes(Scopes = new string[] { "user.read" })]
public async Task<IActionResult> GetMyDetails()
{
var user = await _graphServiceClient.Me.Request().GetAsync();
return new OkObjectResult(user.Photo);
}
Appsettings 格式如下
"b2bAzureAppIdentity": {
"Instance": "https://login.microsoftonline.com/",
"Domain": "",
"TenantId": "",
"ClientId": "",
"ClientSecret": "",
"AppIdUri": ""},
"DownstreamApi": {
"BaseUrl": "https://graph.microsoft.com/v1.0",
"Scopes": "user.read"},
在 Azure 中,API 权限和范围设置正确,这很明显,因为当我从邮递员那里拨打电话时,我能够获得 on_behalf_of 的访问令牌并将其用于通过调用 https://graph.microsoft.com/v1.0/me
获取用户的个人资料详细信息在该行的控制器中
var user = await _graphServiceClient.Me.Request().GetAsync();
我收到一个错误:“没有帐户或登录提示被传递给 AcquireTokenSilent 调用。”
我用谷歌搜索了这个错误,解决方案说用户应该同意这个范围,但是它已经在 Azure 门户中得到了管理员的同意。此外,这在 Postman 中起作用的事实让人相信 APP 和 API 的配置是正确的。
有没有人遇到过类似的问题?
发生这种情况是因为收到的 access_token 没有随获取用户详细信息的请求一起发送。下面是如何实现 on behalf of provider:
的示例// Create a client application.
IConfidentialClientApplication confidentialClientApplication = ConfidentialClientApplicationBuilder
.Create(clientId)
.WithTenantId(tenantID)
// The Authority is a required parameter when your application is configured
// to accept authentications only from the tenant where it is registered.
.WithAuthority(authority)
.WithClientSecret(clientSecret)
.Build();
// Use the API reference to determine which scopes are appropriate for your API request.
// e.g. - https://docs.microsoft.com/en-us/graph/api/user-get?view=graph-rest-1.0&tabs=http
var scopes = new string[] { "User.Read" };
// Create an authentication provider.
ClientCredentialProvider authenticationProvider = new OnBehalfOfProvider(confidentialClientApplication, scopes);
var jsonWebToken = actionContext.Request.Headers.Authorization.Parameter;
var userAssertion = new UserAssertion(jsonWebToken);
// Configure GraphServiceClient with provider.
GraphServiceClient graphServiceClient = new GraphServiceClient(authenticationProvider);
// Make a request
var me = await graphServiceClient.Me.Request().WithUserAssertion(userAssertion).GetAsync();
在这种情况下,令牌被添加到对 WithUserAssertion
的调用中的请求。
如果这对您有帮助,或者您还有其他问题,请告诉我。