扩展Roles表后,实体类型IdentityRole不是当前上下文的模型的一部分 [英] The entity type IdentityRole is not part of the model for the current context after extending Roles table
问题描述
我已经扩展了由实体框架创建的AspNetRoles,使其看起来像这样:
I have extended my AspNetRoles that was created by Entity Framework to look like this:
public class AspNetRoles:IdentityRole
{
public AspNetRoles() : base() { }
public String Label { get; set; }
public String ApplicationId { get; set; }
public AspNetApplications Application { get; set; }
public static readonly String SystemAdministrator = "SystemAdministrator";
}
我了解到,因为我已经扩展了identityrole表,所以必须对usermanager进行更改.这就是我所做的:
I understood that because I have extended the identityrole table, I had to make changes to my usermanager. This is what I have done:
public class ApplicationUserManager : UserManager<ApplicationUser>
{
public ApplicationUserManager(IUserStore<ApplicationUser> store)
: base(store)
{
}
public static ApplicationUserManager Create(IdentityFactoryOptions<ApplicationUserManager> options, IOwinContext context)
{
var manager = new ApplicationUserManager(new UserStore<ApplicationUser>(context.Get<ApplicationDbContext>()));
// Configure validation logic for usernames
manager.UserValidator = new UserValidator<ApplicationUser>(manager)
{
AllowOnlyAlphanumericUserNames = false,
RequireUniqueEmail = true
};
// Configure validation logic for passwords
manager.PasswordValidator = new PasswordValidator
{
RequiredLength = 6,
RequireNonLetterOrDigit = true,
RequireDigit = true,
RequireLowercase = true,
RequireUppercase = true,
};
// Configure user lockout defaults
manager.UserLockoutEnabledByDefault = true;
manager.DefaultAccountLockoutTimeSpan = TimeSpan.FromMinutes(5);
manager.MaxFailedAccessAttemptsBeforeLockout = 5;
// Register two factor authentication providers. This application uses Phone and Emails as a step of receiving a code for verifying the user
// You can write your own provider and plug it in here.
manager.RegisterTwoFactorProvider("Phone Code", new PhoneNumberTokenProvider<ApplicationUser>
{
MessageFormat = "Your security code is {0}"
});
manager.RegisterTwoFactorProvider("Email Code", new EmailTokenProvider<ApplicationUser>
{
Subject = "Security Code",
BodyFormat = "Your security code is {0}"
});
manager.EmailService = new EmailService();
manager.SmsService = new SmsService();
var dataProtectionProvider = options.DataProtectionProvider;
if (dataProtectionProvider != null)
{
manager.UserTokenProvider =
new DataProtectorTokenProvider<ApplicationUser>(dataProtectionProvider.Create("ASP.NET Identity"));
}
return manager;
}
}
// Configure the application sign-in manager which is used in this application.
public class ApplicationSignInManager : SignInManager<ApplicationUser, string>
{
public ApplicationSignInManager(ApplicationUserManager userManager, IAuthenticationManager authenticationManager)
: base(userManager, authenticationManager)
{
}
public override Task<ClaimsIdentity> CreateUserIdentityAsync(ApplicationUser user)
{
return user.GenerateUserIdentityAsync((ApplicationUserManager)UserManager);
}
public static ApplicationSignInManager Create(IdentityFactoryOptions<ApplicationSignInManager> options, IOwinContext context)
{
return new ApplicationSignInManager(context.GetUserManager<ApplicationUserManager>(), context.Authentication);
}
}
public class ApplicationRoleManager : RoleManager<AspNetRoles>, IDisposable
{
public ApplicationRoleManager(RoleStore<AspNetRoles> store) : base(store)
{ }
public static ApplicationRoleManager Create(IdentityFactoryOptions<ApplicationRoleManager> options, IOwinContext context)
{
//AppIdentityDbContext db = context.Get<AppIdentityDbContext>();
//AppRoleManager manager = new AppRoleManager(new RoleStore<AppRole>(db));
return new ApplicationRoleManager(new RoleStore<AspNetRoles>(context.Get<ApplicationDbContext>()));
//return manager;
}
}
public class ApplicationUserStore<TUser> : UserStore<TUser, AspNetRoles, string, IdentityUserLogin, IdentityUserRole, IdentityUserClaim>, IUserStore<TUser>, IUserStore<TUser, string>, IDisposable where TUser : IdentityUser
{
public ApplicationUserStore(DbContext context) : base(context) { }
}
这是我的DBContext:
This is my DBContext:
public class ApplicationDbContext : IdentityDbContext<ApplicationUser, AspNetRoles, string, IdentityUserLogin, IdentityUserRole, IdentityUserClaim>
{
public virtual DbSet<AspNetUsersExtendedDetails> AspNetUsersExtendedDetails { get; set; }
public virtual DbSet<AspNetApplications> AspNetApplications { get; set; }
public virtual DbSet<AspNetEventLogs> AspNetEventLogs { get; set; }
public ApplicationDbContext() : base("AppStudio")
{
}
public static ApplicationDbContext Create()
{
return new ApplicationDbContext();
}
}
但是,在启动应用程序时出现此错误:
However, I get this error when I start my application:
实体类型IdentityRole不是当前上下文模型的一部分.
The entity type IdentityRole is not part of the model for the current context.
我不确定为什么会这样.扩展角色表后,我是否错过了需要更改的内容?
I'm not sure why this is happening. Have I missed something that needs to be changed after extending my roles table?
推荐答案
简短答案
以上代码中的主要问题是UserManager
的Create
方法.在该方法中,您应该使用UserStore
创建一个UserManager
,该UserStore
知道您创建的新角色类.为此,可以使用拥有的ApplicationUserStore
类的实例,或通过以下方式创建新的用户存储:
The main problem in above code is in Create
method of UserManager
. In that method, you should create a UserManager
using a UserStore
which is aware of the new role class which you created. To do so, you can use an instance of ApplicationUserStore
class which you have or create a new user store this way:
new UserStore<ApplicationUser, [YourRoleClass], string,
IdentityUserLogin, IdentityUserRole, IdentityUserClaim(
context.Get<ApplicationDbContext>())
如何向IdentityRole添加自定义属性?
要将新属性添加到IdentityRole
,您可以按照以下步骤操作:
How to Add a custom Property to IdentityRole?
To add a new property to IdentityRole
, you can follow the following steps:
- 创建一个 ASP.NET Web应用程序
- 确保选择 MVC ,并且 Authentication 是单个用户帐户
-
转到 Models 文件夹→打开 IdentityModels.cs 并创建包含要添加的自定义属性的 ApplicationRole 类:
- Create an ASP.NET Web Application
- Make sure you select MVC and the Authentication is Individual User Accounts
Go to Models folder → Open IdentityModels.cs and Create ApplicationRole class containing the custom property that you want to add:
public class ApplicationRole : IdentityRole //My custom role class
{
public string ApplicationId { get; set; } //My custom property
}
更改ApplicationUser
的GenerateUserIdentityAsync
方法以接受UserManager<ApplicationUser, string>
类型的参数:
Change GenerateUserIdentityAsync
method of ApplicationUser
to accept parameter of type of UserManager<ApplicationUser, string>
:
public class ApplicationUser : IdentityUser
{
public async Task<ClaimsIdentity> GenerateUserIdentityAsync(UserManager<ApplicationUser, string> manager)
{
更改ApplicationDbContext
基类并引入所有通用参数:
Change ApplicationDbContext
base class and introduce all the generic parameters:
public class ApplicationDbContext : IdentityDbContext<ApplicationUser, ApplicationRole, string, IdentityUserLogin, IdentityUserRole, IdentityUserClaim>
{
public ApplicationDbContext()
: base("DefaultConnection")
{
构建项目.
Build the project.
转到 App_Start 文件夹→打开 IdentityConfig.cs 并更改ApplicationUserManager
类以从UserManager<ApplicationUser, string>
派生,还更改其Create
方法返回知道ApplicationRole
的UserManage
:
Go to App_Start folder → Open IdentityConfig.cs and Change the ApplicationUserManager
class to derive from UserManager<ApplicationUser, string>
and also change its Create
method to return a UserManage
aware of ApplicationRole
:
public class ApplicationUserManager : UserManager<ApplicationUser, string>
{
public ApplicationUserManager(IUserStore<ApplicationUser, string> store)
: base(store)
{
}
public static ApplicationUserManager Create(IdentityFactoryOptions<ApplicationUserManager> options, IOwinContext context)
{
var manager = new ApplicationUserManager(new UserStore<ApplicationUser, ApplicationRole, string, IdentityUserLogin, IdentityUserRole, IdentityUserClaim>(context.Get<ApplicationDbContext>()));
要管理角色,请在同一文件中创建ApplicationRoleManager
类:
public class ApplicationRoleManager : RoleManager<ApplicationRole>
{
public ApplicationRoleManager(IRoleStore<ApplicationRole, string> store) : base(store) { }
public static ApplicationRoleManager Create(
IdentityFactoryOptions<ApplicationRoleManager> options,
IOwinContext context)
{
return new ApplicationRoleManager(new RoleStore<ApplicationRole>(context.Get<ApplicationDbContext>()));
}
}
转到 App_Start 文件夹→打开 Startup.Auth.cs 并将以下代码添加到ConfigureAuth
方法:
Go to App_Start folder → Open Startup.Auth.cs and add the following code to the ConfigureAuth
method:
ConfigureAuthapp.CreatePerOwinContext<ApplicationRoleManager>(ApplicationRoleManager.Create);
现在,该项目已准备好利用新的ApplicationRole
.
Now the project is ready to take advantage of the new ApplicationRole
.
这篇关于扩展Roles表后,实体类型IdentityRole不是当前上下文的模型的一部分的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!