[英]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.