Reputation: 3240
i tried the whole day to get this working. I learned a lot about EF's Fluent API (e.g. this is an excellent article), however i had no success.
I have three Entities:
public class Address
{
[Key]
public virtual int AddressId { get; set; }
public virtual string AddressString { get; set; }
}
public class User
{
[Key]
public virtual int UserId { get; set; }
public virtual ICollection<Address> Addresses { get; set; }
}
public class House
{
[Key]
public virtual int HouseId { get; set; }
public virtual Address Address { get; set; }
}
and tried all combinations of HasMany, HasOptional, WithOptional, WithOptionalDependent
and WithOptionalPrincipial
i could think of for both User
and House
in the
protected override void OnModelCreating(DbModelBuilder modelBuilder)
I just cannot get it to work. I think it should be clear, what i want. A User may have more than one address (in the first place i want to force at least one, but now i would be happy if a user may have the addresses optional...) while a House has exactly one Address - and this is required. It would be nice if the address of a house would be cascading deleted.
Upvotes: 6
Views: 9292
Reputation:
protected override void OnModelCreating(ModelBuilder modelBuilder) { base.OnModelCreating(modelBuilder);
modelBuilder.Entity<Technician>()
.HasOne(t => t.Service)
.WithMany(t => t.Technicians)
.HasForeignKey(t => t.ServiceId)
.OnDelete(DeleteBehavior.Restrict);
}
Upvotes: 0
Reputation: 30618
I believe the following should work for you
public class Address
{
public int AddressId { get; set; }
public string AddressString { get; set; }
}
public class User
{
public int UserId { get; set; }
public virtual ICollection<Address> Addresses { get; set; }
}
public class House
{
public int HouseId { get; set; }
public virtual Address Address { get; set; }
}
public class TestContext : DbContext
{
public DbSet<Address> Addresses { get; set; }
public DbSet<User> Users { get; set; }
public DbSet<House> Houses { get; set; }
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Entity<User>().HasMany(u => u.Addresses).WithMany();
modelBuilder.Entity<House>().HasRequired(h => h.Address).WithOptional().Map(m => m.MapKey("AddressId"));
}
}
Note that it's often better to specify the foreign key fields yourself, which can make life a lot easier for you later on. If you do this, then you can choose to rewrite House as the following:
public class House
{
public int HouseId { get; set; }
public int AddressId { get; set; }
public virtual Address Address { get; set; }
}
Convention will link up AddressId and Address. If you have a one-to-one mapping between House and Address, you could also link them on their primary keys:
public class House
{
[ForeignKey("Address")]
public int HouseId { get; set; }
public virtual Address Address { get; set; }
}
You mentioned that you would like to enforce at least one address - this isn't possible with a one-to-many relationship. You could only do this if a user had exactly one address, at which point you could add a required AddressId property on the User class.
One other comment - you made everything virtual in your code. You only need to make navigation properties virtual.
Upvotes: 4
Reputation: 14938
What about something like this:
Enforce an Address
for a House
:
modelBuilder.Entity<House>().HasRequired(d => d.Address);
To create a one-to-many relationship between User
and Address
:
modelBuilder.Entity<User>().HasMany(u => u.Addresses).WithMany().Map(x =>
{
x.MapLeftKey("UserId");
x.MapRightKey("AddressId");
x.ToTable("UserAddress");
});
What this should do is create a table called UserAddress
that maps a User
to an Address
.
Alternatively, you could create a POCO for UserAddress
yourself and modify the tables in question:
public class UserAddress
{
[Key]
public int UserAddressId { get; set; }
public User User { get; set; }
public Address Address { get; set; }
}
public User
{
...
public virtual ICollection<UserAddress> UserAddresses { get; set; }
...
}
public Address
{
...
public virtual ICollection<UserAddress> UserAddresses { get; set; }
...
}
You will then need the following to ensure that both User
and Address
are required:
modelBuilder.Entity<UserAddress>().HasRequired(u => u.Address);
modelBuilder.Entity<UserAddress>().HasRequired(u => u.User);
Upvotes: 1