繁体   English   中英

Typescript/Jest mocking 来自服务的共享 function

[英]Typescript/Jest mocking a shared function from a service

我有一个 NestJS 服务,但这并不是真正的 NestJS 问题。 然而,NestJS 测试正在进行中,我用它来创建测试服务等。我的服务用于进行多次调用,并返回所有数据。

在我的场景中,有一个服务 (myservice) 将获得一些结果(object 的数据),它结合了来自各种来源的结果。 外部源都是基于 Promise (async/await),用于获取所有不同的数据点,将其组合到 myservice(由其他人扩展的基础服务)以组合错误处理,promise 结果处理等.

我在尝试模拟外部测试调用时遇到问题。 我的问题是从 NodeJS 读取数据的 function。 我可以毫无问题地自行测试 function,但现在我正试图在我的服务中模拟这个 function 调用。

问题中的函数的简短示例,作为 myservice.ts 的一部分

public async GetData(DataType: number = DATA_TYPES.TEXT): Promise<ResultFile> {
    const ResultData: ResultFile = new ResultFile(DataType);

    return new Promise<ResultFile>((resolve, reject) => {
        const Data$: Promise<string> = this.GetResultFileContents();
        const StatusCode$: Promise<number> = this.GetResultFileStatusCode();

        Promise.all([Data$, StatusCode$])
            .then((Results) => {
                ResultData.Contents = Results[0]; // Promise returns a string
                ResultData.StatusCode = Results[1]; // Promise returns a number

                resolve(ResultData);
            })
            .catch((Exception: Error) => {
                ResultData.StatusCode = HttpStatus.NOT_FOUND;
                ResultData.Contents = Exception.message;
                reject(ResultData);
            });
    });
}

以上是尝试检索不同数据的主要方法。 这调用的承诺比那里的 2 个更多,但有两个会显示我的问题。

public async GetResultFileContents(): Promise<string> {
    try {
        const Results$: string = await ReadFileContents(this.Directory_, this.BaseName_);
        return Results$;
    } catch (Exception) {
        throw new HttpException(`File not found: ${this.BaseName_} in ${this.Directory_}`, HttpStatus.NOT_FOUND);
    }
}

public async GetResultFileStatusCode(): Promise<number> {
    let StatusCode: number = 0;
    try {
        const StatusCodeFile: string = `${this.BaseName_}.err`;
        const StatusCodeData$ = await ReadFileContents(this.Directory_, StatusCodeFile);
        StatusCode = GetIntegerFromText(StatusCodeData$);
    } catch (Exception) {
        StatusCode = HttpStatus.OK; // Return OK since a specific status was not set to be returned.
    }
    return new Promise<number>((resolve) => {
        resolve(StatusCode);
    });
}

被调用的两个方法返回承诺,都使用外部 function,ReadFileContents()。 这是我想要模拟的 function,因为它可以将数据作为字符串返回,或者抛出异常,为包含数据的文件包装操作系统检查(除其他外)。

这个 function 是通用的,并且由许多从文件系统读取数据的方法共享。 还有一些类似于 REST 调用的东西,它们有同样的问题,但这是一个简单的例子。

我的问题现在出现在测试文件中。 虽然我可以测试服务和myservice.ts中的方法,但我不知道如何模拟对 ReadFileContents() 的外部调用以确保我在myservice.ts中的方法正确执行。

我想测试不同的返回字符串,以及文件不存在时的异常捕获等。

我的测试:

import { Test, TestingModule } from '@nestjs/testing';

import { MyService } from './my.service';

// TODO: Need to mock the internal calls to ReadFileContents() to throw exceptions
describe('MyService (mock)', () => {
    let service: MyService;

    afterEach(() => {});

    beforeEach(async () => {
        const module: TestingModule = await Test.createTestingModule({
            providers: [MyService],
        }).compile();

        service = module.get<MyService>(MyService);
    });

    it('should be defined', () => {
        expect(service).toBeDefined();
    });

    it('should handle exceptions when reading missing .hdr file', async () => {
        // const mockReadFileContents = jest.spyOn(service.???);
    });
});

在最后一个 function 中,我不知道如何模拟 ReadFileContents,因为它只是服务中的一个 function,它位于另一个源文件中。

我真的不想在服务中创建一个公共方法,只是调用这个 function,所以我可以模拟它,如果我能帮忙的话。

请忽略缺失的功能或其他拼写错误,因为这是一个快速编辑以提供我正在尝试完成的示例。

我一直在为此努力,但无法让模拟工作。 然后我确实更改了我试图模拟的功能,使其位于另一个服务 (fsService) 中,并通过具有依赖注入的构造函数在 myService 中使用该服务。

constructor(
    private FileSystem_: fsService,
) {}

然后可以通过 mocking 服务在测试中轻松模拟,提供虚假定义({provide: fsService, useClass: fsServiceMock })等。

import { Test, TestingModule } from '@nestjs/testing';
import { MyService } from './my.service';
import { FsService } from './fs.service';

describe('MyService (mock)', () => {
    let service: MyService;
    let FSService_: FsService;

    afterEach(() => {
        jest.resetAllMocks();
    });

    beforeEach(async () => {
        const module: TestingModule = await Test.createTestingModule({
            providers: [
                FSService,
                MyService,
            ],
        }).compile();
        FsService_ = module.get<FSService>(FSService);
        service = module.get<MyService>(MyService);
    });

    it('should be defined', () => {
        expect(service).toBeDefined();
    });

    it('should handle exceptions when reading missing .hdr file', async () => {
        FSService_.ReadFileContents = jest.fn().mockRejectedValue(new Error('ENOENT: File not Found'));
        ...
    });
});

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM