Nest Logo

Description

Nest framework TypeScript starter repository.

Installation

$ npm install

Running the app

# development
$ npm run start

# watch mode
$ npm run start:dev

# production mode
$ npm run start:prod

Test

# unit tests
$ npm run test

# e2e tests
$ npm run test:e2e

# test coverage
$ npm run test:cov

Nest is MIT licensed.

Capítulo 1 Primeiros passos com Nest.js

  • O que é Nest.js

Gerenciar um aplicativo em grande escala pode ser entediante, especialmente quando construído sem uma estrutura bem planejada e uma estratégia rígida de organização de código. Nest.js ajuda nisso, reforçando a modularidade, seguindo assim o Princípio da Responsabilidade Única, um princípio fundamental da engenharia de software sólida. Nest.js é uma estrutura Node.js projetada para criar aplicativos de servidor eficientes, confiáveis e escalonáveis. Construído com base em Express e TypeScript, ele adota padrões de design Node.js e Angular para fornecer uma experiência de desenvolvimento coesa. A arquitetura do framework abrange decorators e injeção de dependência, padrões emprestados do Angular, para facilitar melhor organização e reutilização do código.

  • Que problema o Nestjs resolve

O desafio de manter a consistência em uma base de código extensa pode ser árduo. O Nest.js oferece uma solução ao incorporar padrões de design de engenharia de software, como SOLID e injeção de dependência. Estes padrões promovem uma arquitetura de código limpa, sustentável e escalável, alinhada com o princípio fundamental do Nest.js de criar código facilmente testável e com baixo acoplamento.

Enquanto estruturas não opinativas, como o Express, proporcionam diversas abordagens para a estruturação do código, essa liberdade de escolha pode resultar em paralisia decisória, consumindo tempo que poderia ser melhor empregado no desenvolvimento de recursos. O Nest.js supera esse obstáculo ao oferecer uma abordagem modular pronta para uso, permitindo uma organização flexível do código enquanto mantém um layout estruturado, em consonância com o seu princípio de modularidade.

Seguir uma decisão organizacional de código inicial pode ser desafiador, especialmente com a rotatividade da equipe de desenvolvimento. A abordagem estruturada e modular do Nest.js ajuda a preservar a integridade da arquitetura ao longo do tempo, facilitando a rápida compreensão da base de código por novos desenvolvedores.

Em resumo, o Nest.js visa solucionar o desafio de criar uma arquitetura robusta para aplicativos de backend. Ao oferecer uma estrutura bem definida e integrar padrões de design consolidados, ele aborda uma preocupação frequente entre os desenvolvedores sobre como organizar a arquitetura para aplicativos corporativos ou de grande porte no servidor.

  • Por que Nest.js foi criado

O fundador do Nest.js se baseou na arquitetura de design do Angular, utilizada na construção de aplicativos front-end, como inspiração para criar uma solução no lado do servidor. Essa transferência de conceitos permitiu a adoção de uma arquitetura familiar que incorpora elementos específicos do Nest.js, como módulos, injeção de dependência, provedores e decoradores personalizados. Essa abordagem estratégica reflete o princípio fundamental de modularidade e extensibilidade do Nest.js, facilitando a transição para os desenvolvedores familiarizados com o Angular.

  • Por que você deve usar Nest.js

Se você está planejando desenvolver APIs no lado do servidor utilizando TypeScript, o Nest.js é a escolha ideal. Esta estrutura aproveita a tipagem forte e os decoradores, que são características fundamentais do TypeScript, alinhando-se perfeitamente com os princípios da linguagem. Além disso, o Nest.js oferece suporte integrado para injeção de dependência, promovendo uma arquitetura modular e escalável.

O Nest.js é especialmente vantajoso para desenvolvedores que já têm experiência com Angular, pois compartilha muitos paradigmas e sintaxes comuns. Essa familiaridade reduz a curva de aprendizado, facilitando a transição para o Nest.js e melhorando a reutilização e a manutenção do código.

Esta estrutura possibilita a construção de diversos tipos de serviços no backend, desde APIs RESTful e endpoints GraphQL até aplicativos no padrão MVC e WebSockets. Sua flexibilidade permite a adoção de diversos padrões de comunicação, oferecendo uma ampla gama de soluções para o backend.

Desenvolvido para criar aplicativos monolíticos e microsserviços em grande escala, o Nest.js oferece escalabilidade e uma arquitetura modular. O uso do sistema de "Módulo" para organizar o código facilita a realização de testes de unidade e a colaboração entre os membros da equipe de desenvolvimento.

O Nest.js também oferece integrações nativas com diversas ferramentas externas, como o Mongoose para MongoDB e o TypeORM para diversos bancos de dados, incluindo o Postgres, entre outros. Essa extensibilidade permite que os desenvolvedores incorporem uma variedade de funcionalidades sem escrever código repetitivo, tornando a estrutura extremamente versátil.

  • Benefícios de usar Nest.js

● Aproveite a familiar sintaxe de estilo Angular para o desenvolvimento no backend, um recurso distintivo do Nest.js que promove consistência e reutilização em toda a pilha de tecnologia. Essa sintaxe permite que os desenvolvedores que já estão familiarizados com Angular façam uma transição suave para o desenvolvimento no backend.

● Tire proveito da documentação abrangente e dos exemplos fornecidos pelo Nest.js. Isso não só acelera a curva de aprendizado, mas também está alinhado com o princípio do Nest.js de ser amigável ao desenvolvedor e fácil de aprender.

● Beneficie-se do foco da estrutura em uma arquitetura sólida e no desenvolvimento ágil. O Nest.js adota o padrão de arquitetura modular, facilitando a separação de preocupações e tornando os testes mais simples, o que, por sua vez, acelera o processo de desenvolvimento.

  • Que tipo de aplicativo você pode construir

● Com o Nest.js, você pode desenvolver APIs REST e GraphQL no backend, aproveitando sua arquitetura modular que promove a reutilização e manutenção do código. O suporte integrado da estrutura para essas linguagens de consulta simplifica o desenvolvimento rápido de APIs.

● Construa microsserviços utilizando o Nest.js e aproveite seu suporte nativo a diversas camadas de transporte de mensagens. Essa abordagem está alinhada com os princípios da arquitetura de microsserviços, que preconiza componentes pouco acoplados e implantáveis de forma independente.

