ASP.NET Core 7 Web API – authorization failed. These requirements were not met: DenyAnonymousAuthorizationRequirement: Requires an authenticated user

huangapple go评论109阅读模式
英文:

ASP.NET Core 7 Web API - authorization failed. These requirements were not met: DenyAnonymousAuthorizationRequirement: Requires an authenticated user

问题

Startup.cs

  1. services.AddAuthentication(options =>
  2. {
  3. options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;
  4. options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme;
  5. options.DefaultScheme = JwtBearerDefaults.AuthenticationScheme;
  6. })
  7. .AddJwtBearer(options =>
  8. {
  9. options.TokenValidationParameters = new TokenValidationParameters
  10. {
  11. ValidIssuer = jwtSettings.Issuer,
  12. ValidAudience = jwtSettings.Audience,
  13. IssuerSigningKey = new SymmetricSecurityKey(Encoding.UTF8.GetBytes(jwtSettings.Key)),
  14. ValidateIssuer = true,
  15. ValidateAudience = true,
  16. ValidateLifetime = true,
  17. ValidateIssuerSigningKey = true,
  18. };
  19. });
  20. app.UseMiddleware<ErrorHandlerMiddleware>();
  21. if (env.IsDevelopment())
  22. {
  23. app.UseDeveloperExceptionPage();
  24. app.UseSwagger();
  25. app.UseSwaggerUI(options =>
  26. {
  27. foreach (var description in provider.ApiVersionDescriptions)
  28. {
  29. options.SwaggerEndpoint($"/swagger/{description.GroupName}/swagger.json", description.GroupName.ToUpperInvariant());
  30. }
  31. });
  32. }
  33. app.UseCors();
  34. app.UseHttpsRedirection();
  35. app.UseAuthentication();
  36. app.UseRouting();
  37. app.UseAuthorization();
  38. app.UseEndpoints(endpoints =>
  39. {
  40. endpoints.MapControllers();
  41. });

Token generation:

  1. string CreateToken()
  2. {
  3. var jwtSettings = configuration.GetSection(nameof(AppSettings.Jwt)).Get<AppSettings.Jwt>();
  4. var key = new SymmetricSecurityKey(Encoding.UTF8.GetBytes(jwtSettings.Key));
  5. var credentials = new SigningCredentials(key, SecurityAlgorithms.HmacSha256);
  6. var claims = new List<Claim>
  7. {
  8. new Claim(JwtRegisteredClaimNames.Name, loginDto.Username)
  9. };
  10. var jwtSecurityToken = new JwtSecurityToken(
  11. expires: DateTime.Now.AddMinutes(30),
  12. claims: claims,
  13. signingCredentials: credentials,
  14. issuer: jwtSettings.Issuer,
  15. audience: jwtSettings.Audience);
  16. var jwt = new JwtSecurityTokenHandler().WriteToken(jwtSecurityToken);
  17. return jwt;
  18. }

Controller:

  1. [ApiController]
  2. [ApiVersion("1.0")]
  3. [Route("api/[controller]")]
  4. public class CustomerEnvironmentsController : ControllerBase
  5. {
  6. #region Fields
  7. private readonly ICustomerEnvironmentsRepository customerEnvironmentsRepository;
  8. private readonly IMapper mapper;
  9. private readonly IDtoValidatorFactory apiValidatorFactory;
  10. private readonly IHttpHeaderParser httpHeaderParser;
  11. #endregion
  12. #region Constructor
  13. public CustomerEnvironmentsController(ICustomerEnvironmentsRepository customerEnvironmentsRepository, IMapper mapper, IDtoValidatorFactory apiValidatorFactory, IHttpHeaderParser httpHeaderParser)
  14. {
  15. this.customerEnvironmentsRepository = customerEnvironmentsRepository ?? throw new ArgumentNullException(nameof(customerEnvironmentsRepository));
  16. this.mapper = mapper ?? throw new ArgumentNullException(nameof(mapper));
  17. this.apiValidatorFactory = apiValidatorFactory ?? throw new ArgumentNullException(nameof(apiValidatorFactory));
  18. this.httpHeaderParser = httpHeaderParser ?? throw new ArgumentNullException(nameof(httpHeaderParser));
  19. }
  20. #endregion
  21. [Authorize]
  22. [HttpGet]
  23. public async Task<ActionResult<List<CustomerEnvironmentDto>>> GetCustomerEnvironments()
  24. {
  25. //Ommitted
  26. }
  27. }

I only want this for specific endpoints, so I've added [Authorize] only on one endpoint. I've tried setting my token as auth in Swagger, and I've also tried manually sending my token from an external app with an Authorization header with the value bearer token. I just don't know what else to check.

英文:

