我正在尝试使用ASP.NET Core Identity 2.0在我现有的应用程序中设置身份验证。当我使用自己的数据库架构和类时,我拥有自己的User和Role类,并且必须创建自定义UserStore / UserManager / RoleStore / RoleManager。
我在服务中声明了它们:
services.AddIdentity<User, Profile().AddUserManager<CustomUserManager<User>>().AddRoleManager<CustomRoleManager>().AddDefaultTokenProviders();
services.AddTransient<IUserStore<User>, UserStore>();
services.AddTransient<IRoleStore<Profile>, ProfileStore>();
services.AddTransient<UserResolverService>();
我通过以下方法在UserStore中实现了UserRoleStore接口:
public Task AddToRoleAsync(User user, string roleName, CancellationToken cancellationToken)
{
user.Profiles.Add(db.Profiles.ToList().Find(x => x.Name.Equals(roleName, StringComparison.OrdinalIgnoreCase)));
db.SaveChanges();
return Task.FromResult((object)null);
}
public Task RemoveFromRoleAsync(User user, string roleName, CancellationToken cancellationToken)
{
user.Profiles.Remove(db.Profiles.ToList().Find(x => x.Name.Equals(roleName, StringComparison.OrdinalIgnoreCase)));
db.SaveChanges();
return Task.FromResult((object)null);
}
public Task<IList<string>> GetRolesAsync(User user, CancellationToken cancellationToken)
{
IList<string> ret = new List<string>();
user.Profiles.ToList().ForEach(x => ret.Add(x.Name));
return Task.FromResult(ret);
}
public Task<bool> IsInRoleAsync(Useruser, string roleName, CancellationToken cancellationToken)
{
Profile searchRole = db.Profiles.Include(profile => profile.ProfileUsers).ToList().Find(x => x.Active && x.Name.Equals(roleName, StringComparison.OrdinalIgnoreCase));
return Task.FromResult(searchRole.ProfileUsers.ToList().Find(x => x.UserID == user.ID) == null ? false : true);
}
public Task<bool> IsInRole(User user, string roleName, CancellationToken cancellationToken)
{
return Task.FromResult(true);
}
public Task<IList<Utilisateur>> GetUsersInRoleAsync(string roleName, CancellationToken cancellationToken)
{
Profile currentProfile = db.Profiles.Include(profile => profile.ProfileUsers).ThenInclude(pu => pu.User).ToList().Find(x => x.Active && x.Name.Equals(roleName, StringComparison.OrdinalIgnoreCase));
if (currentProfile == null)
{
return Task.FromResult((IList<User>)null);
}
IList<User> ret = new List<User>();
currentProfile.ProfileUsers.ToList().ForEach(x => ret.Add(x.User));
return Task.FromResult(ret);
}
所以当我尝试做
var result1 = _userManager.AddToRoleAsync(userObject, "WorkOrderViewer");
var res = _userManager.GetUsersInRoleAsync("WorkOrderViewer");
var test = await _userManager.IsInRoleAsync(userObject, "WorkOrderViewer");
它起作用了,我的用户获得了所选角色。但是,当我尝试在视图中设置控制器授权或控件可见性时,它不起作用。对于控制器授权,我有一个拒绝访问的例子。
当我将其放入代码中时:
var test = User.IsInRole("WorkOrderViewer");
即使我登录应用程序(我尝试注销并登录)并且UserManager.IsInRoleAsync返回我“ true”,它也会返回“ false”。
我认为它失败是因为UserManager.AddToRoleAsync()未与User.IsInRole()同步,但是我不知道如何解决它。 你知道我在做什么错吗?
答案 0 :(得分:1)
当您说使用代码User.IsInRole
时,我假设您的意思是Controller
。
方法ClaimsPrincipal.IsInRole
具有以下文档:
IsInRole方法检查此声明主体拥有的身份是否包含类型为ClaimsIdentity.RoleClaimType的声明,其中声明的值等于角色参数指定的值。
这可以通过查看source for ClaimsIdentity.cs来确认。
if (_identities[i].HasClaim(_identities[i].RoleClaimType, role))
因此User.IsInRole
正在检查用户是否具有类型为RoleClaimType的Claim(默认为“ http://schemas.microsoft.com/ws/2008/06/identity/claims/role”。)。
可以根据需要配置声明类型。