● Desenvolva o backend para um aplicativo de streaming utilizando o Nest.js, que oferece suporte à manipulação de dados assíncronos por meio de Observables. O Nest.js facilita a manipulação de fluxos de dados em tempo real, graças à sua integração com bibliotecas como RxJS.

● Crie o backend para um aplicativo em tempo real com o Nest.js, aproveitando seu suporte WebSocket para comunicação bidirecional em tempo real. Isso é fundamental para aplicativos que exigem atualização e interação instantânea de dados, seguindo os princípios do desenvolvimento de sistemas em tempo real.

  • Quem está usando Nest.js em produção

● A Roche, uma empresa multinacional de cuidados de saúde com divisões farmacêutica e diagnóstica, confia no Nest.js em seu site principal para oferecer serviços confiáveis aos pacientes e expandir seus serviços.

● A Adidas, líder europeia e segunda maior fabricante mundial de roupas esportivas, utiliza o Nest.js para construir aplicativos eficientes e escaláveis, aproveitando seu alcance global e popularidade para fornecer uma experiência de usuário superior.

● A Decathlon, varejista de artigos esportivos presente em mais de 1.500 lojas em 57 países, adotou o Nest.js como backend de seu aplicativo web para escalar e manter sua base de código legada.

  • Criando um projeto Nestjs

Antes de começar a criar seu projeto Nest.js, certifique-se de ter o Node.js instalado em sua máquina.

Aqui estão as etapas para criar o projeto Nest.js:

  1. Instale o CLI globalmente:
npm install -g @nestjs/cli
  1. Crie um novo projeto usando o Nest CLI:
nest new [nome do projeto]

Por exemplo, se você deseja criar um projeto chamado "n-fundamentos", execute o seguinte comando:

nest new minhaApi
  1. Inicie o projeto: Após a criação bem-sucedida do projeto, você pode iniciá-lo com o seguinte comando:
cd [nome do projeto]
npm run start:dev

Se você estiver usando o exemplo anterior, o comando será:

cd minhaApi
npm run start:dev

O projeto será executado em http://localhost:3000.

  • Estrutura de diretórios do projeto

A estrutura de diretórios de um projeto Nest.js segue uma convenção específica:

O Nest.js designa a pasta src como o local onde você irá colocar o código-fonte da sua aplicação. Essa organização adere à arquitetura modular do Nest.js, promovendo uma melhor organização e separação de preocupações dentro da sua aplicação.

Dentro da pasta src, o arquivo main.ts serve como o ponto de entrada da sua aplicação Nest.js. Este arquivo é responsável por inicializar a aplicação, utilizando a função principal do Nest.js, NestFactory, para criar uma instância da sua aplicação Nest.

import { NestFactory } from '@nestjs/core';
import { AppModule } from './app.module';

async function bootstrap() {
  const app = await NestFactory.create(AppModule);
  await app.listen(3000);
}

bootstrap();

O arquivo main.ts atua como o ponto de entrada da aplicação e emprega a função principal do NestFactory para instanciar uma aplicação Nest. No Nest.js, o NestFactory é crucial para inicializar a aplicação, configurar o sistema de injeção de dependências e inicializar os módulos. Isso segue o princípio do Nest.js de desenvolvimento modular e configuração centralizada.

Além disso, alguns outros arquivos e pastas importantes incluem:

  • app.controller.ts: Este é um arquivo de controller básico contendo uma única rota. Os controllers no Nest.js são responsáveis pelo tratamento de requisições e envio de respostas, atuando como uma interface entre o cliente e o servidor.
  • app.controller.spec.ts: Este arquivo contém testes unitários para o controller, seguindo o foco do Nest.js no desenvolvimento orientado a testes (TDD).
  • app.module.ts: Este é o módulo raiz da aplicação, que importa outros módulos e provedores. Os módulos do Nest.js funcionam como unidades organizacionais e seguem o Princípio da Responsabilidade Única.
  • app.service.ts: Um arquivo de serviço básico com um único método. No Nest.js, os serviços encapsulam a lógica de negócios e podem ser injetados nos controllers, promovendo a Injeção de Dependência e a Separação de Preocupações.

Além disso, existem alguns arquivos de configuração importantes:

  • nest-cli.json: Utilizado para configurações específicas do Nest.js, este arquivo permite a personalização das opções do compilador, ativos e outras configurações.
  • .prettierrc: Este arquivo é usado para configurar o Prettier, auxiliando na formatação do código e na consistência de estilo dentro do projeto Nest.js.
  • tsconfig.json: Este arquivo de configuração é para o TypeScript e determina como o compilador TypeScript se comportará. Isso está alinhado com o uso do TypeScript pelo Nest.js para tipagem forte e melhor qualidade de código.

Capítulo 2 Criando uma api REST

Oque sao module em nest?

Um módulo no Nest.js serve como a base fundamental de um aplicativo, seguindo o princípio de modularidade da estrutura para uma organização mais eficiente do código. Cada módulo encapsula provedores, controladores, importações e exportações, constituindo uma unidade coesa de funcionalidades relacionadas.

  • Provedores: No Nest.js, os provedores são classes que desempenham funções de serviço, fábrica ou repositório. Eles encapsulam a lógica de negócios e podem ser injetados em controladores ou outros serviços.

  • Controladores: Os controladores têm a responsabilidade de lidar com as solicitações HTTP recebidas e enviar respostas de volta ao cliente, seguindo o padrão de controlador adotado pelo Nest.js para o tratamento de solicitações.

  • Imports: Esta matriz especifica os módulos externos necessários para o módulo atual, promovendo a reutilização de código e a separação de interesses.

  • Exportações: São utilizadas para disponibilizar serviços para outros módulos, refletindo o foco do Nest.js em encapsulamento e design modular.

O seu aplicativo Nest.js geralmente conterá um módulo raiz, específico da estrutura Nest.js. Este módulo serve como ponto de entrada e é responsável por instanciar controladores, provedores e outros elementos essenciais do aplicativo. No Nest.js, essa arquitetura segue o princípio de "Isolamento de Módulo", garantindo que a aplicação seja organizada em módulos funcionais ou baseados em recursos distintos.

