SecurityRequirementsOperationFilter.cs 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using Jellyfin.Api.Auth.DefaultAuthorizationPolicy;
  5. using Jellyfin.Api.Constants;
  6. using Jellyfin.Extensions;
  7. using Microsoft.AspNetCore.Authorization;
  8. using Microsoft.OpenApi.Models;
  9. using Swashbuckle.AspNetCore.SwaggerGen;
  10. namespace Jellyfin.Server.Filters;
  11. /// <summary>
  12. /// Security requirement operation filter.
  13. /// </summary>
  14. public class SecurityRequirementsOperationFilter : IOperationFilter
  15. {
  16. private const string DefaultAuthPolicy = "DefaultAuthorization";
  17. private static readonly Type _attributeType = typeof(AuthorizeAttribute);
  18. private readonly IAuthorizationPolicyProvider _authorizationPolicyProvider;
  19. /// <summary>
  20. /// Initializes a new instance of the <see cref="SecurityRequirementsOperationFilter"/> class.
  21. /// </summary>
  22. /// <param name="authorizationPolicyProvider">The authorization policy provider.</param>
  23. public SecurityRequirementsOperationFilter(IAuthorizationPolicyProvider authorizationPolicyProvider)
  24. {
  25. _authorizationPolicyProvider = authorizationPolicyProvider;
  26. }
  27. /// <inheritdoc />
  28. public void Apply(OpenApiOperation operation, OperationFilterContext context)
  29. {
  30. var requiredScopes = new List<string>();
  31. var requiresAuth = false;
  32. // Add all method scopes.
  33. foreach (var authorizeAttribute in context.MethodInfo.GetCustomAttributes(_attributeType, true).Cast<AuthorizeAttribute>())
  34. {
  35. requiresAuth = true;
  36. var policy = authorizeAttribute.Policy ?? DefaultAuthPolicy;
  37. if (!requiredScopes.Contains(policy, StringComparer.Ordinal))
  38. {
  39. requiredScopes.Add(policy);
  40. }
  41. }
  42. // Add controller scopes if any.
  43. var controllerAttributes = context.MethodInfo.DeclaringType?.GetCustomAttributes(_attributeType, true).Cast<AuthorizeAttribute>();
  44. if (controllerAttributes is not null)
  45. {
  46. foreach (var authorizeAttribute in controllerAttributes)
  47. {
  48. requiresAuth = true;
  49. var policy = authorizeAttribute.Policy ?? DefaultAuthPolicy;
  50. if (!requiredScopes.Contains(policy, StringComparer.Ordinal))
  51. {
  52. requiredScopes.Add(policy);
  53. }
  54. }
  55. }
  56. if (!requiresAuth)
  57. {
  58. return;
  59. }
  60. if (!operation.Responses.ContainsKey("401"))
  61. {
  62. operation.Responses.Add("401", new OpenApiResponse { Description = "Unauthorized" });
  63. }
  64. if (!operation.Responses.ContainsKey("403"))
  65. {
  66. operation.Responses.Add("403", new OpenApiResponse { Description = "Forbidden" });
  67. }
  68. var scheme = new OpenApiSecurityScheme
  69. {
  70. Reference = new OpenApiReference
  71. {
  72. Type = ReferenceType.SecurityScheme,
  73. Id = AuthenticationSchemes.CustomAuthentication
  74. },
  75. };
  76. // Add DefaultAuthorization scope to any endpoint that has a policy with a requirement that is a subset of DefaultAuthorization.
  77. if (!requiredScopes.Contains(DefaultAuthPolicy.AsSpan(), StringComparison.Ordinal))
  78. {
  79. foreach (var scope in requiredScopes)
  80. {
  81. var authorizationPolicy = _authorizationPolicyProvider.GetPolicyAsync(scope).GetAwaiter().GetResult();
  82. if (authorizationPolicy is not null
  83. && authorizationPolicy.Requirements.Any(r => r is DefaultAuthorizationRequirement))
  84. {
  85. requiredScopes.Add(DefaultAuthPolicy);
  86. break;
  87. }
  88. }
  89. }
  90. operation.Security = [new OpenApiSecurityRequirement { [scheme] = requiredScopes }];
  91. }
  92. }