javascript - 如何模拟正在测试的服务调用的函数的实现?

标签 javascript typescript jestjs nestjs

我正在开发一个 NestJS 项目,我需要为我的服务编写单元测试。

我有一个名为 BigQueryService 的服务,它使用 @google-cloud/bigquery 访问 Big Query 数据集并执行查询。我还有另一个服务(我们称之为 MyService),其工作是根据其他逻辑构建我需要的查询,并将其传递给 BigQueryService,从中接收查询结果并将其返回给 Controller , Controller 将在然后通过端点发送数据。

我需要为 MyService 编写单元测试,为此我需要以不需要解析 BigQueryService 依赖项的方式模拟 BigQueryService。这是我的一些代码:

bigquery.service.ts:

import { Injectable } from '@nestjs/common';
import { BigQuery } from '@google-cloud/bigquery';
...
@Injectable()
export class BigqueryService {
  ...
  constructor(
    ...
  ) {
    ...
  }

  async executeQuery(querySentence: string): Promise<Array<any>> {
    ...
    return response;
  }
}

MyService.service.ts:

import { Injectable } from '@nestjs/common';
import { BigqueryService } from '../bigquery/bigquery.service';
//the following is just a service that helps log the results of this service
import { MyLogger } from '../../config-service/logger.service';
...
@Injectable()
export class MyService {
  constructor(
    private readonly bigqueryService: BigqueryService,
    private readonly logger: MyLogger,
  ) { }
  ...
  async myFunc(request: RequestInterface): Promise<Array<ResponseInterface>> {
    let query = (some code to create a query i want)
    return await this.bigqueryService.executeQuery(query);
  }

对于测试,我遵循了此线程中的答案:Mock a method of a service called by the tested one when using Jest

jest.mock('../services/bigquery/bigquery.service', () => jest.fn()) 
const bq = require('../services/bigquery/bigquery.service')
jest.mock('../config-service/logger.service', () => jest.fn())
const ml = require('../config-service/logger.service')

const executeQuery = jest.fn()
executeQuery.mockReturnValue('desired value')
bq.mockImplementation(() => ({executeQuery}))


describe("Testing consumption moment service function", () => {

  it("should call the mock service", () => {
    const ms = new MyService(bq,ml)
    ms.myFunc(requestBody) //requestBody is a RequestInterface
    expect(bq.executeQuery).toHaveBeenCalled
    expect(bq.executeQuery).toHaveReturned
 });
});

该测试通过了,所以我假设我正确地模拟了 bigquery 服务。但是,当我尝试断言返回的值是正确的值时,我将测试设为异步,以便在 myFunc 实际运行完成并且我可以比较结果之前测试不会完成。

 it("should call the mock service", async () => {
   const ms = new MyService(bq,ml) 
   await ms.myFunc(requestBody)
   expect(bq.executeQuery).toHaveBeenCalled
   expect(bq.executeQuery).toHaveReturned
 });

这会出现错误:TypeError:this.bigqueryService.executeQuery 不是函数 该错误指向 myFunc 调用 this.bigqueryService.executeQuery 的行。

我尝试了不同的模拟示例,以便可以模拟对此函数的调用,但没有一个像上面的示例那么接近。我也尝试过使用

jest.spyOn(bq, 'executeQuery') 

但这也说明了executeQuery 不是一个函数:无法监视executeQuery 属性,因为它不是一个函数;改为未定义

有人能给我指出正确的方向吗?我是否缺少某些东西来使此测试正常进行?我预先感谢你们能给我的任何帮助。

最佳答案

我最终弄清楚了,所以如果有人遇到同样的情况,我在这里找到了答案:https://jestjs.io/docs/en/jest-object

测试是这样修复的:

jest.mock('../config-service/logger.service', () => jest.fn())
const ml = require('../config-service/logger.service')
const executeQuery = jest.fn()

describe("Testing service function", () => {

  it("should call the mock service", async () => {
    jest.mock('../services/bigquery/bigquery.service', () => {
      return {
        executeQuery: jest.fn(() => 'desired output'),
      };
    })
    const bq = require('../services/bigquery/bigquery.service')

    const ms = new MyService(bq,ml)
    const p = await ms.myFunc(requestBody) //requestBody is a RequestInterface
    expect(bq.executeQuery).toHaveBeenCalled
    expect(bq.executeQuery).toHaveReturned
    expect(p).toEqual(desired result)
 });
});

关于javascript - 如何模拟正在测试的服务调用的函数的实现?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58547135/

相关文章:

javascript - 如何设计 Twitter 弹出窗口

node.js - 函数参数 - id 与对象

html - 使用 "ngIf"进行 Angular 插值

xpath - 如何在 Protractor 出错的情况下返回元素名称?

jestjs - 无论如何要扩展一个 Jest 配置文件?

javascript-单个onclick事件来突出显示不同表格中的单元格

javascript - Angular 种子克隆 "npm start"命令错误

javascript - 使用 Sequelize 通过 object_2_object 表获取相关数据

testing - 为什么 jest 期望匿名函数捕获错误?

unit-testing - Vue-测试工具 | Jest : How to handle dependencies?