@Module({
  imports: [SongsModule],
  controllers: [AppController],
  providers: [AppService],
})
export class AppModule {}

Para ilustrar, vamos considerar a construção do backend para um aplicativo semelhante ao Spotify. Podemos dividir nosso aplicativo em módulos de acordo com as funcionalidades e recursos específicos.

  • Modulo da Aplicação
  1. Artist Module
  2. Songs Module
  3. Auth Module

Para organizar os casos de uso do seu aplicativo de forma modular, divida-os em módulos de recursos, como Módulo de Artistas, Módulo de Músicas e Módulo de Autenticação. No Nest.js, os módulos são unidades organizacionais fundamentais que seguem o princípio da modularização, permitindo uma melhor reutilização do código e uma separação mais clara de interesses. Esta abordagem agiliza o desenvolvimento, pois os módulos encapsulam funcionalidades relacionadas e podem ser desenvolvidos ou mantidos de forma independente.

Cada módulo terá seus próprios provedores, serviços e controladores. Agora é hora de criar um módulo. Por exemplo, para criar um módulo de músicas usando o Nest CLI, você pode executar o seguinte comando:

nest generate module songs

Isso criará um módulo de músicas em seu aplicativo e adicionará a entrada SongsModule no AppModule automaticamente, permitindo que você comece a desenvolver as funcionalidades relacionadas a músicas de forma organizada e modular.

  • Oque é um Controller

Os controladores no Nest.js desempenham um papel fundamental no gerenciamento das solicitações recebidas e no envio de respostas de volta ao cliente. Eles representam a camada de controle no padrão MVC (Model-View-Controller) que o Nest.js utiliza para arquitetura de aplicativos.

No contexto da construção do backend de um aplicativo semelhante ao Spotify, imagine que você deseje buscar todas as músicas do artista Lairton e seus teclados. A responsabilidade de lidar com esse tipo de solicitação recai principalmente nos controladores, específicos para esse caso de uso - o SongsController.

Quando o navegador inicia uma solicitação para buscar todas as músicas, seu aplicativo Nest.js irá processá-la por meio do SongsController e seu método findAll, que foi projetado especificamente para interagir com os serviços subjacentes, recuperar os dados e enviá-los de volta ao cliente.

Você pode criar esses endpoints no seu aplicativo. Por exemplo:

  • GET http://localhost:3000/songs
  • GET http://localhost:3000/songs/1
  • POST http://localhost:3000/songs
  • PUT http://localhost:3000/songs/1
  • DELETE http://localhost:3000/songs/1

Criar um controlador é bastante simples com o Nest CLI. Por exemplo, para criar um SongsController, você pode executar o seguinte comando:

nest generate controller songs

Isso criará o SongsController dentro do diretório de músicas e o Nest.js também adicionará a entrada para o SongsController no SongsModule. Em seguida, você pode definir os métodos do controlador para lidar com os diferentes endpoints, como mostrado no exemplo abaixo:

@Controller('songs')
export class SongsController {
  @Post()
  create() {
    return 'create a new song endpoint';
  }

  @Get()
  findAll() {
    return 'find all songs endpoint';
  }

  @Get(':id')
  findOne() {
    return 'find song based on id';
  }

  @Put(':id')
  update() {
    return 'update song based on id';
  }

  @Delete(':id')
  remove() {
    return 'delete a song based on id';
  }
}

Neste exemplo, estamos retornando mensagens simples para indicar que as rotas foram criadas. No Nest.js, você pode especificar parâmetros dinâmicos em suas rotas usando dois pontos seguidos do nome do parâmetro, como @Get(':id'). Isso segue o princípio do Nest.js de utilizar decoradores para lidar com tarefas HTTP comuns, simplificando a base de código e tornando-a mais legível.

  • Oque é um service

Os service no Nest.js são provedores, o que significa que você pode injetá-los em módulos e classes por meio de injeção de dependência. No Nest.js, um serviço não é apenas uma construção, mas um cidadão de primeira classe, gerenciado pelo contêiner de Inversão de Controle (IoC) integrado à estrutura. Ao contrário do Express.js, onde middleware ou funções JavaScript simples frequentemente servem ao mesmo propósito, mas sem gerenciamento formal de dependências, os service Nest.js oferecem uma maneira estruturada de escrever lógica de negócios, tornando o aplicativo mais fácil de manter e testável.

Um serviço em Nest.js é responsável por buscar dados do banco de dados e salvá-los de volta no banco de dados, funcionando como uma ligação entre o controlador e o banco de dados. Este conceito está alinhado com a adesão do Nest.js ao Princípio de Responsabilidade Única, separando a lógica de negócios da camada do controlador, um forte contraste com o Express.js, onde as funções são frequentemente delineadas de forma menos clara.

Um serviço pode ser injetado em um controlador usando o sistema integrado de injeção de dependência do Nest.js. Você também pode exportar o serviço do módulo atual, possibilitando seu uso em outras partes desse módulo específico. Esta é mais uma característica onde o Nest.js se diferencia de frameworks como o Express.js, oferecendo suporte nativo para modularidade e reutilização de código através de seu sistema de exportação.

Nesta lição, vamos nos concentrar na criação do serviço de músicas, um componente específico feito sob medida para gerenciar dados relacionados a músicas. Criar serviços especializados para diferentes aspectos do seu aplicativo promove melhor capacidade de manutenção e é a base da arquitetura modular do Nest.js.

  • Criando um service

Para completar a implementação do serviço de músicas, vamos detalhar mais a configuração e funcionalidades.

Primeiro, é necessário definir o serviço de músicas (SongsService). Podemos fazer isso usando o Nest CLI:

nest generate service songs

Este comando criará o SongsService dentro da pasta de músicas e registrará automaticamente o SongsService no SongsModule como um provedor, permitindo que ele seja injetado em outros componentes.

@Injectable()
export class SongsService {
  private readonly songs: Song[] = [];

  create(song: Song): Song[] {
    this.songs.push(song);
    return this.songs;
  }

  findAll(): Song[] {
    return this.songs;
  }
}

