Efcore: Relational: TPC inheritance mapping pattern

Created on 18 Sep 2015  路  37Comments  路  Source: dotnet/efcore

Consider configuring or issuing a warning to help with split identity seeding or shared sequence for the PK
FKs that point to a TPC entity type won't be enforced

Abstract base mapped to null table should produce TPC

area-migrations area-model-building area-query area-relational-mapping area-save-changes consider-for-current-release ef6-parity type-enhancement

Most helpful comment

Will this ever make it in? 884 days and counting...

https://days.to/18-september/2015

All 37 comments

Can I make assumption that TPC is now going to be in RC? judged by the following work as expected

public class Program
{
    public void Main(string[] args)
    {
        var context = new InheritanceContext();
        var sarin = new Manager() {Name = "Sarin"};
        var siriphan = new Employee() {Name = "Siriphan", Manager = sarin};
        context.Managers.Add(sarin);
        context.Employees.Add(siriphan);
        context.SaveChanges();
    }
}

public class InheritanceContext : DbContext
{
    public DbSet<Manager> Managers { get; set; }
    public DbSet<Employee> Employees { get; set; }
    protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
    {
        optionsBuilder.UseSqlServer(@"Server=(localdb)\MSSQLLocalDB;Database=Ef7Inheritance;Trusted_Connection=True;");
    }
}

public class Person
{
    public int Id { get; set; }
    public string Name { get; set; }
}

public class Manager : Person
{
    public Collection<Employee> Staffs { get; set; } 
}

public class Employee : Person
{
    public Manager Manager { get; set; }
}

tpc
This code is tested EntityFramework 7.0.0-beta8

awesome!

@wangkanai you are really just getting two separate entity types with an unmapped base type. They each have a separate key space and you won't be able to run a LINQ query on Person. So it's not really inheritance in EF, it's two separate entity types that happen to share a base type in the CLR for implementation.

If their are in same key space would that work well with sql server auto incremental primary key?

@wangkanai not with Identity (since that is per table) but you could use a sequence to generate values. What I really meant is that with true TPC you can't have Manager with Id = 1 and an Employee with Id = 1 since then you would have two instances of Person with Id = 1 (and EF would throw if you every tried to have this). In you setup this is possible since they aren't really in an inheritance hierarchy as far as EF is concerned.

@rowanmiller so if in theory we make a convention to use Id as identity key as Guid (in theory it unique), this should work right?

I'd wanna see TPT before TPC. As @rowanmiller said, @wangkanai example will be used in rare cases when you explicitly wanna keep the entities unmapped or whatever other reason I can't think of now.
But TPT is even more important than TPH, because it can save a bunch of columns generated in each table completely DRYed and less-productive for some purposes.
Lack of TPC made me start a new project in MVC5 and give up the new beta.
Will this be implemented for the RC?

@wangkanai yep that would work. Just to be clear though... with the code you listed back at the start there is no need to ensure that the keys are unique between types since EF is treating them as completely separate types. But for true TPC, yes GUID keys is one option, or a sequence for numeric values, etc.

@weitzhandler I agree. Our database teams design with high normal forms in mind. TPT is what's required. I am of the opinion that TPC is a weak compromise compared to TPT. I often need to query base classes, and TPT makes this easy. A classic example of this is Martin Fowlers paper on Roles. Our clients will not be moving to EF until TPT is implemented.

What is the status on this feature? This turns out to be a really important feature in our case.

The implementation could be simplified if we used a database model #8258

+1 For Me - Note to all that you should "thumbs up" the initial post by @rowanmiller to vote up this issue. This and TPT is how DB's are/should be modeled.

The roadmap lists both TPC and TPT as high priority. Sad to not see either of these tickets in the 2.1 or 3.0 milestone.

Note: see also #10739

Will this ever make it in? 884 days and counting...

https://days.to/18-september/2015

Any news about TPC ?

