模拟对象仍然打电话给服务

时间:2013-10-14 16:08:06

标签: c# wcf unit-testing asp.net-mvc-4 justmock

所以我正在为我们的MVC4应用程序编写测试,并且我正在测试Controller操作。正如我在标题中提到的,测试仍然命中服务(WCF)而不是返回测试数据。我有这个控制器:

public class FormController : Controller
{
    public SurveyServiceClient Service { get; set; }
    public SurveyDao Dao { get; set; }

    public FormController(SurveyServiceClient service = null, SurveyDao dao = null)
    {
        this.Service = service ?? new SurveyServiceClient();
        this.Dao = dao ?? new SurveyDao(Service);
    }

    //
    // GET: /Form/

    public ActionResult Index()
    {
        var formsList = new List<FormDataTransformContainer>();
        Dao.GetForms().ForEach(form => formsList.Add(form.ToContainer()));

        var model = new IndexViewModel(){forms = formsList};
        return View("Index", model);
    }

它使用这个DAO对象:

public class SurveyDao
{
    private readonly SurveyServiceClient _service;
    private readonly string _authKey;

    public SurveyDao(SurveyServiceClient serviceClient)
    {
        _service = serviceClient;
    }

    ....

    public FormContract[] GetForms()
    {
        var forms = _service.RetrieveAllForms();
        return forms;
    }

这是我使用JustMock的测试,GetForms()上的mock会在helper类中返回一些测试数据:

[TestClass]
public class FormControllerTest
{
    private SurveyDao mockDao;
    private SurveyServiceClient mockClient;

    public FormControllerTest()
    {
        mockClient = Mock.Create<SurveyServiceClient>();
        mockDao = Mock.Create<SurveyDao>(mockClient);
    }

    [TestMethod]
    public void TestIndexAction() 
    {
        //Arrange
        var controller = new FormController(mockClient, mockDao);
        Mock.Arrange(() => mockDao.GetForms()).Returns(TestHelpers.FormContractArrayHelper);

        //Act
        var result = controller.Index() as ViewResult;

        //Assert
        Assert.IsInstanceOfType(result.Model, typeof(IndexViewModel));
    }
}

我的问题是,当我运行测试时,仍在调用服务。我已经使用Fiddler验证了这一点,并调试了测试并检查了使用我们服务的测试数据填充的“结果”的值。

修改

我已将测试构造函数更改为[TestInitialize]函数,因此Test现在看起来像这样:

[TestClass]
public class FormControllerTest
{
    private SurveyDao mockDao;
    private SurveyServiceClient mockClient;

    [TestInitialize]
    public void Initialize()
    {
        mockClient = Mock.Create<SurveyServiceClient>();
        mockDao = Mock.Create<SurveyDao>(Behavior.Strict);
    }

    [TestMethod]
    public void TestIndexAction() 
    {
        //Arrange
        var controller = new FormController(mockClient, mockDao);
        Mock.Arrange(() => mockDao.GetForms()).Returns(TestHelpers.FormContractArrayHelper);

        //Act
        var result = controller.Index() as ViewResult;

        //Assert
        Assert.IsInstanceOfType(result.Model, typeof(IndexViewModel));
    }
}

2 个答案:

答案 0 :(得分:3)

请确认您使用的是JustMock的正确程序集。有几个不同的(VisualBasic,Silverlight,JustMock)。 JustMock就是您应该在项目中包含的那个。

如果未包含正确的行为,将导致您所描述的行为(方法未正确存根)。

答案 1 :(得分:1)

JustMock manual解释(由我突出显示):

  

默认情况下,Telerik JustMock使用松散模拟并允许您拨打电话   给定类型的任何方法无论方法调用是否正确   安排与否你可以打电话给它。

您可以在调用Mock的Create()方法时控制此行为:

var foo = Mock.Create<IFoo>(Behavior.Strict);

如果您没有明确实现某个方法,可以指定模拟对象应该执行的操作。在你的情况下(我认为这是默认行为),mock确实在你想要模拟的对象上调用原始方法。

Behavior Enumeration枚举中有以下选项:

  • Loose :指定默认情况下模拟调用的行为类似于存根,除非明确设置。
  • RecursiveLoose :指定默认情况下模拟调用将返回模拟对象,除非明确设置。
  • 严格:指定如果未明确设置,对模拟执行的任何调用都将抛出异常。
  • CallOriginal :指定默认情况下在mock上进行的所有调用都将调用其对应的原始成员,除非设置了一些expecations。