基础

优质
小牛编辑
129浏览
2023-12-01

用户提供商

当你想直接绑定到控制容器的Nest反转时,有很多场景。例如,任何常量值,基于当前环境创建的配置对象,外部库或预先计算的值(取决于其他几个定义的提供程序)。此外,您可以覆盖默认实现,例如在需要时使用不同的类或使用各种测试双打(用于测试目的)。

你应该始终牢记的一件重要事情就是Nest使用tokens(口令)来标识依赖关系。通常,自动生成的标记等于类。如果你想创建一个自定义提供者,你需要选择一个令牌。大多数情况下,自定义令牌都由纯字符串表示。遵循最佳实践,您应该在分隔文件中保存这些标记,例如,在constants.ts中。

我们来看看可用的选项。

使用值

useValue语法在定义常量值,将外部库放入Nest容器或用模拟对象替换实际实现时非常有用。

  1. import { connection } from './connection';
  2. const connectionProvider = {
  3. provide: 'Connection',
  4. useValue: connection,
  5. };
  6. @Module({
  7. providers: [connectionProvider],
  8. })
  9. export class ApplicationModule {}

为了注入自定义用户提供,我们使用@Inject()装饰器。这个装饰器接受一个参数 - 令牌。

  1. @Injectable()
  2. class CatsRepository {
  3. constructor(@Inject('Connection') connection: Connection) {}
  4. }

!> @Inject()装饰器从@nestjs/common包中导入。

当您想要覆盖默认提供者的值时,比方说,您想强制Nest使用模拟CatsService以进行测试,您可以简单地使用现有类作为标记。

  1. import { CatsService } from './cats.service';
  2. const mockCatsService = {};
  3. const catsServiceProvider = {
  4. provide: CatsService,
  5. useValue: mockCatsService,
  6. };
  7. @Module({
  8. imports: [CatsModule],
  9. providers: [catsServiceProvider],
  10. })
  11. export class ApplicationModule {}

在上面的例子中,CatsService将被传递的mockCatsService模拟对象覆盖。这意味着,Nest不是手动创建CatsService实例,而是将此提供者视为已解决,并使用mockCatsService作为其代表值。

使用类

useClass语法允许您对每个选定的因素使用不同的类。例如,我们有一个抽象的(或默认的)ConfigService类。根据当前环境,Nest应该使用不同的配置服务实现。

  1. const configServiceProvider = {
  2. provide: ConfigService,
  3. useClass: process.env.NODE_ENV === 'development'
  4. ? DevelopmentConfigService
  5. : ProductionConfigService,
  6. };
  7. @Module({
  8. providers: [configServiceProvider],
  9. })
  10. export class ApplicationModule {}

!>我们使用了ConfigService类,而不是自定义标记,因此我们已经覆盖了默认的实现。

在这种情况下,即使任何类依赖于ConfigService,Nest也会注入提供的类的实例(DevelopmentConfigServiceProductionConfigService)。

使用工厂

useFactory是一种动态创建提供程序的方式。实际提供者将等于工厂函数的返回值。工厂功能可以依靠几个不同的提供商或保持完全独立。这意味着工厂可以接受参数,Nest将在实例化过程中解析并传递参数。另外,这个函数可以异步返回值。这里点击 更多详解。当你的提供者不得不被动态的计算或者为了解决异步操作时可以使用它。

  1. const connectionFactory = {
  2. provide: 'Connection',
  3. useFactory: (optionsProvider: OptionsProvider) => {
  4. const options = optionsProvider.get();
  5. return new DatabaseConnection(options);
  6. },
  7. inject: [OptionsProvider],
  8. };
  9. @Module({
  10. providers: [connectionFactory],
  11. })
  12. export class ApplicationModule {}

?> 如果您的工厂需要其他提供者,则必须将其标记传入inject数组中。 Nest会以相同顺序将实例作为函数的参数传递。

导出自定义提供者

为了导出自定义提供者,我们可以使用令牌或整个对象。以下示例显示了一个标记案例:

  1. const connectionFactory = {
  2. provide: 'Connection',
  3. useFactory: (optionsProvider: OptionsProvider) => {
  4. const options = optionsProvider.get();
  5. return new DatabaseConnection(options);
  6. },
  7. inject: [OptionsProvider],
  8. };
  9. @Module({
  10. providers: [connectionFactory],
  11. exports: ['Connection'],
  12. })
  13. export class ApplicationModule {}

但是你也可以使用整个对象:

  1. const connectionFactory = {
  2. provide: 'Connection',
  3. useFactory: (optionsProvider: OptionsProvider) => {
  4. const options = optionsProvider.get();
  5. return new DatabaseConnection(options);
  6. },
  7. inject: [OptionsProvider],
  8. };
  9. @Module({
  10. providers: [connectionFactory],
  11. exports: [connectionFactory],
  12. })
  13. export class ApplicationModule {}

