23

我正在尝试在我的数据库中播种用户实体。该User实体有一个 owend 属性EmailPermissions

当我运行命令时

dotnet ef 迁移添加初始;

我得到错误

无法添加实体类型“用户”的种子实体,因为它设置了导航“电子邮件权限”。要为关系播种,您需要将相关实体种子添加到“EmailPermissions”并指定外键值 {“UserId”}。

但由于EmailPermissions是一个拥有的实体,我没有给它一个明确的UserId属性,这意味着我不能在数据库中单独播种它。

实体

public sealed class User : IdentityUser
{
    public User()
    {
        EmailPermissions = new EmailPermissions();
    }

    /* [..] */

    public string TemporaryEmailBeforeChange { get; set; }
    public bool IsEmailAwaitingUpdate { get; set; }
    public EmailPermissions EmailPermissions { get; set; }
    public ICollection<Registeration> Registerations { get; set; }

    /* [..] */

}

[Owned]
public class EmailPermissions
{
    /* [..] */

    public bool Newsletter { get; set; }
    public bool PromotionalOffers { get; set; }
    public bool PrestationReminders { get; set; }
    public bool PrestationOffers { get; set; }
}

播种电话

private void SeedUser(ModelBuilder builder)
{
    builder.Entity<User>().HasData(
        new User
        {
            Id = "37846734-172e-4149-8cec-6f43d1eb3f60",
            Email = "foo@foo.foo",
            UserName = "foo@foo.foo",
            PasswordHash = "AQAAAAEAACcQAAAAEIytBES+jqKH9jfuY3wzKyduDZruyHMGE6P+ODe1pSKM7BuGjd3AIe6RGRHrXidRsg==",
            SecurityStamp = "WR6VVAGISJYOZQ3W7LGB53EGNXCWB5MS",
            ConcurrencyStamp = "c470e139-5880-4002-8844-ed72ba7b4b80",
            EmailConfirmed = true
        });
}   

如果我从构造函数中删除属性的实例化,则会EmailPermissions收到以下错误

'User' 类型的实体与'EmailPermissions' 类型的实体共享表'AspNetUsers',但没有此类型的实体具有已标记为'Added' 的相同键值。

.HasData当用户拥有自有财产时,如何通过该方法播种用户?

4

5 回答 5

37

目前,文档中缺少此信息(由#710 跟踪:记录如何播种拥有的类型)。EF Core 团队(带有示例)在#12004:包含拥有类型线程的问题播种数据中对此进行了解释:

拥有的类型必须在HasData调用之后通过OwnsOne调用播种。此外,由于按惯例拥有的类型具有在影子状态下生成的主键,并且由于种子数据需要定义键,因此这需要使用匿名类型并设置键。

这基本上就是异常消息告诉您的内容。

按照建议,您应该EmailPermissions从构造函数中删除属性的实例化并添加如下的种子代码:

builder.Entity<User>().OwnsOne(e => e.EmailPermissions).HasData(
    new
    {
        UserId = "37846734-172e-4149-8cec-6f43d1eb3f60",
        // other properties ...
    }
);

由于需要知道影子 PK 名称和匿名类型的使用,因此非常烦人且容易出错。正如同一个成员提到的

请注意,如果播种支持导航,这将变得更容易,这由#10000 跟踪:数据播种:添加对导航的支持

于 2018-06-14T17:33:09.003 回答
8

感谢 Ivan Stoev 的回答。我添加了一些更容易想象的代码。这是基于示例的种子数据函数代码。

  • 首先添加用户的数据。
  • 之后添加拥有对象的数据。
  • 拥有对象的数据必须是匿名的,因为 PK 会请求。此 PK 不会出现在数据库中。名称应为实体名称 + Id

示例:实体 XXX => PK 将是 XXXId

private void SeedUser(ModelBuilder builder)
{
    builder.Entity<User>(b =>
    {
        b.HasData(new User
        {
            Id = "37846734-172e-4149-8cec-6f43d1eb3f60",
            Email = "foo@foo.foo",
            UserName = "foo@foo.foo",
            // more properties of User
        });
        b.OwnsOne(e => e.EmailPermissions).HasData(new 
        {
                UserId = "37846734-172e-4149-8cec-6f43d1eb3f60",
                Newsletter = true,
                PromotionalOffers = true,
                PrestationReminders = true,
                PrestationOffers = true
        });
    });
}
于 2019-07-19T18:45:12.427 回答
1

如果您想避免使用匿名类型来指定影子属性键,您可以在模型类中显式声明它们,并使用 Fluent API 将它们配置为键。这样您就不必猜测属性名称,并且不易出错。

如果提供给 Property 方法的名称与现有属性的名称(影子属性或在实体类上定义的属性)匹配,则代码将配置该现有属性,而不是引入新的影子属性。 资源

于 2019-04-03T18:04:18.403 回答
0

在我的场景中,我希望在父类中自动初始化拥有类型的属性:

public class User
{
  EmailPermissions _EmailPermissions;
  public EmailPermissions 
  {
    get => _EmailPermissions ??= new EmailPermissions();
    set => _EmailPermissions = value;
  }
}

当我尝试添加种子数据时,我得到了那个讨厌的异常。

解决方案是在其调用中传递Useras 匿名类型。HasData

于 2019-12-02T16:37:22.587 回答
0

我有同样的问题在启动时播种了我的数据。 这是github问题的链接。

于 2020-03-17T20:44:10.993 回答