This issue is in the Backlog milestone. This means that it is not going to happen for the 2.1 release. We will re-assess the backlog following the 2.1 release and consider this item at that time. However, keep in mind that there are many other high priority features with which it will be competing for resources.

@rowanmiller We're using TPC with something like https://github.com/aspnet/EntityFrameworkCore/issues/3170#issuecomment-149801318 but we're getting stuck trying to get one to one relationships to work. We get the following exception:

System.InvalidOperationException: A key cannot be configured on 'BloodPressure' because it is a derived type. The key must be configured on the root type 'Observation'. If you did not intend for 'Observation' to be included in the model, ensure that it is not included in a DbSet property on your context, referenced in a configuration call to ModelBuilder, or referenced from a navigation property on a type that is included in the model.

One to many relationships Works correctly, even considering that TPC is not supported in EF Core. Any idea for that?

@alexdrl If you think you are hitting a bug, then can you please open a new issue including a runnable project/solution/repo or complete code listing that exhibits the behavior?

PostgreSQL as an object-relational database supports inheritance on the table level (https://www.postgresql.org/docs/current/static/tutorial-inheritance.html). I don't know whether other databases also support this kind of inheritance, but it could map very well to C# as it's also single-inheritance, and thus could be exposed in EF Core. It should deliver both optimal speed and space usage in most cases, thus performing better on average than TPC, TPH, TPT or the hackish "Use a single table for the base class, and stash all additional fields from subclasses in a JSON column" way.

@markusschaber PostgreSQL table inheritance is already tracked by #10739.

@roji Thanks for the hint! Subscribing immediately :-)

@ajcvickers The problem was a non-mapped reference to the abstract class that was not mapped to a table, this caused EF Core to acknowledge that table, and throwing an exception because the base class was being tracked in the context.

Based in the example by @wangkanai, this would be something like:

public class Program
{
    public void Main(string[] args)
    {
        var context = new InheritanceContext();
        var sarin = new Manager() {Name = "Sarin"};
        var siriphan = new Employee() {Name = "Siriphan", Manager = sarin};
        context.Managers.Add(sarin);
        context.Employees.Add(siriphan);
        context.SaveChanges();
    }
}

public class InheritanceContext : DbContext
{
    public DbSet<Manager> Managers { get; set; }
    public DbSet<Employee> Employees { get; set; }
    public DbSet<Computer> Computers { get; set; }
    protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
    {
        optionsBuilder.UseSqlServer(@"Server=(localdb)\MSSQLLocalDB;Database=Ef7Inheritance;Trusted_Connection=True;");
    }
}

public class Person
{
    public int Id { get; set; }
    public string Name { get; set; }
}

public class Manager : Person
{
    public Collection<Employee> Staffs { get; set; } 
}

public class Employee : Person
{
    public Manager Manager { get; set; }
}

public class Computer
{
    public string Name { get; set; }
    public Person User { get; set; }
}

What is the timeline on this ? kind of a blocking Item for us