支持我们

Nest是一个MIT许可的开源项目。它可以发展得益于这些令人敬畏的人们的支持。如果你想加入他们,请阅读 更多。

异步提供者

例如,在完成一些异步任务之前,应用程序启动必须被延迟,直到与数据库的连接建立,您应该考虑使用异步提供程序。为了创建一个异步提供者,我们使用useFactory。工厂必须返回Promise(因此异步函数也适合)。

  1. {
  2. provide: 'AsyncDbConnection',
  3. useFactory: async () => {
  4. const connection = await createConnection(options);
  5. return connection;
  6. },
  7. },

?> 在这里了解 更多关于定制提供商的语法。

异步提供者可以通过它们的标记(在上述情况下,通过AsyncDbConnection标记)简单地注入其他组件。一旦异步提供程序已解析,每个依赖于异步提供程序的类都将被实例化

以上示例用于演示目的。如果你正在寻找更详细的,请看 这里。

循环依赖

例如,当A类需要B类,而B类也需要A类时,就会产生循环依赖。 Nest允许在提供者和模块之间创建循环依赖关系,但我们建议您尽可能避免。有时候避免这种关系真的很难,这就是为什幺我们提供了一些方法来解决这个问题。

正向引用

正向引用允许Nest引用目前尚未被定义的参考。当CatsServiceCommonService相互依赖时,关系的两端需要使用@Inject()forwardRef()实用工具,否则Nest不会实例化它们,因为所有基本元数据都不可用。让我们看看下面的代码片段:

cats.service.ts

  1. @Injectable()
  2. export class CatsService {
  3. constructor(
  4. @Inject(forwardRef(() => CommonService))
  5. private readonly commonService: CommonService,
  6. ) {}
  7. }

?> forwardRef()函数是从@nestjs/common包中导入的。

这是第一层的关系。现在让我们对CommonService进行相同的操作:

common.service.ts

  1. @Injectable()
  2. export class CommonService {
  3. constructor(
  4. @Inject(forwardRef(() => CatsService))
  5. private readonly catsService: CatsService,
  6. ) {}
  7. }

!> 你不能保证哪个构造函数会被首先调用。

为了在模块之间创建循环依赖关系,必须在模块关联的两个部分上使用相同的forwardRef()实用程序:

common.module.ts

  1. @Module({
  2. imports: [forwardRef(() => CatsModule)],
  3. })
  4. export class CommonModule {}

模块参考

Nest提供了可以简单地注入到任何组件中的ModuleRef类。

cats.service.ts
```typescript
@Injectable()
export class CatsService implements OnModuleInit {
private service: Service;
constructor(private readonly moduleRef: ModuleRef) {}

onModuleInit() {
this.service = this.moduleRef.get(Service);
}
}

  1. ?> `ModuleRef`类是从`@nestjs/core`包中导入的。
  2. 模块引用有一个`get()`方法,它允许检索当前模块中可用的任何组件。
  3. ## 平台不可知论
  4. Nest的整个观点是作为一个平台不可知的框架。平台独立性使得**创建可重用的逻辑部分**成为可能,人们可以利用多种不同类型的应用程序。框架的架构专注于适用于任何类型的服务器端解决方案。
  5. **概述**类别主要指HTTP服务器(REST APIs)。但是,所有这些构建模块都可以轻松用于不同的传输层(`microservices`或`websockets`包)。此外,Nest还配备了专用的 [GraphQL]($5.0-graphql?id=快速开始)模块。最后但并非最不重要的一点是,执行上下文功能有助于创建在Nest顶部的Node.js上运行的所有内容。
  6. Nest鼓励成为一个完整的平台,为您的应用带来更高级别的可重用性。建造一次,随处使用!
  7. ## 测试
  8. 自动测试是**全功能软件产品**的重要组成部分。这对于至少覆盖系统中最敏感的部分非常重要。为了实现这个目标,我们产生了一系列不同的测试,例如集成测试,单元测试,e2e测试等等。 Nest提供了一系列改进测试体验的测试实用程序。
  9. 通常,您可以使用您喜欢的任何**测试框架**。我们不强制模具,选择适合您要求的任何东西。主要的Nest应用程序启动程序与Jest框架集成在一起,以减少开始编写测试时的开销,但仍然可以轻松删除它并使用任何其他工具。
  10. ### 安装
  11. 首先,我们需要安装所需的软件包:
  12. ```bash
  13. $ npm i --save-dev @nestjs/testing

单元测试

在下面的例子中,我们有两个不同的类,分别是CatsControllerCatsService。如前所述,Jest被用作一个完整的测试框架。该框架的行为像一个测试运行者,并提供断言函数和测试双工实用程序,以帮助模拟,间谍等。我们已经手动强制执行catsService.findAll()方法来返回结果变量,一旦被调用。由此,我们可以测试catsController.findAll()是否返回预期的结果。

