- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我有这种自定义授权配置。问题是我想检查子模块以及它是否有权限。我还有一个登录 Controller ,我在其中构建声明等,并添加它们。但我想我也在这里遗漏了一些东西。我想我应该向当前的主体添加声明,但我不知道如何:
这是我在登录 Controller 中的登录 Action
[HttpGet("login")]
public IActionResult Login()
{
var userName = _httpContextAccessor.HttpContext.User.Identity.Name.GetUserNameFromHttpContext();
var user = _userClient.GetUserByUserName(userName);
var loggedIUserDto = new LoggedInUserDto {UserName = userName};
var claims = new List<Claim>();
if (user == null)
{
return Unauthorized($"User {userName} does not exits in DB");
}
var userModulesWithSubmodules = _loginClient.GetUserModulesWithSubmodules(userName);
loggedIUserDto.UserModulesWithSubmodules = userModulesWithSubmodules;
if (userModulesWithSubmodules.Count == 0)
{
return Conflict($"User {userName} has no modules");
}
foreach (var module in userModulesWithSubmodules)
{
foreach (var submodule in module.Submodules)
{
var submoduleActionList = new List<string>();
if (submodule.CanAdd)
{
submoduleActionList.Add("CanAdd");
}
if (submodule.CanEdit)
{
submoduleActionList.Add("CanEdit");
}
if (submodule.CanRead)
{
submoduleActionList.Add("CanRead");
}
if (submodule.CanDelete)
{
submoduleActionList.Add("CanDelete");
}
claims.Add(new Claim(ClaimTypes.Name, user.UserName));
claims.Add(new Claim(submodule.SubmoduleName, string.Join(',', submoduleActionList)));
}
}
var claimsIdentity = new ClaimsIdentity(claims);
var claimsPrincipal = new ClaimsPrincipal(claimsIdentity);
Thread.CurrentPrincipal = claimsPrincipal;
return Ok(loggedIUserDto);
}
客户授权
public class CustomAuthorize : AuthorizeAttribute
{
private SubmoduleActionType _submoduleActionType;
private SubmoduleType _submoduleType;
public SubmoduleActionType ActionType;
public SubmoduleType Type {
get => _submoduleType;
set
{
_submoduleType = value;
Policy = $"{_submoduleType.ToString()};{_submoduleActionType.ToString()}";
}
}
public CustomAuthorize(SubmoduleActionType submoduleActionType, SubmoduleType submoduleType)
{
_submoduleActionType = submoduleActionType;
_submoduleType = submoduleType;
}
}
下面是我的需求类:
public class SubmoduleTypeRequirement : IAuthorizationRequirement
{
public SubmoduleActionType? ActionType { get; set; }
public SubmoduleType? Type { get; set; }
public SubmoduleTypeRequirement(SubmoduleActionType actionType, SubmoduleType type)
{
Type = type;
ActionType = actionType;
}
}
这是我的处理程序类
public class SubmoduleAuthorizationHandler : AuthorizationHandler<SubmoduleTypeRequirement>
{
protected override Task HandleRequirementAsync(AuthorizationHandlerContext context, SubmoduleTypeRequirement submoduleRequirement)
{
if (!submoduleRequirement.ActionType.HasValue)
{
throw new ArgumentException("No action type provided");
}
if (!submoduleRequirement.Type.HasValue)
{
throw new ArgumentException("No submodule type provided");
}
if (!context.User.HasClaim(uc => uc.Type == submoduleRequirement.Type.ToString()))
{
context.Fail();
return Task.FromResult(0);
}
var grantedRights = Convert.ToString(context.User.FindFirst(c => c.Type == submoduleRequirement.Type.ToString()));
if (grantedRights.Contains(submoduleRequirement.ActionType.ToString()))
{
context.Succeed(submoduleRequirement);
}
return Task.FromResult(0);
}
}
最后是策略类:
public class SubmodulePolicy : IAuthorizationPolicyProvider
{
public SubmodulePolicy(IOptions<AuthorizationOptions> options)
{
DefaultPolicyProvider = new DefaultAuthorizationPolicyProvider(options);
}
public DefaultAuthorizationPolicyProvider DefaultPolicyProvider { get; }
public Task<AuthorizationPolicy> GetPolicyAsync(string policyName)
{
var submoduleTypeAndAction = policyName.Split(";");
var submoduleTypeString = submoduleTypeAndAction[0];
var actionTypeString = submoduleTypeAndAction[1];
var submoduleTypeParsed = System.Enum.TryParse(submoduleTypeString, out SubmoduleType submoduleType);
var actionTypeParsed = System.Enum.TryParse(actionTypeString, out SubmoduleActionType submoduleActionType);
if (actionTypeParsed && submoduleTypeParsed)
{
var policy = new AuthorizationPolicyBuilder();
policy.AddRequirements(new SubmoduleTypeRequirement(submoduleActionType, submoduleType));
return Task.FromResult(policy.Build());
}
if (!actionTypeParsed || !submoduleTypeParsed)
{
throw new ArgumentException("Cannot parse action or submoduleType from Policy");
}
return DefaultPolicyProvider.GetPolicyAsync(policyName);
}
public Task<AuthorizationPolicy> GetDefaultPolicyAsync()
{
return DefaultPolicyProvider.GetDefaultPolicyAsync();
}
}
问题是我的代码从未到达我的 GetPolicy。它总是转到 GetDefaultPolicyAsync 方法。甚至属性也不会被触发。我应该在某处注册自定义属性吗?我不信。无论我在互联网上找到什么,即使在官方文档中也总是有一个参数,但我需要两个。
哦,这是我在启动类中的配置
services.AddTransient<IAuthorizationPolicyProvider, SubmodulePolicy>();
services.AddSingleton<IAuthorizationHandler, SubmoduleAuthorizationHandler>();
services.AddAuthorization();
知道哪里出了问题吗?只有一个论点有什么限制吗?我也在考虑用过滤器替换它。
编辑:
根据我的启动类(class)的要求。只是评论 - jwt 配置被禁用,因为我在获取带有常量的 json 文件时遇到问题(获取文件时提示我输入用户名和密码 - iis 服务器)。这个 web api 结合了 angular 2+ (v7)
我的启动类:
public class Startup
{
public Startup(IConfiguration configuration)
{
Configuration = configuration;
}
public IConfiguration Configuration { get; }
public IContainer ApplicationContainer { get; private set; }
// This method gets called by the runtime. Use this method to add services to the container.
public IServiceProvider ConfigureServices(IServiceCollection services)
{
services.AddAuthentication(IISDefaults.AuthenticationScheme);
services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2)
.AddJsonOptions(options => options.SerializerSettings.ContractResolver = new CamelCasePropertyNamesContractResolver());
services.AddTransient<IAuthorizationPolicyProvider, SubmodulePolicy>();
services.AddSingleton<IAuthorizationHandler, SubmoduleAuthorizationHandler>();
//var appSettingsSection = Configuration.GetSection("Settings");
//services.Configure<AppSettings>(appSettingsSection);
//var appSettings = appSettingsSection.Get<AppSettings>();
//var key = Encoding.ASCII.GetBytes(appSettings.Secret);
//services
// .AddAuthentication(auth =>
// {
// auth.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;
// auth.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme;
// })
// .AddJwtBearer(jwtBearer =>
// {
// jwtBearer.RequireHttpsMetadata = false;
// jwtBearer.SaveToken = true;
// jwtBearer.TokenValidationParameters = new TokenValidationParameters
// {
// ValidateIssuerSigningKey = true,
// IssuerSigningKey = new SymmetricSecurityKey(key),
// ValidateIssuer = false,
// ValidateAudience = false
// };
// });
services.AddAuthorization();
services.Configure<IISServerOptions>(options =>
{
options.AutomaticAuthentication = true;
});
services.AddHttpContextAccessor();
var container = new ContainerBuilder();
container.RegisterType<PermissionsClient>()
.As<IPermissionsClient>()
.WithParameter("baseServiceUrl", Configuration.GetSection("Settings")["BaseApiUrl"])
.WithParameter("clientCertThumbprint", Configuration.GetSection("Settings")["BaseApiClientThumbPrint"])
.SingleInstance();
container.RegisterType<LoginClient>()
.As<ILoginClient>()
.WithParameter("baseServiceUrl", Configuration.GetSection("Settings")["BaseApiUrl"])
.WithParameter("clientCertThumbprint", Configuration.GetSection("Settings")["BaseApiClientThumbPrint"])
.SingleInstance();
container.RegisterType<UserClient>()
.As<IUserClient>()
.WithParameter("baseServiceUrl", Configuration.GetSection("Settings")["BaseApiUrl"])
.WithParameter("clientCertThumbprint", Configuration.GetSection("Settings")["BaseApiClientThumbPrint"])
.SingleInstance();
container.RegisterType<SenderClient>()
.As<ISenderClient>()
.WithParameter("baseServiceUrl", Configuration.GetSection("Settings")["BaseApiUrl"])
.WithParameter("clientCertThumbprint", Configuration.GetSection("Settings")["BaseApiClientThumbPrint"])
.SingleInstance();
container.Populate(services);
ApplicationContainer = container.Build();
return new AutofacServiceProvider(ApplicationContainer);
}
// This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
}
app.UseAuthentication();
app.UseMvc();
}
}
最佳答案
我已经设法让它工作了。问题出在我创建的属性中。这很奇怪,因为这里有文档 https://learn.microsoft.com/pl-pl/aspnet/core/security/authorization/iauthorizationpolicyprovider?view=aspnetcore-2.2
无论如何,在属性的 setter 中设置策略名称的 parrt 现在似乎对我有用。我不知道为什么。在我发现(尽管当你继承某些东西时这是显而易见的)除了我的自定义参数之外,我还可以传递一个策略名称之后,我再次尝试对其进行测试。当我将策略名称硬编码到我的自定义属性时,GetPolicyAsync 方法中的断点突然被击中。所以我尝试过类似的东西
public class CustomAuthorize : AuthorizeAttribute
{
private const string _policyName = "SubmodulePolicy"
private SubmoduleActionType _submoduleActionType;
private SubmoduleType _submoduleType;
public SubmoduleActionType ActionType;
public SubmoduleType Type { get; set;}
public CustomAuthorize(SubmoduleActionType submoduleActionType, SubmoduleType submoduleType)
{
_submoduleActionType = submoduleActionType;
_submoduleType = submoduleType;
Policy = $"{_policyName}-{submoduleActionType.ToString()}-{submoduleType.ToString()}"
}
}
突然间它开始工作了。另一件事是,在登录 Controller 中,您可以向用户添加声明,但如果这是此处的 Windows 身份验证
public class SubmoduleAuthorizationHandler : AuthorizationHandler<SubmoduleTypeRequirement>
{
protected override Task HandleRequirementAsync(AuthorizationHandlerContext context, SubmoduleTypeRequirement submoduleRequirement)
{
if (!submoduleRequirement.ActionType.HasValue)
{
throw new ArgumentException("No action type provided");
}
if (!submoduleRequirement.Type.HasValue)
{
throw new ArgumentException("No submodule type provided");
}
if (!context.User.HasClaim(uc => uc.Type == submoduleRequirement.Type.ToString()))
{
context.Fail();
return Task.FromResult(0);
}
var grantedRights = Convert.ToString(context.User.FindFirst(c => c.Type == submoduleRequirement.Type.ToString()));
if (grantedRights.Contains(submoduleRequirement.ActionType.ToString()))
{
context.Succeed(submoduleRequirement);
}
return Task.FromResult(0);
}
}
在 http 上下文中,您将获得不同的上下文,然后我在登录方法中进行更新。所以我不得不使用这个 Add custom claims to identity when using Windows authentication并在此处填充声明而不是登录方法。最后一个对很多人来说可能很明显,但对我来说却不是。也许我的冒险也会帮助某人。
关于c# - ASP.NET Core 自定义授权属性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56378992/
主要思想是将 EF Core nuget 包添加到 .NET Core 库项目,然后在一堆应用程序(例如 ASP.NET Core、Win 服务、控制台应用程序)中使用该库,而无需在每个应用程序中配置
我想要实现的是编写一个简单的.net核心后台工作程序(.net core 3.1)的代码,在该工作程序作为Windows服务运行时,我在其中将数据写入SQL Server数据库(通过EF Core 3
关于 .Net Core SDK download page 二进制文件有什么用?它与安装程序有何不同? 最佳答案 二进制文件是 .NET Core 的编译代码。它们拥有运行 .NET Core 所需
.NET Core 和 Entity Framework Core 之间的区别?我们可以在 .NET Core 中使用 Entity Framework Core 吗?两者都有什么优势? 最佳答案 E
.NET Core 和 ASP.NET Core 到底有什么区别? 它们是相互排斥的吗?我听说 ASP.NET Core 是基于 .NET Core 构建的,但它也可以基于完整的 .NET 框架构建。
我对 ASP.NET Core 开发完全陌生。我正在尝试使用单个模型和 mysql 创建一个简单的 asp.net core Web api 来存储模型数据,然后我想使用 Swagger 将其作为 R
.NET Core 和 Entity Framework Core 之间的区别?我们可以在 .NET Core 中使用 Entity Framework Core 吗?两者都有什么优势? 最佳答案 E
好吧,作为一个新的 .net 开发生态系统,我有点迷失在核心工具、版本等方面。 有人可以解释我之间的区别吗 VS 2015 核心工具预览版 x - See here .NET Core/SDK 与否
我已阅读有关如何通过信号器核心集线器从后台服务向客户端发送通知的文档。如何从客户端接收到后台服务的通知? 后台服务应该只是一个单例。 public class Startup { public
关闭。这个问题是opinion-based .它目前不接受答案。 想改善这个问题吗?更新问题,以便可以通过 editing this post 用事实和引文回答问题. 4年前关闭。 Improve t
非常简单的问题: 我正在尝试创建一个像这样的谓词构建器: var predicate = PredicateBuilder.False(); 但似乎在Net Core和EF Core中不可用。
在 .NET Core 自包含应用程序 中...我们需要在 project.json 中指定运行时 (RID) 我们希望我们的应用程序针对...发布为什么会这样? .NET Core 是跨平台的,与我
如何用 iCloud Core Data 替换我现有的 Core Data?这是我的持久商店协调员: lazy var persistentStoreCoordinator: NSPersistent
关闭。这个问题是opinion-based 。目前不接受答案。 想要改进这个问题吗?更新问题,以便 editing this post 可以用事实和引文来回答它。 . 已关闭 2 年前。 Improv
今天我正在学习新的 ASP.net 核心 API 3.1,我想将我的旧网站从 MVC4 转移到 Web API。除了一件事,一切都很好。数据库连接。在我的旧网站中,我为每个客户端(10/15 数据库)
我在 Visual Studio 2015 Update 3 和 .NET Core 1.0 中工作。我有一个 .NETCoreApp v1.0 类型的 Web API 项目。当我添加一个 .NET
我一直在尝试遵循 Ben Cull ( http://benjii.me/2016/06/entity-framework-core-migrations-for-class-library-proj
当我打开我的 vs 代码程序时,我收到以下消息: 无法找到 .NET Core SDK。 .NET Core 调试将不会启用。确保 .NET Core SDK 已安装并且在路径上。 如果我安装甚至卸载
我偶然发现了一个非常奇怪的问题。每当 Web 应用程序启动时,dotnet.exe 都会使用相当多的内存(大约 300M)。然而,当它触及某些部分时(我感觉这与 EF Core 使用有关),它会在短时
ASP.NET Core Web (.NET Core) 与 ASP.NET Core Web (.NET Framework) 有什么区别? .NET Framework 是否提供 similar
我是一名优秀的程序员,十分优秀!