Entity Framework : Sharing entities across different DbContexts
Asked Answered
N

5

16

I'm developing a plugin application with EF6, code first.

I have one main context with an entity called User:

public class MainDataContext : DbContext
{
    public MainDataContext(): base("MainDataContextCS") {}
    public DbSet<User> Users { get; set; }
}

And then another context for PluginX, on another project which references the base one:

public class PluginDataContext : DbContext
{
    public PluginDataContext () : base("MainDataContextCS") {
    }

    protected override void OnModelCreating(DbModelBuilder modelBuilder) {
        modelBuilder.HasDefaultSchema("PluginX");
        base.OnModelCreating(modelBuilder);
    }

    public DbSet<Booking> Bookings { get; set; }
}

And this neatly creates, on the same Database (same connection string), the PluginX.Bookings Table.

The problem here is that the Booking entity contains a reference to User entity:

public class Booking
{
    public int Id { get; set;}
    public virtual User CreationUser { get; set;}
    public BookingStatus Status { get; set; }
}

And when running Add-Migration for the plugin context EF will try to create another User entity called PluginX.User.

How can this be solved? Is there a way to share a common entity, in another DbContext?

Negligible answered 15/1, 2015 at 17:53 Comment(1)
why dont inherit PluginContext from MainContext.Bernardinabernardine
T
15

When you work with multiple contexts you have two options:

  1. Treat each context like they were separate applications. Imagine your user is an external resource that you get from a web service. You won't be able to add a foreign key to that. What you would do in this is either add only the userId in your tables and when you need the user details call the external service to get them or have a local light copy of the user in the Bookings context that you would update every now and then from the Users context. This approach is good when you work with a large system and you want to isolate the parts (read about DDD and bounded contexts)
  2. Apart from your 2 contexts, create a third context with the whole model (users, bookings, etc). You will use the complete context to create the migrations and maintain the DB structure, but in the application you will use the smaller contexts. This is a very simple solution. It's easy to maintain the migrations with a single context and it still allows you to isolate the DB operation in smaller contexts that don't have access to unrelated entities.
Toussaint answered 27/11, 2015 at 17:22 Comment(2)
This also helped me think about my model, and try to de-normalize certain parts of my application.Agonic
Hey Franscesc, any changes to this approach? Is it still not possible with EF Core 3.0 to link entities across contexts?Greenback
L
2

When you add the Booking entity, don't use the DbSet.Add() method. Instead use the DbSet.Attach() method and set the DbContext.Entry(Entity).State property for the Booking to EntityState.Added and make sure the DbContext.Entry(Entity).State for User stays EntityState.Unchanged.

So for example instead of doing this:

pluginDataContext.dbBooking.Add(myNewBooking);

Do this:

pluginDataContext.dbBooking.Attach(myNewBooking);
pluginDataContext.Entry(myNewBooking).State = EntityState.Added;

This is because the Add() method marks all entities in the object graph as EntityState.Added which will cause inserts without checking if the entity already exists in the database. The Attach() method simply makes the context begin tracking the entity.

This is why I almost never use DbSet.Add().

Leavis answered 15/1, 2015 at 19:2 Comment(0)
D
2

You can try using views, declare the user as a view in PluginDataContext and when you perform the migration, type the method "create User view as ...", this allows you to relate the book to the user.

Danczyk answered 9/3, 2020 at 9:25 Comment(0)
L
1

This solution could help you:Entity Framework 6 Code First Migrations with Multiple Data Contexts. However, in this case, both context are in the same project. I don't know if works with contexts that are in two different projects (I think it should if you are using the same class to map User). As the blog said, you need to comment the generated code related to the Users table when you run the Add-Migration command for the PluginX Context.

Laclos answered 15/1, 2015 at 21:5 Comment(1)
I know this is old but.. I couldn't stop thinking that this solution is not great at all. Every time a new migration is added, you have to "comment" code? Sounds like a huge hack to me.Lendlease
D
0

It's 2024 now, but still relevant.
One more way exits here.
You can add this line to your OnModelCreating method in PluginDataContext:

modelBuilder.Entity<User>().Metadata.SetIsTableExcludedFromMigrations(true);

this prevents User from being involved into migration for PluginDataContext.

If you have many tables with FK to another DbContext, perhaps reflection can be used for this approach

Daydream answered 26/6, 2024 at 12:24 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.