cats.controller.spec.ts
```typescript
import { CatsController } from ‘./cats.controller’;
import { CatsService } from ‘./cats.service’;

describe(‘CatsController’, () => {
let catsController: CatsController;
let catsService: CatsService;

beforeEach(() => {
catsService = new CatsService();
catsController = new CatsController(catsService);
});

describe(‘findAll’, () => {
it(‘should return an array of cats’, async () => {
const result = [‘test’];
jest.spyOn(catsService, ‘findAll’).mockImplementation(() => result);

  1. expect(await catsController.findAll()).toBe(result);
  2. });

});
});

  1. ?> 保持你的测试文件在附近的测试类。测试文件应该有`.spec`或`.test`后缀。
  2. 到目前为止,我们没有使用任何现有的Nest测试工具。由于我们手动处理实例化测试类,因此上面的测试套件与Nest无关。这种类型的测试称为**隔离测试**。
  3. ### 测试工具
  4. `@nestjs/testing`包给了我们一套提升测试过程的实用程序。让我们重写前面的例子,但现在使用暴露的`Test`类。
  5. > cats.controller.spec.ts
  6. ```typescript
  7. import { Test } from '@nestjs/testing';
  8. import { CatsController } from './cats.controller';
  9. import { CatsService } from './cats.service';
  10. describe('CatsController', () => {
  11. let catsController: CatsController;
  12. let catsService: CatsService;
  13. beforeEach(async () => {
  14. const module = await Test.createTestingModule({
  15. controllers: [CatsController],
  16. providers: [CatsService],
  17. }).compile();
  18. catsService = module.get<CatsService>(CatsService);
  19. catsController = module.get<CatsController>(CatsController);
  20. });
  21. describe('findAll', () => {
  22. it('should return an array of cats', async () => {
  23. const result = ['test'];
  24. jest.spyOn(catsService, 'findAll').mockImplementation(() => result);
  25. expect(await catsController.findAll()).toBe(result);
  26. });
  27. });
  28. });

Test类有一个createTestingModule()方法,该方法将模块元数据(与在@Module()装饰器中传递的对象相同的对象)作为参数。这个方法创建了一个TestingModule实例,它反过来提供了一些方法,但是当涉及到单元测试时,它们中只有一个是有用的-compile()。该功能是异步的,因此必须等待。一旦模块编译完成,您可以使用get()方法检索任何实例。

为了模拟一个真实的实例,你可以用自定义的提供者覆盖现有的 用户提供者。

端到端测试

当应用程序增长时,很难手动测试每个API端点的行为。端到端测试帮助我们确保一切工作正常并符合项目要求。为了执行e2e测试,我们使用与单元测试相同的配置,但另外我们利用允许模拟HTTP请求的超类库。

cats.e2e-spec.ts
```typescript
import * as request from ‘supertest’;
import { Test } from ‘@nestjs/testing’;
import { CatsModule } from ‘../../src/cats/cats.module’;
import { CatsService } from ‘../../src/cats/cats.service’;
import { INestApplication } from ‘@nestjs/common’;

describe(‘Cats’, () => {
let app: INestApplication;
let catsService = { findAll: () => [‘test’] };

beforeAll(async () => {
const module = await Test.createTestingModule({
imports: [CatsModule],
})
.overrideProvider(CatsService)
.useValue(catsService)
.compile();

  1. app = module.createNestApplication();
  2. await app.init();

});

it(/GET cats, () => {
return request(app.getHttpServer())
.get(‘/cats’)
.expect(200)
.expect({
data: catsService.findAll(),
});
});

afterAll(async () => {
await app.close();
});
});
`` ?> 将您的e2e测试文件保存在.e2e-spec.e2e-test`后缀。

cats.e2e-spec.ts测试文件包含一个HTTP端点测试(/ cats)。我们使用app.getHttpServer()方法来获取在Nest应用程序的后台运行的底层HTTP服务器。请注意,TestingModule实例提供了overrideProvider()方法,因此我们可以覆盖导入模块声明的现有提供程序。另外,我们可以分别使用相应的方法,overrideGuard()overrideInterceptor()overrideFilter()overridePipe()来相继覆盖守卫,拦截器,过滤器和管道。

编译好的模块有几种在下表中详细描述的方法:

createNestInstance()基于给定模块创建一个Nest实例(返回INestApplication),请注意,需要使用init()方法手动初始化应用程序
createNestMicroservice()基于给定模块创建Nest微服务实例(返回INestMicroservice)
get()检索应用程序上下文中可用的控制器或提供程序(包括警卫,过滤器等)的实例
select()例如,浏览模块树,从所选模块中提取特定实例(与启用严格模式一起使用)