测试扩展类时的 ES6 基类(超级方法)的玩笑模拟方法

Posted

技术标签:

【中文标题】测试扩展类时的 ES6 基类(超级方法)的玩笑模拟方法【英文标题】:Jest mock method of base ES6 class (super method) when testing extended class 【发布时间】:2021-03-31 03:13:10 【问题描述】:

在测试扩展类时,我在测试使用某些参数调用原始方法(来自基类)时遇到问题。我要测试的类是:

// ApiDataSource.js
import  RESTDataSource  from "apollo-datasource-rest";

export default class ApiDataSource extends RESTDataSource 
  constructor() 
    super();
    this.baseURL = 'test';
  

  //We add the authorization params to the get method
  async get(path, params = , init = ) 
    return super.get(
      path,
      
        ...params,
        app_id: "test app id",
        app_key: "test app key",
      ,
      init
    );
  


基本上,我想模拟 super.get() 以断言当调用 ApiDataSource.get() 时,使用授权参数调用 super 方法。

类似:

// ApiDataSource.test.js
import ApiDataSource from './ApiDataSource'

// ...
test("adds the authorization parameters to the get call", async () => (
  const class = new ApiDataSource();
  await class.get("test");

  expect(mockedSuperGet).toHaveBeenCalledWith("test", app_id: "test app id", app_key: "test app key")
);

知道怎么做吗?已经试过jest.mock()jest.spyOn等等,我似乎无法得到它......

代码沙盒: https://codesandbox.io/s/example-test-api-data-source-6ldpk?file=/src/ApiDataSource.test.ts

【问题讨论】:

当然,只是模拟/窥探RESTDataSource.prototype.get? 您能否提供一个关于如何做到这一点的示例@Bergi(基于上述代码)? 我手头没有,不,但请参阅他们的文档。你能不能edit你的问题包括你的尝试? 已经快速编译了这个@Bergi:codesandbox.io/s/example-test-api-data-source-6ldpk?file=/src/… jest.mock() 似乎有问题,但也许你明白了......还尝试用jest.mock("apollo-datasource-rest", () => ( RESTDataSource: jest.fn()) 模拟整个班级或者其他什么,但也没有任何成功...... 你不要嘲笑超类。继承的行为是行为的一部分。取而代之的是模拟合作者 【参考方案1】:

您可以使用jest.spyOn(object, methodName) 为RESTDataSource.prototype.get 方法创建一个模拟函数。

例如

ApiDataSource.js:

import  RESTDataSource  from 'apollo-datasource-rest';

export default class ApiDataSource extends RESTDataSource 
  constructor() 
    super();
    this.baseURL = 'test';
  

  async get(path, params = , init = ) 
    return super.get(
      path,
      
        ...params,
        app_id: 'test app id',
        app_key: 'test app key',
      ,
      init
    );
  

ApiDataSource.test.js:

import  RESTDataSource  from 'apollo-datasource-rest';
import ApiDataSource from './ApiDataSource';

describe('65391369', () => 
  test('adds the authorization parameters to the get call', async () => 
    const mockedSuperGet = jest.spyOn(RESTDataSource.prototype, 'get').mockResolvedValueOnce('fake data');
    const apiDataSource = new ApiDataSource();
    await apiDataSource.get('test');
    expect(mockedSuperGet).toHaveBeenCalledWith('test',  app_id: 'test app id', app_key: 'test app key' , );
    mockedSuperGet.mockRestore();
  );
);

单元测试结果:

 PASS  examples/65391369/ApiDataSource.test.js (5.273 s)
  65391369
    ✓ adds the authorization parameters to the get call (4 ms)

------------------|---------|----------|---------|---------|-------------------
File              | % Stmts | % Branch | % Funcs | % Lines | Uncovered Line #s 
------------------|---------|----------|---------|---------|-------------------
All files         |     100 |      100 |     100 |     100 |                   
 ApiDataSource.js |     100 |      100 |     100 |     100 |                   
------------------|---------|----------|---------|---------|-------------------
Test Suites: 1 passed, 1 total
Tests:       1 passed, 1 total
Snapshots:   0 total
Time:        6.28 s

【讨论】:

一些关于这是否是好的做法的评论可能会有所帮助。仅仅因为你可以并不一定意味着你应该

以上是关于测试扩展类时的 ES6 基类(超级方法)的玩笑模拟方法的主要内容,如果未能解决你的问题,请参考以下文章

使用实体基类时的 EF CTP5 映射问题

构造虚拟基类时的编译器行为

在不同的命名空间中定义基类和派生类时的编译错误

扩展抽象测试类时找不到测试类

玩笑:使用默认和命名导出模拟 ES6 模块

使用 TypeScript 装饰器扩展 ES6 类时扩展类型