Evil 域

当Evil遇上先知

导航

基于Entity Framework 4.1实现一个适用于测试的MockDbContext(上)

Posted on 2011-07-22 14:46  Saar  阅读(1544)  评论(0编辑  收藏  举报

声明:我最近在微软加拿大开发中心工作,这是我个人的博客,跟公司没有关系。如果你在我的博客里看到我推荐微软的产品,就权当广告好了。

我们在作一些CRUD相关的单元测试的时候,通常不会真的连接到数据库,而是写一个Mock的Repository,把Entity放到一个集合啊或者Hash table啊什么的里面。
最近用Entity Framework 4.1写点小项目,在写一个Mock的Repository的时候还走了些弯路,费了一些时间,在此把过程写出来,希望能帮大家节省一点时间。
项目中用的是Model First,在Database First模型中应该也适用,先设计完数据库模型,生成数据库和两个.tt文件——MTBDbContext.Context.tt和MTBDbContext.tt。第一个文件里是一个DbContext类,担任数据持久化操作;第二个文件里是实体类。这两个文件原来与数据库模型是在同一个项目中的,做了一些小动作,把它们分开了^_^。

image

根据这样的结构,写MockDbContext的思路是这样子的:以添加一个实体类为例,我们通常会写这样的代码:

        public void AddBatch(Handbook handbook)
        {
            dbContext.Set<Handbook>().Add(handbook);
            dbContext.SaveChanges();
        }

如果直接调用EF4.1的DbContext,在调用 DbContext里的SaveChanges(),数据就会被固化到数据库里。但我们相信,只要数据能够在本地保存,通过DbContext,它就一定会存到数据库里,因此,测试时没有必要把数据库写到数据库中去,只要在本地进行验证。
综上,我们需要的MockDbContext只要满足两个条件:第一,SaveChanges()不把数据固化到数据库,而是存在本地;第二,可以在本地作数据验证。
我们首先来满足第一个条件。打开DbContext看一下:

    public partial class MTBContainer : DbContext
    {
        public MTBContainer()
            : base("name=MTBContainer")
        {
        }
    
        protected override void OnModelCreating(DbModelBuilder modelBuilder)
        {
            throw new UnintentionalCodeFirstException();
        }
    
        public DbSet<Handbook> Handbooks1 { get; set; }
        public DbSet<Trip> Trips { get; set; }
        // … more DbSet<T>...
    }

这个类信息不多,因此推测逻辑都在其基类DbContext里。因此,在Object Browser里打开DbContext看了一下:

    public class DbContext : IDisposable, IObjectContextAdapter
    {
        protected DbContext();
           //… more constructors
          // Other code ...
        public virtual int SaveChanges();
        public DbSet<TEntity> Set<TEntity>() where TEntity : class;
        public DbSet Set(Type entityType);
                  //...
    }

里面有一个SaveChanges()方法,而且还是虚的。这样,事情就简单了(起初是这样认为的,也是从这里开始走的弯路):重写一下SaveChanges()方法,让它什么都不做,它不就不会把数据存回数据库了吗?OK,第一个条件基本达成。

第二个条件,本地验证。由于实体类对象都保存在DbSet<T>里,而DbSet<T>里有一个ObservableCollection<T>类型的Local属性保存的正是本地实体类对象(这个理解有问题的)。这样,事情就好办了,我们写测试验证的时候直接验证这里的结果就行了。

按着这个思路,复制粘贴了一份MTBDbContext.Context.tt,改名为MockMTBDbContext.Context.tt并且放到了对应的单元测试项目中。然后,稍微修改了一下模板:添加了一些using的命名空间,改了生存的类名和构造函数名,然后就是重点添加一个什么都不做的SaveChanges()的重写方法——保存,自动生成代码如下:

...         
    public partial class MockMTBContainer : DbContext, ITestableDbContext
    {
        public MockMTBContainer()
            : base("name=MTBContainer")
        {
        }
    
        public override int SaveChanges()
        {
            //Do nothing
            return 0;
        }
    
        ...    
        public DbSet<Handbook> Handbooks1 { get; set; }
        public DbSet<Trip> Trips { get; set; }
    }

接下来,就写了一个测试代码来看看一个基本操作:

        /// <summary>
        ///A test for Add
        ///</summary>
        [TestMethod()]
        public void AddTest()
        {
            var mockDbContext = new MockMTBContainer();
            BizHandbook target = new BizHandbook(mockDbContext); // use mockDbContext here.
            Handbook handbook = new Handbook();
            target.Add(handbook);
            Assert.AreEqual<int>(1, mockDbContext.Handbooks1.Local.Count); // see whether the entity object's added
        }

跑Case,成功Pass。哈。没想到这么轻松。

但是…

但是…

CRUD四项操作中,CUD都可以测试,R(Retrieve)的时候,却怎么也得不到结果……添加的记录在Local可以看到,但是.ToList()的时候怎么都是null。

 

=====

下篇解释了错误的原因,并且重新设计了一个接口以完成目标。点击继续...