Agora, vamos revisar a funcionalidade do service de músicas:

  • create: Este método permite adicionar novas músicas ao array de músicas. Ele segue a filosofia de design do Nest.js de estruturar o código em torno de módulos e serviços bem definidos, proporcionando uma abordagem mais estruturada e declarativa em comparação com o Express.js.

  • findAll: Este método retorna todas as músicas do array. No Nest.js, este é um caso de uso comum para um manipulador de solicitação GET dentro de um controlador e é mais declarativo em comparação com o tratamento de rotas no Express.js, que geralmente requer middleware para tal funcionalidade.

Com o SongsService configurado, podemos injetá-lo em outros componentes, como o SongsController, para fornecer funcionalidades de busca e criação de músicas. Essa abordagem promove a modularidade e a reutilização de código, fundamentais para a arquitetura do Nest.js.

  • Injetando o Service no Controller

Para utilizar o SongsService dentro do SongsController, primeiro, vamos injetá-lo no construtor do controlador. No Nest.js, isso demonstra o sistema de Injeção de Dependência (DI), um princípio fundamental da estrutura para desacoplar componentes. Isso promove uma abordagem mais estruturada e modular, alinhada aos princípios SOLID, em comparação com o Express.js, onde as funções de middleware e gerenciamento de rotas geralmente se misturam à lógica de negócios.

export class SongsController {
  constructor(private songsService: SongsService) {}

  @Post()
  create() {
    return this.songsService.create(
      'Morango do nordeste de Lairton e seus teclados',
    );
  }

  @Get()
  findAll() {
    return this.songsService.findAll();
  }
}

Chamamos o método create do SongService no endpoint de criação de músicas (POST) e o método findAll no endpoint de busca de todas as músicas (GET).

  • Testando a rota GET songs

Para testar a rota GET /songs, execute o aplicativo e envie uma solicitação para http://localhost:3000/songs. Se nenhum item estiver presente no array songs, espere uma resposta vazia [] com um código de status 200. Ao contrário do Express.js, onde você definiria manualmente o código e o corpo da resposta, o Nest.js aproveita decoradores e injeção de dependência para agilizar o gerenciamento de respostas da API.

  • Testando a rota POST songs

Quando você envia a solicitação POST http://localhost:3000/songs do seu cliente REST, isso criará uma nova música no array de músicas gerenciado pelo SongsService.

Após criar a música, ao enviar a solicitação GET http://localhost:3000/songs novamente, você receberá o array atualizado de músicas como resposta. Isso demonstra a funcionalidade completa do endpoint de criação de músicas e como o sistema de serviço gerencia os dados para fornecer uma resposta atualizada ao solicitar todas as músicas.

Capítulo 3 Middlewares / Exception Filters e Pipes

  • Oque e um middlewares no nestjs?

Execute uma função de middleware antes de executar o manipulador de rota, por exemplo, antes do método findAll em SongsController. Em comparação com estruturas como o Express, o middleware no Nest.js oferece uma abordagem mais organizada e modular, alinhando-se estreitamente com os princípios da programação orientada a objetos e funcional.

O middleware terá acesso aos objetos req e res, assim como à próxima função, permitindo a personalização do objeto de solicitação. Embora semelhante ao Express, o Nest.js fornece uma arquitetura mais robusta e escalonável para a construção de aplicativos complexos. Isso facilita a criação de pipelines de middleware claros e modulares para a manipulação de solicitações antes de chegarem aos manipuladores de rota específicos.

  • Execute qualquer código dentro do middleware.

No Nest.js, o middleware é semelhante ao middleware Express.js, mas é mais baseado em classes e modular, adaptando-se bem à forte arquitetura modular do Nest. Ao contrário do Express, onde o middleware às vezes pode se tornar incontrolável em aplicativos grandes, o Nest.js oferece uma maneira mais estruturada de lidar com o middleware.

Modificando o objeto de solicitação (req), no Express.js tradicional, isso geralmente é feito diretamente na função de middleware. No Nest.js, entretanto, você pode contar mais com a injeção de dependência (DI) e a modularidade para fazer essas alterações de forma mais organizada.

Encerrando o ciclo de resposta, assim como no Express, o middleware no Nest.js pode encerrar o ciclo de solicitação-resposta. No entanto, o middleware Nest.js aproveita async/await e decoradores, oferecendo uma abordagem mais moderna e uma sintaxe mais limpa para lidar com tais operações.

Chamando o próximo middleware na pilha, tanto no Nest.js quanto no Express, o middleware pode passar o controle para a próxima função de middleware na pilha usando a função next(). No entanto, o Nest.js traz segurança de tipo e DI para o cenário, tornando mais fácil a construção de aplicativos robustos e de fácil manutenção.

Middleware do registrador, enviando a solicitação ao servidor por meio de um navegador. No aplicativo Nest.js, execute o middleware do criador de logs antes de executar o manipulador de solicitação. Esta arquitetura segue a abordagem modular do Nest.js, em que funções de registro semelhantes a middleware podem ser organizadas e reutilizadas em diferentes módulos de forma mais eficaz do que em uma estrutura como o Express, que não possui esse sistema modular integrado.

Os sistemas Logger são essenciais para rastrear atividades, diagnosticar problemas e compreender o comportamento de um aplicativo. Ao contrário das configurações tradicionais em que o registro pode ser uma reflexão tardia, o Nest.js permite a integração de mecanismos de registro sofisticados devido à sua natureza modular e extensível. Isso contrasta com estruturas menos opinativas como o Express, onde o log é frequentemente implementado por meio de middleware externo sem qualquer estrutura padrão.

  • Criando um logger middleware

Usaremos o Nest CLI para gerar o LoggerMiddleware. Crie a pasta common dentro do diretório src e também crie uma pasta middleware dentro do diretório common. Esta poderia ser a estrutura de diretório src/common/middleware/

nest g comum/middleware/logger --no-spec --no-flat

· --no-spec significa que não quero o arquivo de teste · --no-flat significa não criar o novo diretório com middleware logger. Você deve criar o arquivo logger.middleware.ts Você encontrará o logger.middleware.ts dentro da pasta middleware.

@Injetável()
export class LoggerMiddleware implements NestMiddleware {
  use(req: any, res: any, next: () => void) {
    console.log('Solicitação ....', new Date().toDateString());
    next();
  }
}