Startup.cs:

  1. services.AddAuthentication(options =&gt;
  2. {
  3. options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;
  4. options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme;
  5. options.DefaultScheme = JwtBearerDefaults.AuthenticationScheme;
  6. })
  7. .AddJwtBearer(options =&gt;
  8. {
  9. options.TokenValidationParameters = new TokenValidationParameters
  10. {
  11. ValidIssuer = jwtSettings.Issuer,
  12. ValidAudience = jwtSettings.Audience,
  13. IssuerSigningKey = new SymmetricSecurityKey(Encoding.UTF8.GetBytes(jwtSettings.Key)),
  14. ValidateIssuer = true,
  15. ValidateAudience = true,
  16. ValidateLifetime = true,
  17. ValidateIssuerSigningKey = true,
  18. };
  19. });
  20. app.UseMiddleware&lt;ErrorHandlerMiddleware&gt;();
  21. if (env.IsDevelopment())
  22. {
  23. app.UseDeveloperExceptionPage();
  24. app.UseSwagger();
  25. app.UseSwaggerUI(options =&gt;
  26. {
  27. foreach (var description in provider.ApiVersionDescriptions)
  28. {
  29. options.SwaggerEndpoint($&quot;/swagger/{description.GroupName}/swagger.json&quot;, description.GroupName.ToUpperInvariant());
  30. }
  31. });
  32. }
  33. app.UseCors();
  34. app.UseHttpsRedirection();
  35. app.UseAuthentication();
  36. app.UseRouting();
  37. app.UseAuthorization();
  38. app.UseEndpoints(endpoints =&gt;
  39. {
  40. endpoints.MapControllers();
  41. });

Token generation:

  1. string CreateToken()
  2. {
  3. var jwtSettings = configuration.GetSection(nameof(AppSettings.Jwt)).Get&lt;AppSettings.Jwt&gt;();
  4. var key = new SymmetricSecurityKey(Encoding.UTF8.GetBytes(jwtSettings.Key));
  5. var credentials = new SigningCredentials(key, SecurityAlgorithms.HmacSha256);
  6. var claims = new List&lt;Claim&gt;
  7. {
  8. new Claim(JwtRegisteredClaimNames.Name, loginDto.Username)
  9. };
  10. var jwtSecurityToken = new JwtSecurityToken(
  11. expires: DateTime.Now.AddMinutes(30),
  12. claims: claims,
  13. signingCredentials: credentials,
  14. issuer: jwtSettings.Issuer,
  15. audience: jwtSettings.Audience);
  16. var jwt = new JwtSecurityTokenHandler().WriteToken(jwtSecurityToken);
  17. return jwt;
  18. }

Controller:

  1. [ApiController]
  2. [ApiVersion(&quot;1.0&quot;)]
  3. [Route(&quot;api/[controller]&quot;)]
  4. public class CustomerEnvironmentsController : ControllerBase
  5. {
  6. #region Fields
  7. private readonly ICustomerEnvironmentsRepository customerEnvironmentsRepository;
  8. private readonly IMapper mapper;
  9. private readonly IDtoValidatorFactory apiValidatorFactory;
  10. private readonly IHttpHeaderParser httpHeaderParser;
  11. #endregion
  12. #region Constructor
  13. public CustomerEnvironmentsController(ICustomerEnvironmentsRepository customerEnvironmentsRepository, IMapper mapper, IDtoValidatorFactory apiValidatorFactory, IHttpHeaderParser httpHeaderParser)
  14. {
  15. this.customerEnvironmentsRepository = customerEnvironmentsRepository ?? throw new ArgumentNullException(nameof(customerEnvironmentsRepository));
  16. this.mapper = mapper ?? throw new ArgumentNullException(nameof(mapper));
  17. this.apiValidatorFactory = apiValidatorFactory ?? throw new ArgumentNullException(nameof(apiValidatorFactory));
  18. this.httpHeaderParser = httpHeaderParser ?? throw new ArgumentNullException(nameof(httpHeaderParser));
  19. }
  20. #endregion
  21. [Authorize]
  22. [HttpGet]
  23. public async Task&lt;ActionResult&lt;List&lt;CustomerEnvironmentDto&gt;&gt;&gt; GetCustomerEnvironments()
  24. {
  25. //Ommitted
  26. }
  27. }

And I only want this for specific endpoints so I've added [Authorize] only on one endpoint. I've tried setting my token as auth in swagger, and I've also tried manually sending my token from an external app with an Authorization header with value bearer token.

I just don't know what else to check.

答案1

得分: 1

添加 System.IdentityModel.Tokens.Jwt NuGet 包似乎解决了这个问题。这一定是个 bug,没有任何地方表明缺少这个包。没有错误,没有警告,什么都没有。如果需要的话,它应该是主要 jwt 包的依赖项。

感谢用户 bsebe 的 这个答案,最终我解决了这个问题。

英文:

Ok so apparently adding System.IdentityModel.Tokens.Jwt nuget package solved it. This has to be a bug, nothing anywhere indicates that this package is missing. No error, no warnings, no nothing. If its needed it should be a dependency for the main jwt package.

Thanks to this answer from user bsebe i finally solved it.

huangapple
  • 本文由 发表于 2023年6月27日 17:30:29
  • 转载请务必保留本文链接:https://go.coder-hub.com/76563479.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定