基于Entity Framework 4.1实现一个适用于测试的MockDbContext(上)
Posted on 2011-07-22 14:46 Saar 阅读(1555) 评论(0) 收藏 举报声明:我最近在微软加拿大开发中心工作,这是我个人的博客,跟公司没有关系。如果你在我的博客里看到我推荐微软的产品,就权当广告好了。    
我们在作一些CRUD相关的单元测试的时候,通常不会真的连接到数据库,而是写一个Mock的Repository,把Entity放到一个集合啊或者Hash table啊什么的里面。   
最近用Entity Framework 4.1写点小项目,在写一个Mock的Repository的时候还走了些弯路,费了一些时间,在此把过程写出来,希望能帮大家节省一点时间。    
项目中用的是Model First,在Database First模型中应该也适用,先设计完数据库模型,生成数据库和两个.tt文件——MTBDbContext.Context.tt和MTBDbContext.tt。第一个文件里是一个DbContext类,担任数据持久化操作;第二个文件里是实体类。这两个文件原来与数据库模型是在同一个项目中的,做了一些小动作,把它们分开了^_^。
根据这样的结构,写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。
=====
下篇解释了错误的原因,并且重新设计了一个接口以完成目标。点击继续...
Little knowledge is dangerous.
 
                    
                
 
                
            
         浙公网安备 33010602011771号
浙公网安备 33010602011771号