I am having "duplicate primary key values" problem and I have read somewhere that there are two ways to resolve this

  1. Use [DatabaseGenerated(DatabaseGenerationOption.None)] on base class primary key and not map it on child classes at all and manage Id values manually (which I don't want to do)

  2. Use the different initial seed for different types. Now I have 15 classes inherited from abstract BaseEntity class and I couldn't find any example of using the different initial seed for code first approach.

Can anyone please explain it to me how "different initial seed" works and is there any alternative to fix this without having to manage Id manually

Cheers

It continues to sit in the backlog. We had to rewrite everything in 4.7 as this was too much, and we had concerns around other things that may not be finished yet that we would find out later in our development process.

I forgot to add the code for the IDbContextOptions implemation. Maybe something needs to be activily dispoosed here?

```c#
public abstract class DbContextOptionsBase : IDbContextOptions
{
protected readonly DbContextOptionsBuilder Builder = new DbContextOptionsBuilder();

    public virtual DbContextOptions GetOptions(string connectionString)
    {
        Configuring(Builder, connectionString);
        var conventionSet = GetConventionSet();
        return conventionSet == null ? Builder.Options : AddModelBuilder(conventionSet);
    }

    protected virtual void Configuring(DbContextOptionsBuilder optionsBuilder, string connectionString)
    {
        optionsBuilder.UseQueryTrackingBehavior(QueryTrackingBehavior.TrackAll);
        if (string.IsNullOrWhiteSpace(connectionString)) throw new ArgumentNullException("Connection string cannot be null");
        Configuration(optionsBuilder, connectionString.ResolveConnectionString());
    }

    // e.g. optionsBuilder.UseSqlServer(ConnectionString);
    // e.g. optionsBuilder.UseInMemoryDatabase(ConnectionString); 
    protected abstract void Configuration(DbContextOptionsBuilder optionsBuilder, string connectionString);

    protected virtual DbContextOptions AddModelBuilder(ConventionSet conventionSet)
    {
        ModelBuilder = new ModelBuilder(conventionSet);
        ModelBuilder = ModelCreating(ModelBuilder);
        if (ModelBuilder != null)
        {
            Builder.UseModel(ModelBuilder.Model);
        }
        return Builder.Options;
    }

    public ModelBuilder ModelBuilder { get; protected set; }

    protected virtual ConventionSet GetConventionSet()
    {
        return GetConventionSet<DbContext>();
    }

    protected virtual ModelBuilder ModelCreating(ModelBuilder modelBuilder)
    {
        return null;
    }

    protected virtual IEnumerable<IMutableEntityType> GetModelBuilderTypes()
    {
       return ModelBuilder.Model?.GetEntityTypes();
    }

    protected virtual void RemoveModelBuilderTypes(IEnumerable<IMutableEntityType> entityTypes)
    {
        foreach (var entityType in entityTypes)
        {
            ModelBuilder.Model?.RemoveEntityType(entityType);    
        }
    }

    protected virtual ConventionSet GetConventionSet<TDbContext>() where TDbContext : DbContext
    {
        var serviceProvider = GetServiceProviderForConventionSet().BuildServiceProvider();
        using (var serviceScope = serviceProvider.GetRequiredService<IServiceScopeFactory>().CreateScope())
        {
            using (var context = serviceScope.ServiceProvider.GetService<TDbContext>())
            {
                return ConventionSet.CreateConventionSet(context);
            }
        }
    }

    protected abstract IServiceCollection GetServiceProviderForConventionSet();
}

```

In the docs about breaking changes of EF Core 3.0 it says that

Starting with EF Core 3.0 and in preparation for adding TPT and TPC support in a later release

I assume that means TPT / TPC will be supported in 3.1 maybe 3.2?

@MSiffert It does not imply that. It just means we know it's a break we need to make, and making the break sooner is preferable to leaving it like this for longer.

We are now moving a project that uses TPC to EF.Core. It is already built into architecture, is there any workaround we can use before TPC is officially supported? We used MapInheritedProperties(), but we never queried using Base type, as it was abstract. Will ordinary ToTable include inherited properties?

@Grasher134 Using the base type as an "unmapped base class" sounds like it might work for you. This is generally a good approach, but does require that entity types don't reference the base class directly. Mapping inherited properties should not be a problem.

Still no news about it?! 馃様

The text says it pretty clearly - TPT is a planned feature for 5.0, while TPC is currently only considered a stretch goal and may not make it in.

  • 2nd priority based on official high level plan. ref
  • The next planned stable release is EF Core 5.0, scheduled for November 2020.
  • daily builds are also available for download and feedback. ref

For shape of the query projection, consider discussion https://github.com/dotnet/efcore/issues/2266#issuecomment-653661902 and final decision from #21509

Was this page helpful?
0 / 5 - 0 ratings