How to switch between DatabaseGeneratedOption.Identity, Computed and None at runtime without having to generate empty DbMigrations

后端 未结 1 1015
终归单人心
终归单人心 2021-01-02 17:48

I am migrating a legacy database to a new database which we need to access and \"manage\" (as oxymoronic as it might sound) primarily through Entity Framework Code-First.

相关标签:
1条回答
  • 2021-01-02 18:11

    A certain amount of the configuration of the context is always going to be dependent on the runtime environment - for example, proxy generation and validation. As such, runtime configuration of the Entity Framework DbContext is something I leverage quite heavily.

    Although I've never used this approach to switch the configuration of the context on a per use-case basis, I see no reason why this would not work.

    In its simplest form, this can be achieved by having a set of EntityTypeConfiguration classes for each environment. Each configuration set is then wired to the DbContext on a per-environment basis. Again, in its simplest form this could be achieved by having a DbContext type per environment. In your case, this would be per use-case.

    Less naively, I usually encapsulate the configuration of the context in an environment-specific unit of work. For example, the unit of work for an Asp.Net environment has an underlying DbContext configured to delegate validation to the web framework, as well as to turn off proxy generation to prevent serialisation issues. I imagine this approach would have similar usefulness to your problem.

    For example (using brute force code):

    // Foo Configuration which enforces computed columns
    public class FooConfiguration : EntityTypeConfiguration<Foo>
    {
        public FooConfiguration()
        {           
            Property(p => p.DateTime).HasDatabaseGeneratedOption(DatabaseGeneratedOption.Computed);
            Property(p => p.Guid).HasDatabaseGeneratedOption(DatabaseGeneratedOption.Computed);
        }
    }
    
    // Foo configuration that allows computed columns to be overridden
    public class FooConfiguration2 : EntityTypeConfiguration<Foo>
    {
        public FooConfiguration2()
        {           
            Property(p => p.DateTime).HasDatabaseGeneratedOption(DatabaseGeneratedOption.None);
            Property(p => p.Guid).HasDatabaseGeneratedOption(DatabaseGeneratedOption.None);
        }
    }
    
    // DbContext that enforces computed columns
    public class MyContext : DbContext
    {
        protected override void OnModelCreating(DbModelBuilder modelBuilder)
        {
            modelBuilder.Configurations.Add(new FooConfiguration());     
        }
    }
    
    // DbContext that allows computed columns to be overridden
    public class MyContext2 : DbContext
    {
        protected override void OnModelCreating(DbModelBuilder modelBuilder)
        {
            modelBuilder.Configurations.Add(new FooConfiguration2());     
        }
    }
    

    This can obviously be tidied up - we usually use a combination of factory and strategy patterns to encapsulate the creation of a runtime specific context. In combination with a DI container this allows the correct set up configuration classes to be injected on a per-environment basis.

    Example usage:

    [Fact]
    public void CanConfigureContextAtRuntime()
    {
        // Enforce computed columns
        using (var context = new EfContext())
        {
            var foo1 = new Foo();
            context.Foos.Add(foo1);                             
            context.SaveChanges();
        }
    
        // Allow overridden computed columns
        using (var context = new EfContext2())
        {              
            var foo2 = new Foo { DateTime = DateTime.Now.AddYears(-3) };
            context.Foos.Add(foo2);
            context.SaveChanges();
        }
    
        // etc
    }
    
    0 讨论(0)
提交回复
热议问题