视图模型到领域模型,映射测试应该放在哪里?

3
我目前在一个MVC项目中尝试使用类似于自动映射器的映射框架。

在我的控制器中,我使用框架的扩展方法“Map”,将视图模型映射到域模型(似乎是合法的地方)。

当然,如果映射出现错误(例如某些属性名称更改导致不匹配),我的代码将无法正常工作。

但是,应该在哪里测试映射是否按预期工作呢?

控制器不是负责这个“单元”的。 视图模型和域模型也不是。

我认为我可以创建自己的包装器(IoC可注入实例),并使用视图模型到域模型的映射对其进行单元测试,但感觉有点尴尬(代码读者如何知道需要在特定的类上执行这些测试?)。

我有点迷失了。

编辑:(对John Mc答案的反思)

using System;
using NSubstitute;
using Models = TestEncodeLines.Models;
using Controllers = TestEncodeLines.Controllers;
using ViewModels = TestEncodeLines.ViewModels;
using Infrastructure = TestEncodeLines.Infrastructure;
using Xunit;

namespace Tests.TestControllers
{
    public class TestActivityController
    {
        private Controllers.ActivityController _controller;

        public TestActivityController()
        {
            _controller = new Controllers.ActivityController();
        }

        [Fact]
        public void Save_Project()
        {
            // Arrange
            var viewModel = new ViewModels.ActivitiesViewModel();
            var model = Substitute.For<Models.IActivitiesModel>();
            var mapper =
                Substitute.For<Infrastructure.IMapper<ViewModels.ActivitiesViewModel, Models.IActivitiesModel>>();
            mapper.Map(viewModel).Returns(model);

            // Act
            _controller.SaveActivities(viewModel);

            // Assert
            model.Received().Save();
        }

        [Fact]
        public void Save_Project_TestMapping /* Here ??? */ ()
        {
            // Arrange
            var viewModel = new ViewModels.ActivitiesViewModel
            {
                Activities = new[]
                {
                    new ViewModels.ActivitiesViewModel.Project
                    {
                        From = new DateTime(2016, 02, 23, 8, 0, 0, DateTimeKind.Utc),
                        To = new DateTime(2016, 02, 23, 10, 0, 0, DateTimeKind.Utc),
                        Name = "Test"
                    }
                },
                Date = new DateTime(2016, 02, 23, 0, 0, 0, DateTimeKind.Utc)
            };
            var mapper = new Infrastructure.Mapper<ViewModels.ActivitiesViewModel, Models.IActivitiesModel>();

            // Act
            _controller.SaveActivities(viewModel);

            // Assert
            // Somehow (https://github.com/jamesfoster/DeepEqual ??) check the mapping
        }
    }
}

1
在您当前的控制器测试中,您如何处理映射?您是模拟它还是实际调用它?如果您实际调用它,那么如果映射出现问题,您现有的控制器测试应该会失败。基本上,ViewModel进入,Model进入依赖项,然后返回时反转。如果任何映射失败,您现有的交互都应该失败... - forsvarir
嘲弄它。在我贴在这里的代码中,我添加了另一个(不完整且可能不合适)测试来调用真实的东西。 - Serge Intern
1个回答

1
我将实现映射框架抽象为 IMappingService,并通过构造函数注入。这样,您可以确保在控制器单元测试中进行必要的调用以调用映射框架。
至于测试映射,Automapper 有一个配置选项,可以确保您配置的映射文件不会失败:
AutoMapper.AssertConfigurationIsValid()

这并不意味着映射是正确的。

你为什么不能创建专门针对映射部分的映射特定单元测试?为什么不在单元测试中实例化你的源对象,并尝试将其映射到目标对象,然后在那里断言它们的正确性呢?


你是在建议我按照我在问题中添加的(第二个测试)代码所做的吗? - Serge Intern
我会为每个映射配置文件创建一个专用的单元测试类。你正在测试映射而不是控制器的功能。 - John Mc

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接