Web API Controller中的Mapper函数在单元测试期间返回null

如何解决Web API Controller中的Mapper函数在单元测试期间返回null

我正在制作ASP.NET核心Web API,并且使用了Entity Framework Core。我正在尝试使用N-Unit对控制器进行单元测试 我在嘲笑映射器和存储库。这些API可以正常工作,我什至创建了前端或API,但是我在单元测试中遇到了问题。

控制器代码-

[AllowAnonymous]
    [HttpGet("{id}")]
    public async Task<IActionResult> GetMovie(int id)
    {
        var movie = await _repo.GetMovie(id);
        if (movie == null)
        {
            return BadRequest("Object with Id not found");
        }
        var movieToReturn = _mapper.Map<MovieForDetailedDto>(movie);
        return Ok(movieToReturn);
    }

    [AllowAnonymous]
    [HttpPost()]
    public async Task<IActionResult> AddMovie(MovieForDetailedDto movieForDetailedDto)
    { 
        if (await _repo.MovieExists(movieForDetailedDto.ATitle))
            return BadRequest("movie already exists");

        else if(!ModelState.IsValid || movieForDetailedDto.ATitle == null || movieForDetailedDto.APrice == null || movieForDetailedDto.AMovieDescription ==null)
        {
            return BadRequest("movie details not valid");
        }

        var movieToCreate = _mapper.Map<TblMovie>(movieForDetailedDto);

        var createdMovie = await _repo.AddMovie(movieToCreate);

        return Ok(createdMovie);
    }

在我所有的函数中,在将DTO映射到模型的那一行中,该行在单元测试期间返回一个空对象,但在单元测试之外它们仍然可以正常工作。

控制器的我的单元测试代码-

[TestFixture]
public class MoviesControllerTests
{
    private Mock<IMovieRepository> _mockMovieRepository;
    private Mock<IMapper> _mockMovieMapper;
    private MoviesController _moviesController;
    [Test]
    public async Task CallGetRequest_WhenCalledWithId_ReturnsTheMovieWithTheSameId()
    {
        getMoviesHelper getMoviesHelper = new getMoviesHelper();
        List<TblMovie> movieList = getMoviesHelper.getMovieFromList();
        var movie = getMoviesHelper.movieById(3);
        _mockMovieRepository = new Mock<IMovieRepository>();
        _mockMovieMapper = new Mock<IMapper>();
        _mockMovieMapper.Setup(mapper => mapper.Map<TblMovie>(It.IsAny<MovieForDetailedDto>()))
            .Returns(getMoviesHelper.movieById(3));
        _mockMovieRepository.Setup(repo => repo.GetMovie(3))
                .ReturnsAsync(getMoviesHelper.movieById(3));
        _moviesController = new MoviesController(_mockMovieRepository.Object,_mockMovieMapper.Object);
        var tblMovie = await _moviesController.GetMovie(3);
        var okResult = tblMovie as OkObjectResult;
        //Assert.AreEqual(200,okResult.StatusCode);
        Assert.NotNull(tblMovie);
        Assert.IsAssignableFrom<OkObjectResult>(tblMovie);
        var result = ((OkObjectResult)tblMovie).Value;
        var resultValue = ((OkObjectResult)tblMovie).Value as TblMovie;
        Assert.AreEqual(resultValue.ATitle,"Raging Bull");
        Assert.NotNull(result);
        Assert.IsAssignableFrom<TblOrder>(result);
    }

    [Test]
    public async Task GivenAValidMovie_WhenIPostANewMovie_ThenItReturnsOkWithResponse()
    {
        _mockMovieRepository = new Mock<IMovieRepository>();
        _mockMovieMapper = new Mock<IMapper>();
        _mockMovieMapper.Setup(mapper => mapper.Map<TblMovie>(It.IsAny<MovieForDetailedDto>()))
            .Returns(new TblMovie());
        _mockMovieRepository.Setup(repo => repo.AddMovie(It.IsAny<TblMovie>()))
            .ReturnsAsync((TblMovie movie) => movie);

        _moviesController = new MoviesController(_mockMovieRepository.Object,_mockMovieMapper.Object);
        var tblMovie = await _moviesController.AddMovie(new MovieForDetailedDto
        {
            AMovieId = 55,ATitle = "redemption",AMovieDescription = "An action comedy adventure about brilliant robotics prodigy Hiro Hamada,who finds himself in the grips of a criminal plot that threatens to destroy the fast-paced,high-tech city of San Fransokyo. With the help of his closest companion-a robot named Baymax-Hiro joins forces with a reluctant team of first-time crime fighters on a mission to save their city.",ADuration = "105 min",APrice = "10",APurchasePrice = "25",ARating = 5,AImageLink = "http://upload.wikimedia.org/wikipedia/en/4/4b/Big_Hero_6_%28film%29_poster.jpg",ATrailerLink = "//www.youtube.com/embed/z3biFxZIJOQ",AGenre = "Comedy",AWideImage = "https://github.com/tushar23091998/MovieRentalApp-FrontEnd/blob/master/src/app/images/bighero6.jpg?raw=true"
        });
        var okResult = tblMovie as OkObjectResult;
        Assert.AreEqual(200,okResult.StatusCode);
        Assert.NotNull(okResult);
        Assert.IsAssignableFrom<OkObjectResult>(tblMovie);
        var result = ((OkObjectResult)tblMovie).Value;
        Assert.NotNull(result);
        Assert.IsAssignableFrom<TblMovie>(result);
    }

在测试用例和其他测试用例中,回购设置都可以正常工作,但是在定义控制器后,当我调用控制器函数获取值时,在调试中发现控制器代码中的映射器行返回null

我不确定现在如何设置模拟映射器,以及如何传递值。

调试输出-

Mapper return object value during debugging

解决方法

使用AutoMapper时,我们在单元测试中所做的不是模拟它,而是对其进行初始化,如此处所述:https://kenbonny.net/2018/01/15/injecting-automapper-profiles-in-tests/ 我看到这样做的缺点是,由于映射问题,您的测试可能会失败,并且未严格测试该方法。 以您的示例为例,您的设置不正确。 对于get方法测试,您应该设置为TblMovie返回MovieForDetailedDto,但是您要执行相反的操作:

_mockMovieMapper.Setup(mapper => mapper.Map<TblMovie>(It.IsAny<MovieForDetailedDto>())).Returns(getMoviesHelper.movieById(3));

应该是这样的:

var expectedMovieDto = new MovieForDetailedDto(){//insert values here}
_mockMovieMapper.Setup(mapper => mapper.Map<MovieForDetailedDto>(It.IsAny<TblMovie>())).Returns(expectedMovieDto);

对于POST方法,您将其设置为返回新的TblMovie(因此所有属性均设置为其默认值)

_mockMovieMapper.Setup(mapper => mapper.Map<TblMovie>(It.IsAny<MovieForDetailedDto>())) .Returns(new TblMovie());
        

应为:

var expectedMovie = new TblMovie()
{
ADuration = "some value",AGenre = "some value"
//other }

_mockMovieMapper.Setup(mapper => mapper.Map<TblMovie>(It.IsAny<MovieForDetailedDto>())).Returns(expectedMovie);

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 &lt;select id=&quot;xxx&quot;&gt; SELECT di.id, di.name, di.work_type, di.updated... &lt;where&gt; &lt;if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 &lt;property name=&quot;dynamic.classpath&quot; value=&quot;tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-