Crie uma classe LoggerMiddleware que implemente NestMiddleware. Certifique-se de escrever a implementação do método de uso. Personalize o objeto req conforme necessário; por exemplo, você poderia registrar a data atual.

  • Aplique middleware no AppModule.
export class AppModule implements NestModule {
  configure(consumer: MiddlewareConsumer) {
    // consumer.apply(LoggerMiddleware).forRoutes('songs'); // option no 1
    // consumer
    // .apply(LoggerMiddleware)
    // .forRoutes({ path:  'songs',  method:  RequestMethod.POST }); //option no 2
    consumer.apply(LoggerMiddleware).forRoutes(SongsController); //option no 3
  }
}

Escolha uma das três opções fornecidas para aplicar o middleware. Na última opção, implemente o LoggerMiddleware para as rotas SongsController. Teste o middleware Inicie o aplicativo usando npm run start:dev. Ao enviar uma solicitação para qualquer rota da API de songs, certifique-se de que ela exiba a data atual. Envie uma solicitação GET para localhost:3000/songs.

  • Tratamento de exceções

Se ocorrer um erro no código, lidar com ele torna-se crucial. NestJS oferece tratamento de exceção HTTP integrado que agiliza o processo de envio de respostas informativas e bem estruturadas ao cliente, um recurso que o diferencia de estruturas como Express, que requerem middleware adicional para funcionalidade semelhante. Lançar uma exceção no método SongsService findAll pode ser feito com facilidade. No NestJS, o uso de throw new HttpException('Description', HttpStatus.STATUS_CODE) permite mensagens personalizadas e códigos de status HTTP, fornecendo um mecanismo de tratamento de erros mais amigável e robusto ao desenvolvedor do que algumas outras estruturas de back-end como Flask, onde as exceções geralmente exigem mais configuração manual.

  findAll() {
 throw  new  Error('Error in Db while fetching record');
 return  this.songs;
 }

