5

我刚刚开始使用 RavenDB,到目前为止我很喜欢它。然而,我坚持如何对与之交互的控制器操作进行单元测试。

我发现的所有问题/文章都像这样:单元测试 RavenDb 查询告诉我应该在内存中使用 RavenDB 而不是模拟它,但我找不到一个可靠的例子来说明这是如何完成的。

例如,我有一个控制器操作来将员工添加到数据库(是的,它过于简化了,但我不想让问题复杂化)

public class EmployeesController : Controller
{

  IDocumentStore _documentStore;
  private IDocumentSession _session;

  public EmployeesController(IDocumentStore documentStore)
  {
    this._documentStore = documentStore;

  }

  protected override void OnActionExecuting(ActionExecutingContext filterContext)
  {
    _session = _documentStore.OpenSession("StaffDirectory");
  }

  protected override void OnActionExecuted(ActionExecutedContext filterContext)
  {
      if (_session != null && filterContext.Exception == null) {
        _session.SaveChanges();
        _session.Dispose();
    }
  }

  [HttpGet]
  public ViewResult Create()
  {
    return View();
  }

  [HttpPost]
  public RedirectToRouteResult Create(Employee emp)
  {
    ValidateModel(emp);
    _session.Store(emp);
    return RedirectToAction("Index");
  }

如何验证单元测试中添加到数据库的内容?有没有人有任何在 MVC 应用程序中涉及 RavenDb 的单元测试示例?

如果这很重要,我正在使用 MSTest,但我很乐意尝试翻译来自其他框架的测试。

谢谢。

编辑

好的,我的测试初始化​​创建了注入到控制器构造函数中的文档存储,但是当我运行我的测试时,OnActionExecuting 事件没有运行,因此没有要使用的会话,并且测试失败并出现空引用异常。

[TestClass]
public class EmployeesControllerTests
{
  IDocumentStore _store;

  [TestInitialize]
  public void InitialiseTest()
  {
    _store = new EmbeddableDocumentStore
    {
      RunInMemory = true
    };
    _store.Initialize();
  }

  [TestMethod]
  public void CreateInsertsANewEmployeeIntoTheDocumentStore()
  {
    Employee newEmp = new Employee() { FirstName = "Test", Surname = "User" };

    var target = new EmployeesController(_store);
    ControllerUtilities.SetUpControllerContext(target, "testUser", "Test User", null);

    RedirectToRouteResult actual = target.Create(newEmp);
    Assert.AreEqual("Index", actual.RouteName);

    // verify employee was successfully added to the database.
  }
}

我错过了什么?如何创建会话以在测试中使用?

4

3 回答 3

7

运行单元测试后,只需断言数据库中有一个新文档并且它设置了正确的字段。

var newDoc = session.Load<T>(docId)

或者

var docs = session.Query<T>.Where(....).ToList();

存在 RavenDB 内存模式,因此您不必模拟它,您只需执行以下操作:

  • 打开一个新的内存嵌入式文档存储(没有数据)
  • 如果需要,插入您的单元测试需要运行的任何数据
  • 运行单元测试
  • 查看内存存储中的数据,看看它是否已正确更新

更新如果你想要一个完整的示例,看看 RacoonBlog 代码是如何做到的,这是运行Ayende 博客的代码。查看这两个文件:

于 2012-05-02T22:34:12.977 回答
3

如何验证单元测试中添加到数据库的内容?

你没有。我们不在单元测试中测试这些东西。这是集成测试的责任,而不是单元测试。

如果您想对依赖于某些外部源(如您的数据库)的类进行单元测试,请模拟数据库访问。

编辑:

为了纠正一些提到的错误,我将引用 MSDN 的定义(但是所有其他资源都同意这一点):

单元测试的主要目标是获取应用程序中最小的可测试软件,将其与代码的其余部分隔离开来,并确定它的行为是否完全符合您的预期。

如果不嘲笑,您将忽略单元测试的基本原则——隔离和测试尽可能小的部分。单元测试需要持久无知,不应该依赖于某些外部类。如果数据库随时间变化怎么办?重写所有测试,即使功能完全一样?

来吧。你可以给我 -1 你想要多少次,但这不会让你正确。

于 2012-05-02T22:00:05.920 回答
1

由于您提到的链接到的线程通过嵌入 RavenDB 来使用 EmbeddableDocumentStore。

以下是设置方法:http: //msdn.microsoft.com/en-us/magazine/hh547101.aspx

以下是如何将存储库模式与 raven 一起使用,以便您轻松测试:http: //novuscraft.com/blog/ravendb-and-the-repository-pattern

于 2012-05-02T21:50:30.820 回答