Uma falsa mensagem de erro foi enviada para simular um problema ao buscar dados do banco de dados. Enviar uma solicitação para buscar todas as músicas de http://localhost:3000/songs resultará em uma mensagem de erro acompanhada por um código de status 500.

  • Tratamento de exceção com Try/Catch
 @Get()
 findAll() {
try {
   return  this.songsService.findAll();
  }
   catch (err) {
   throw  new  HttpException(
   'server error',
   HttpStatus.INTERNAL_SERVER_ERROR,{ cause:  err },
); }

● O tratamento de exceções é possível usando o bloco try/catch, uma construção de programação padrão. No escopo do NestJS, isso é mais estruturado e seguro de tipo em comparação ao Express, onde o tratamento de erros geralmente depende de funções de middleware e não possui suporte nativo a TypeScript.

● O registro de mensagens no bloco catch serve como uma prática recomendada para fins de depuração e auditoria. No NestJS, o registro em log pode ser mais simplificado graças à sua arquitetura modular e à classe Logger integrada, ao contrário do Express, onde uma biblioteca de terceiros, como Winston ou Morgan, geralmente é necessária para um registro robusto.

● O envio de códigos de status HTTP específicos junto com mensagens de erro é facilitado no NestJS por meio de sua classe HttpException integrada. Isso fornece mais granularidade e controle sobre respostas de erro em comparação com o Express, que geralmente requer bibliotecas adicionais, como http-errors, para funcionalidade semelhante.

● Optar por um erro interno do servidor 500 é uma opção que indica um problema no servidor. Como prática recomendada, os engenheiros principais podem optar por mapear exceções para códigos de status HTTP específicos com base na natureza do erro, um recurso que tem suporte nativo e é simplificado no NestJS em comparação ao Express.

  • Pipes

● O recurso de Transform Param usando ParseInt é uma ferramenta poderosa do NestJS que facilita a conversão de tipos de forma eficiente. Enquanto estruturas como o Express carecem de transformação de parâmetros integrada, os pipes no NestJS oferecem uma abordagem mais automatizada e nativa para coerção de tipos, seguindo as melhores práticas para garantir a verificação robusta de tipos, algo altamente valorizado por engenheiros experientes.

● Existem dois casos principais de uso para pipes: transformação de valores e validação de parâmetros de entrada. Enquanto o Express demanda middleware adicional ou bibliotecas como express-validator para alcançar funcionalidades semelhantes, os pipes do NestJS se integram perfeitamente ao ecossistema da estrutura, oferecendo uma solução mais elegante e sustentável para transformação de valores e validação de entrada, alinhada com as melhores práticas de arquitetura.

// MúsicasContoller.ts
@Get(':id')
findOne(
     @Param(
         'id',
         new ParseIntPipe({
             errorHttpStatusCode: HttpStatus.NOT_ACCEPTABLE
        }),
      ),
id: number, ){
     return `buscar música com base no id ${typeof id}`;
}

● Os parâmetros dinâmicos podem ser capturados usando o decorador @Param, onde o nome do argumento precisa ser especificado. Em contraste com o Express, onde os parâmetros de solicitação são extraídos usando req.params, o NestJS fornece uma maneira mais declarativa e segura de tipo de fazer isso, aderindo às práticas recomendadas ao impor um sistema de tipos mais rígido.

● O parâmetro id é do tipo string por padrão. Utilizar ParseIntPipe converterá automaticamente esse valor de string em um número. Ao contrário do Express, que exigiria conversão manual de tipo, o uso de pipes no NestJS permite a transformação automática de tipo, tornando o código mais robusto e fácil de manter, uma prática que qualquer engenheiro experiente apreciaria.

● Enviar uma solicitação para http://localhost:3000/songs/1 resultará no registro do tipo de ID como um número. Isso mostra a capacidade do NestJS de utilizar pipes para tarefas de transformação, uma área onde ele mantém uma vantagem sobre estruturas como o Express, que necessitam de middleware separado para tais operações.

● O código de status de erro também pode ser fornecido ao ParseIntPipe. Caso um valor de string seja fornecido, um erro será gerado. Essa abordagem permite um melhor tratamento de erros no NestJS em comparação com os métodos mais manuais de verificação de erros exigidos no Express.

● Enviar uma solicitação para http://localhost:3000/songs/abc produzirá uma mensagem de erro informando “não aceitável”. Em estruturas como Express, a lógica de validação para lidar com tais erros geralmente precisaria ser escrita explicitamente, enquanto o NestJS permite mecanismos de validação mais configuráveis e integrados. Esse recurso está alinhado às práticas recomendadas de manutenção e escalabilidade.

Valide os parâmetros da solicitação com o validador de classe Para validar parâmetros de solicitação, o validador de classe é frequentemente usado no NestJS. A instalação de dois pacotes necessários inicia esse recurso, tornando a validação uma parte integrante do processo de tratamento de solicitações, ao contrário do Express, onde a lógica de validação pode ser codificada manualmente ou extraída por meio de middleware adicional. O uso do validador de classe no NestJS permite regras de validação declarativas em classes DTO (Data Transfer Object) usando vários decoradores, como @IsString() ou @IsNotEmpty(). Essa abordagem promove a reutilização e a capacidade de manutenção da lógica de validação, alinhando-se com as melhores práticas para uma arquitetura de aplicativo escalonável, enquanto no Express, muitas vezes são necessárias bibliotecas de validação separadas, como validator ou express-validator.

  • Escopo de Pipes global

· A próxima etapa envolve vincular o ValidationPipe do pacote @nestjs/common. Esse recurso oferece uma vantagem sobre estruturas como o Express, onde a validação geralmente requer bibliotecas adicionais ou middleware personalizado. No NestJS, os pipes oferecem vários escopos para validação: escopo de parâmetro, escopo de método, escopo de controlador ou escopo global, proporcionando maior flexibilidade e modularidade ao aplicativo.

· A opção por escopo global requer registro no arquivo main.ts. Esta é uma prática recomendada para garantir consistência na validação em todo o aplicativo, minimizando as chances de falta de lógica de validação em qualquer parte do aplicativo. Ao contrário do Express, que exigiria uma função de middleware global para funcionalidade semelhante, o NestJS simplifica a definição de regras de validação globais. src/main.ts

import { ValidationPipe } from '@nestjs/common';
// ...
app.useGlobalPipes(new ValidationPipe());
  • Criando um CreateSongDTO

Um DTO (Data Transfer Object) serve como um modelo de como os dados serão enviados pela rede. Embora as interfaces TypeScript e as classes simples possam definir o esquema DTO, as classes são recomendadas neste contexto. Em contraste com o Express, onde o DTO e a validação geralmente exigem bibliotecas ou middleware adicionais, o NestJS oferece uma abordagem mais integrada por meio de suas técnicas de validação baseadas em decorador. Como melhor prática, armazenar DTOs em um diretório dedicado garante que o aplicativo siga o princípio de separação de interesses, facilitando a manutenção e o dimensionamento futuro. Para instanciar o DTO, uma nova classe deve ser criada dentro do arquivo src/songs/dto/create-song.dto.ts. Esta é uma abordagem mais estruturada do que em estruturas como o Express, onde o esquema e a validação podem ser misturados com manipuladores de rotas ou middleware. Ter um arquivo DTO dedicado conduz a um código mais modular e de fácil manutenção.

import {
  IsArray,
  IsDateString,
  IsMilitaryTime,
  IsNotEmpty,
  IsString,
} from 'class-validator';

export class CreateSongDto {
  @IsString()
  @IsNotEmpty()
  title: string;

  @IsArray()
  @IsString({ each: true })
  artists: string[];

  @IsDateString()
  @IsNotEmpty()
  releaseDate: Date;

  @IsMilitaryTime()
  @IsNotEmpty()
  duration: Date;
}

Quatro campos – title, artists, releaseDate, duration – estão presentes. O validador de classe permite a adição de validações baseadas em decorador. A função isDateString() é empregada para validar a data, enquanto isMilitaryTime() é usada para validação de hora no formato HH:MM.

Aplicar CreateSongDTO como decorador corporal Você tem que atualizar o método create dentro de src/songs/songs.controller.ts

import { Body, Controller, Post } from '@nestjs/common';
import { CreateSongDto } from './dto/create-song.dto';

@Controller('songs')
export class SongsController {
  constructor(private readonly songsService: SongsService) {}

  @Post()
  create(@Body() createSongDTO: CreateSongDto) {
    const results = this.songsService.create(createSongDTO);
    return results;
  }
}
  • Testando a aplicação

Para iniciar o aplicativo, execute o comando npm run start:dev e prossiga para enviar a solicitação ‘create song’. Ao contrário do Express, onde o nodemon ou um pacote semelhante seria instalado separadamente para recarregamento a quente, o NestJS inclui esse recurso por padrão com o script start:dev, proporcionando uma experiência de desenvolvimento mais pronta para uso. Como prática recomendada, separar a lógica de ‘create song’ em um método de serviço dedicado garante um código mais limpo e de fácil manutenção.

POST http://localhost:3001/songs Content-Type: application/json

{
  "title": "Clovis",
  "artists": ["Sua mae  "],
  "releaseDate": "2022-07-29",
  "duration": "02:34"
}

Capítulo 3 Injeção de dependência

  • Custom Providers

● A injeção de dependência baseada em construtor é uma prática essencial no Nest.js, permitindo a injeção de instâncias, especialmente provedores de serviços como SongsService, nas classes usando o construtor (por exemplo, constructor(private readonly songsService: SongsService)). Essa abordagem contrasta com estruturas como o Express, onde a injeção de dependência não é suportada nativamente e muitas vezes requer o uso de bibliotecas de terceiros para funcionalidades semelhantes. A injeção baseada em construtor é preferida como prática recomendada devido à sua clareza e facilidade de teste.

● Quando o contêiner Nest IoC (Inversão de Controle) cria uma instância de SongsController, ele automaticamente verifica quaisquer dependências, como SongsService. Ao identificar a dependência, o Nest instancia o SongsService, armazena-o em cache e o retorna. Se uma instância já estiver armazenada em cache, ela é reutilizada. Esse comportamento difere do Express, que geralmente não possui um contêiner IoC integrado, exigindo instanciação manual ou soluções de terceiros. Aproveitar o cache para instâncias de serviço pode levar a otimizações de desempenho significativas.

● A injeção de dependência baseada em construtor é uma prática essencial no Nest.js, permitindo a injeção de instâncias, especialmente provedores de serviços como SongsService, nas classes usando o construtor (por exemplo, constructor(private readonly songsService: SongsService)). Essa abordagem contrasta com estruturas como o Express, onde a injeção de dependência não é suportada nativamente e muitas vezes requer o uso de bibliotecas de terceiros para funcionalidades semelhantes. A injeção baseada em construtor é preferida como prática recomendada devido à sua clareza e facilidade de teste.

● Quando o contêiner Nest IoC (Inversão de Controle) cria uma instância de SongsController, ele automaticamente verifica quaisquer dependências, como SongsService. Ao identificar a dependência, o Nest instancia o SongsService, armazena-o em cache e o retorna. Se uma instância já estiver armazenada em cache, ela é reutilizada. Esse comportamento difere do Express, que geralmente não possui um contêiner IoC integrado, exigindo instanciação manual ou soluções de terceiros. Aproveitar o cache para instâncias de serviço pode levar a otimizações de desempenho significativas.

  • Existem várias técnicas para utilizar provedores no NestJS:
  1. Provedores padrão: São classes que são instanciadas pelo sistema de injeção de dependência do NestJS. Ao contrário do Express, onde a injeção de dependência geralmente deve ser implementada manualmente ou por meio de bibliotecas de terceiros, o NestJS fornece esse recurso nativamente. Como prática recomendada, aproveite os provedores padrão para serviços que exigem instanciação.

  2. Provedores de valor: São valores codificados ou configurações injetadas em outras classes. Esse recurso pode substituir a necessidade de variáveis de ambiente ou arquivos de configuração, que em estruturas como o Express, normalmente seriam gerenciados por pacotes separados. O uso de provedores de valor para constantes e definições de configuração contribui para a manutenção do código.

  3. Tokens de provedor não baseados em classe: São tokens personalizados que podem ser usados para injetar valores ou serviços. Ao contrário de frameworks como Django, que depende mais de visualizações baseadas em funções e não possui equivalente direto, o NestJS permite maior flexibilidade na injeção de dependência. Use tokens personalizados criteriosamente para evitar gráficos de dependência excessivamente complexos.

  4. Provedores de classes useClass: Permitem a substituição de um provedor de classes por outro, proporcionando polimorfismo. Isso é diferente do Express, onde tal abstração pode exigir padrões de fábrica mais manuais. Empregar useClass é benéfico para alcançar a reutilização e flexibilidade do código.

  5. FactoryProviders com useFactory: Permitem a instanciação condicional de uma classe, algo não suportado nativamente em frameworks como Flask. Usar FactoryProviders quando o fornecimento de uma classe ou valor depende de condições em tempo de execução.

  6. Provedores que não são de serviços: São provedores que não estão necessariamente vinculados a um serviço e podem fornecer funções utilitárias, por exemplo. Essa granularidade geralmente não está disponível em configurações básicas do Express e normalmente exigiria módulos ou utilitários adicionais. A incorporação de provedores que não sejam de serviços pode ajudar a manter a base de código DRY (Don’t Repeat Yourself).

Será realizada a exploração de todas as seis técnicas, cada uma elucidada através de um exemplo. Vamos brincar com todas essas 6 técnicas. Explicarei cada técnica com a ajuda de um exemplo.

  • Provedores padrão:

Esta é a técnica padrão de provedor que você usou em nosso aplicativo. Você também pode converter a sintaxe acima para esta:

@Módulo({
    controller: [SongsController],
    providers: [{
        provide: SongsService,
        useClass: SongsService,
    }],
})

Essa forma de declaração permite uma maior flexibilidade ao configurar provedores, especialmente quando há a necessidade de usar useClass para fornecer uma classe específica como provedor.

  • Provedor de valor:

A sintaxe useValue é útil para injetar um valor constante, colocar uma biblioteca externa no contêiner Nest ou substituir uma implementação real por um objeto simulado. Digamos que você queira forçar o Nest a usar um SongsService simulado para fins de teste.

Exemplo de uso:

const mockSongsService = {
  findAll() {
    return [
      {
        id: 1,
        title: 'Morango do nordeste',
      },
    ];
  },
};

@Module({
  controllers: [SongsController],
  providers: [
    SongsService,
    {
      provide: SongsService,
      useValue: mockSongsService,
    },
  ],
})
export class SongsModule {}

Quando você envia uma solicitação GET para localhost:3000/songs, ele executará o método findAll() de mockSongsService em vez de SongsService original.

Exemplo de uso com constantes:

// src/common/constants/connection.ts
export const connection: Connection = {
  CONNECTION_STRING: 'CONNECTION_STRING',
  DB: 'POSTGRES',
  DBNAME: 'TEST',
};

export type Connection = {
  CONNECTION_STRING: string;
  DB: string;
  DBNAME: string;
};

Essas constantes podem ser usadas em todo o seu aplicativo e injetadas como valores por meio da sintaxe useValue quando necessário.

  • Provedores:

Para injetar um objeto como dependência, useValue pode ser utilizado. Esse recurso distingue o NestJS de estruturas como o Express, onde a injeção manual baseada em construtor costuma ser a norma. No caso em questão, foi criado um novo arquivo chamado connection.ts, contendo um objeto de conexão com CONNECTION_STRING, BD e DBNAME. Como prática recomendada, manter as configurações de conexão em arquivos separados e importá-las conforme necessário garante uma configuração de aplicativo modular e facilmente configurável.

Exemplo de uso:

// src/common/constants/connection.ts
export const connection: Connection = {
  CONNECTION_STRING: 'CONNECTION_STRING',
  DB: 'POSTGRES',
  DBNAME: 'TEST',
};

export type Connection = {
  CONNECTION_STRING: string;
  DB: string;
  DBNAME: string;
};

O objeto de conexão agora pode ser injetado em qualquer controlador ou serviço dentro do SongsModule. Isso contrasta com estruturas como Express, onde a injeção de dependência não é nativa e muitas vezes requer bibliotecas de terceiros como ‘Awilix’ para funcionalidades semelhantes. Como prática recomendada, isolar conexões de banco de dados em arquivos de provedores dedicados garante um código mais modular e de fácil manutenção, facilitando assim testes unitários e separação de interesses.

import { connection } from 'src/common/constants/connection';

@Módulo({
  controllers: [SongsController],
  providers: [
    SongsService,
    // Provedores não baseados em classe
    // Você pode usá-lo para adicionar valores constantes
    {
      provide: 'CONEXÃO',
      useValue: connection,
    },
  ],
})
export class SongsModule {}

Sera injetado este objeto de conexão em SongsService usando o decorador @Inject com o nome do token CONNECTION. Isso permite que o serviço acesse as configurações de conexão de forma fácil e segura.

@Injectable()
export class SongsService {
  constructor(
    @Inject('CONNECTION')
    connection: Connection,
  ) {
    console.log('connection string', connection.CONNECTION_STRING);
  }
}
  • Class Providers: useClass

Uma classe DevConfigService foi criada e registrada como provedor. Com esta configuração, o AppModule é capaz de injetar esse provedor como uma dependência em qualquer classe para utilização. Em contraste com o Express, que normalmente depende de bibliotecas de terceiros, como dotenv, para gerenciamento de configuração, o NestJS oferece uma maneira mais integrada de lidar com configurações por meio de provedores de serviços personalizados. Como prática recomendada, encapsular a lógica de configuração em uma classe de serviço dedicada permite testes e manutenção mais fáceis, aderindo ao princípio de Separação de Preocupações.

// src/common/providers/DevConfigService.ts
import { Injectable } from '@nestjs/common';

@Injectable()
export class DevConfigService {
  DBHOST = 'localhost';

  getDBHOST() {
    return this.DBHOST;
  }
}

@Injectable()
export class AppService {
  constructor(private devConfigService: DevConfigService) {}

  getHello(): string {
    return `Olá, estou aprendendo os fundamentos do Nest.js ${this.devConfigService.getDBHOST()}`;
  }
}

O DevConfigService foi injetado no AppService e o método getDBHOST do DevConfigService é invocado. Ao contrário do Express, onde a injeção de dependência não é suportada nativamente e pode exigir bibliotecas de terceiros, o NestJS simplifica isso com injeção de dependência integrada, levando a soluções mais modulares e código testável. Como prática recomendada, as configurações específicas do ambiente, como detalhes do host do banco de dados, devem ser abstraídas em serviços de configuração separados, garantindo que o código esteja de acordo com a metodologia do Aplicativo de Doze Fatores.

  • Non-Service Provider

Um provedor no NestJS tem flexibilidade para fornecer qualquer valor, tornando-o um componente versátil para injeção de dependência. A sintaxe useFactory facilita a criação dinâmica de provedores, um recurso que o diferencia do Express, que geralmente depende de bibliotecas externas como awilix ou injeção manual de construtor para funcionalidades semelhantes. Como prática recomendada, isolar lógica de negócios complexa nos provedores de fábrica pode levar a um código mais modular e testável.

const devConfig = {
  port: 3000,
};

const proConfig = {
  port: 400,
};

@Module({
  providers: [
    {
      provide: 'CONFIG',
      useFactory: () => {
        return process.env.NODE_ENV === 'development' ? devConfig : proConfig;
      },
    },
  ],
})
export class AppModule {}

Dois objetos, devConfig e proConfig, foram criados. A atribuição dinâmica de valores é possível por meio da função useFactory. Ao contrário do Express, onde as configurações específicas do ambiente geralmente exigem arquivos JSON ou JS separados, o NestJS permite uma configuração dinâmica em linha mais simplificada. Como prática recomendada, isolar a lógica de configuração em módulos ou serviços dedicados promove a capacidade de manutenção e a escalabilidade.

@Injectable()
export class AppService {
  constructor(
    @Inject('CONFIG')
    private configuração: {
      port: string;
    },
  ) {
    console.log(config);
  }
}

O Provider CONFIG foi injetado dentro do AppService.

  • Injection Scopes

Os escopos de Provider no NestJS podem ser categorizados em três tipos principais:

  1. PADRÃO

    • Neste escopo, uma única instância do Provider é compartilhada por todo o aplicativo, um recurso semelhante aos serviços singleton do Express, mas explicitado no NestJS. Quando o aplicativo solicita esse Provider pela segunda vez, o NestJS o recupera de um cache interno, um mecanismo que melhora o desempenho do aplicativo. Como prática recomendada, é aconselhável usar provedores de cache para minimizar a sobrecarga computacional.
  2. PEDIDO

    • Ao contrário do escopo PADRÃO, uma nova instância do Provider é criada para cada solicitação recebida. Embora isso forneça melhor isolamento, é um modelo diferente da arquitetura baseada em middleware do Express, onde o isolamento em nível de solicitação geralmente envolve o escopo da função. Este escopo é ideal para cenários onde cada solicitação pode precisar de um Provider com estado que não deve afetar outras solicitações.
  3. TRANSITÓRIO

    • Os provedores transitórios são únicos porque não são compartilhados entre diferentes consumidores. Quando um Provider transitório é injetado, o consumidor recebe uma instância nova e dedicada, diferentemente do Express, onde esse nível de granularidade de injeção não é nativo e muitas vezes deve ser gerenciado manualmente. A contratação de fornecedores transitórios é uma boa prática quando o isolamento estatal entre os consumidores é crucial.
@Injectable({
  scope: Scope.TRANSIENT,
})
export class SongsService {}
@Controller({
  path: 'songs',
  scope: Scope.REQUEST,
})
export class SongsController {}

Uma nova instância do SongsController é criada para cada solicitação recebida, oferecendo uma arquitetura mais sem estado. Em contraste, estruturas como o Express usam um padrão singleton para controladores, reutilizando assim a mesma instância para todas as solicitações, o que pode introduzir problemas relacionados ao estado. Adotar uma arquitetura sem estado é uma prática recomendada para melhorar a escalabilidade e a capacidade de manutenção.

A depuração de dependências pode apresentar desafios, mas é essencial compreender o uso de diferentes escopos. Ao contrário do Express, que não possui um sistema DI (Injeção de Dependência) integrado, o NestJS possui vários escopos para injeção de dependência, tornando-o mais flexível para casos de uso complexos. Utilizar o escopo correto para dependências é considerado uma prática recomendada, pois melhora o desempenho e a arquitetura geral.