Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/etienne-bechara/bp-nestjs

Full back-end boilerplate based on NestJS and MikroORM. Written in TypeScript.
https://github.com/etienne-bechara/bp-nestjs

back-end backend mikro-orm nestjs redis typescript

Last synced: about 1 month ago
JSON representation

Full back-end boilerplate based on NestJS and MikroORM. Written in TypeScript.

Awesome Lists containing this project

README

        

# NestJS Boilerplate

⚠️ Atenção!

Este repositório foi dividido em vários pacotes menores e não será mais atualizado, confira os componentes em:
- [@bechara/nestjs-core](https://github.com/etienne-bechara/nestjs-core): Framework principal com agregando logger, http adapter, e outras amenidades ao NestJS.
- [@bechara/nestjs-orm](https://github.com/etienne-bechara/nestjs-orm): Framework opcional agregando ORM e abstrações para manipular entidades.
- [@bechara/nestjs-redis](https://github.com/etienne-bechara/nestjs-redis): Plugin opcional para conectar banco de dados Redis.
- [@bechara/eslint-config-bechara-ts](https://github.com/etienne-bechara/eslint-config-bechara-ts): Múltiplas regras e plugins de lint em TypeScript.
---

Boilerplate baseado em NestJS com intuito de prover iniciação rápida de projetos em Node.js.

**Guia Rápido**

1\. Clone este repositório renomeando-o de acordo com seu novo projeto:

```shell
git clone https://github.com/etienne-bechara/bp-nestjs.git meu-novo-projeto
cd meu-novo-projeto
```

2\. Execute o script de setup, que irá instalar as dependências e renomear o remote `origin` para `boilerplate`:

```shell
npm run boilerplate:setup
```

2a. Futuramente, caso deseje sincronizar as atualizações deste boilerplate, execute:

```shell
npm run boilerplate:udpate
```

2b. Caso não queira utilizar algum dos serviços opcionais, desinstale-os via:

```shell
npm run uninstall:orm
npm run uninstall:redis
npm run uninstall:sentry
```

3\. Suba a aplicação localmente através de:

```shell
npm start
```

---

## Índice

- [Utilização](#utilização)
* [Boilerplate](#boilerplate)
* [Depuração](#depuraçãoo)
* [Dependências](#dependências)
- [Componentes](#componentes)
* [Frameworks](#frameworks)
* [Utilitários](#utilitários)
- [Domain](#domain)
- [Config](#config)
* [Variáveis de Ambiente](#variáveis-de-ambiente)
* [Opções do Serviço](#opções-do-serviço)
- [Entity](#entity)
- [Modules](#modules)
- [Controllers](#controllers)
- [DTO](#dto)
- [Providers](#providers)
- [Middlewares](#middlewares)
- [Interceptors](#interceptors)
- [Filters](#filters)
- [Testes](#testes)
- [Migrations](#migrations)
- [Templates](#templates)
* [API](#api)
* [CRUD](#crud)

---

## Utilização

Após seguir os passos do Guia Rápido acima, envie uma requisição `GET` para `localhost:8080`.

O retorno a seguir indica que a aplicação subiu com sucesso:

```json
{
"error": 404,
"message": "Cannot GET /",
"details": {}
}
```

### Boilerplate

Sempre que quiser atualizar o boilerplate a última versão, execute:

```shell
npm run boilerplate:update
```

Caso tenha alterado arquivos na raiz ou no diretório `/source/core` provavelmente será necessário resolver conflitos de merge antes que possa commitar as alterações.

### Depuração

Por padrão, a aplicação irá expor um sessão de debug na porta `9229`.

Ao utilizar a ferramenta `VSCode` como IDE de desevolvimento, basta executar `npm start` e pressionar `F5` para conectar o debugger.

Você pode criar os `breakpoint` diretamente no arquivo `.ts` que eles serão automaticamente mapeados pelos `.js` em execução.

### Dependências

Todas os pacotes que este projeto utiliza estão configurados com a versão exata.

Para realizar atualização dos mesmos, execute um dos scripts a seguir de acordo com o nível de versão que deseja subir.

Siga o padrão Semantic Versioning: `{major}.{minor}.{patch}`

```shell
npm run update:patch
npm run update:minor
npm run update:major
```

## Componentes

Vários dos frameworks que este projeto é composto são opcionais e podem ser facilmente excluídos para economia de memória RAM durante execução.

### Frameworks

Documentação | Reponsabilidades | Observação
---|---|---
[NestJS](https://docs.nestjs.com/) | • Injeção de Dependências
• Inicialização do Servidor (Express)
• Middlewares e Fluxos de Validação
• Filtro Global de Exceções | Irá carregar automaticamente todos os arquivos nomeados como `*.module.ts`.
[Jest](https://jestjs.io/docs/en/getting-started) | • Testes Unitários
• Testes E2E | Instalado apenas em ambiente de desenvolvimento.
Crie os arquivos de teste no padrão `*.spec.ts`.
[Sentry](https://www.npmjs.com/package/@sentry/node) | • Monitoramento em Tempo Real
• Rastreio de Exceções | **Opcional**
Habilite configurando a variável `SENTRY_DSN` no `.env`.
[MikroORM](https://mikro-orm.io/docs/installation) | • Abstração de Banco de Dados como Entidades
• Geração e Execução de Migrations | **Opcional**
Habilite configurando as variáveis `ORM_*` no `.env`.
[Redis](https://www.npmjs.com/package/redis) | • Armazenamento de Dados do Tipo Chave/Valor
• Compartilhamento de Alta Performance em Serviços Distribuídos | **Opcional**
Habilite configurando as variáveis `REDIS_*` no `.env`.

### Utilitários

Documentação | Reponsabilidades | Utilização
---|---|---
[Axios](https://www.npmjs.com/package/axios) | • Requisições HTTP(s) externas | Foi criado um wrapper em torno da bilblioteca para padronização de exceções.
Injete o serviço `HttpsService` na classe que deseja utilizar.
[Class Validator](https://www.npmjs.com/package/class-validator) | • Decorators para Validação de Objetos | Dentro da classe, antes da propriedade, utilize um ou mais dos decorators do pacote.
No caso dos controllers a validação é aplicada automaticamente.
[Class Transformer](https://www.npmjs.com/package/class-transformer) | • Conversão de Objetos para Classes
• Conversão de Tipo de Propriedades | Utilize um dos métodos do pacote em conjunto com os decorators fornecidos.
Em geral, não será necessário ao menos que altere algo a nível de boilerplate.
[Moment](https://www.npmjs.com/package/moment) | • Parsing e Formatação de Datas | Inicialize através de `moment()` ou `moment(stringDate)`, e siga os métodos conforme documentação.

## Domain

É o conjunto de todos os componentes que definem um módulo do projeto.

Por exemplo, os usuários, as empresas, a autenticação, a API XPTO externa, etc.

O domínio é representado por uma pasta dentro do diretório `/source`.

Os inclusos no boilerplate estão dentro de `/source/core` para melhor organização.

## Config

Cada domínio, pode ter uma grupo de configurações definidas em um arquivo `*.config.ts`.

Ao criar um serviço que extenda a class `AppProvider` (detalhes adiante), é possível obter as recém criadas configurações através do método `this.getConfig()`.

Exemplo:

```ts
export class MailerService {
private config: MailerConfig = this.getConfig();
}
```

As configurações são divididas em duas categorias:

### Variáveis de Ambiente

- Possuem informações sensíveis ou que variam de accordo com o ambiente.
- São declaradas em modelo chave/valor dentro do arquivo `.env` na raiz do projeto.
- Todas são inicializadas como `string`.
- Utilize o decorator `@Transform()` da lib `class-transformer` para convertê-las de tipo.
- Devem ser declaradas sem valor padrão.

Exemplo:

```ts
export class AppConfig {

@IsIn(['DEVELOPMENT', 'STAGING', 'PRODUCTION'])
public NODE_ENV: 'DEVELOPMENT' | 'STAGING' | 'PRODUCTION';

@Transform((v) => parseInt(v))
@IsNumber()
public PORT: number;

@IsOptional()
@IsString() @IsNotEmpty()
public APP_AUTHORIZATION: string;

}
```

### Opções do Serviço

- Não possuem informações sensíveis e são idênticas em qualquer ambiente.
- São declaradas dentro do próprio arquivo `*.config.ts`.
- Devem ser inicializadas com valor padrão.

Exemplo:

```ts
export class AppConfig {

public APP_TIMEOUT: number = 2 * 60 * 1000;

public APP_CORS_OPTIONS: CorsOptions | boolean = {
origin: '*',
methods: 'GET, POST, PUT, DELETE',
allowedHeaders: 'Content-Type, Accept',
};

public APP_VALIDATION_RULES: ValidationPipeOptions = {
whitelist: true,
forbidNonWhitelisted: true,
};

}
```

## Entity

> Conceito aplicável apenas se utilizado o ORM

Define uma entidade da modelagem de dados, em um banco relacional, podemos considerar como uma tabela.

São definidas pelos arquivos `*.entity.ts` utilizando decorators para configurar tipo das propriedades e relacionamentos.

Refira-se a [Mikro ORM - Decorators Reference](https://mikro-orm.io/docs/decorators/) para todas as opções.

### Predefinidos

Classe | Arquivo | Descrição
---|---|---
OrmIdEntity | `ormid.entity.ts` | Extenda essa classe para já incluir uma coluna UUID primária.
OrmTimestampEntity | `ormtimestamp.entity.ts` | Extenda essa classe para já incluir colunas UUID primária, data de criação e data de atualização.

### Exemplos

```ts
@Entity({ collection: 'user' })
export class UserEntity {
// utilize extends OrmTimestampEntity acima para simplificar

@PrimaryKey()
public id: string = v4();

@Property()
@Unique()
public name!: string;

@ManyToOne()
public team!: TeamEntity;

@Index()
@Property({ columnType: 'timestamp', onUpdate: () => new Date() })
public updated: Date = new Date();

@Index()
@Property({ columnType: 'timestamp' })
public created: Date = new Date();

}
```

## Modules

Unifica todos os componentes de um domínio em um arquivo `*.module.ts` que será lido e carregado pela aplicação.

Novas classes de módulos devem ser decoradas com `@Module()` e serão carregadas automaticamente quando o serviço iniciar.

A criação de qualquer componente descrito a seguir, sem integrá-lo em um módulo, implicará no mesmo não ter efeito algum.

Refira-se a [Nest JS - Modules](https://docs.nestjs.com/modules) para mais informações.

### Predefinidos

Classe | Arquivo | Descrição
---|---|---
AppModule | `app.module.ts` | Ponto de entrada da aplicação, carrega todos os outros módulos.
HttpsModule | `https.module.ts` | Carrega o serviço abstraído de requisições HTTP(s) baseado em Axios. opcional de envio de e-mails via SMTP.
OrmModule | `orm.module.ts` | Carrega a camada opcional de abstração para armazenamento em banco de dados.
RedisModule | `redis.module.ts` | Carrega o serviço opcional para armazenamento chave/valor de alta performance.

### Exemplos

```ts
@Module({
controllers: [ UserController ],
providers: [ UserService ],
exports: [ UserService ],
})
export class UserModule { }
```

## Controllers

São responsáveis por receber as requisições HTTP, aplicar validações, e redirecionar os dados para o respectivo serviço de manipulação.

Os controllers seguem o padrão `*.controller.ts`, e devem ser criados dentro da pasta de seu respectivo domínio e importados na propriedade `controllers` do módulo correspondente.

Para definir um controller utilize o decorator `@Controller('domain_name')`.

Sendo que `domain_name` será a rota base ao executar requisições HTTP.

Refira-se a [Nest JS - Controllers](https://docs.nestjs.com/controllers) para mais informações.

### Predefinidos

Classe | Arquivo | Descrição
---|---|---
OrmController | `ormcontroller.ts` | Extenda essa classe para já criar métodos GET, GET:id, POST, PUT, PUT:id, e DELETE:id.
Aplicável apenas se o ORM estiver habilitado.

### Exemplos

Customizado:

```ts
@Controller('user')
export class UserController {

public constructor(private readonly userService: UserService) { }

@Get(':id')
public async getUser(@Param('id') id: string): Promise {
return this.userService.readUser(id);
}

}
```

Basedo em um serviço de ORM:

```ts
@Controller('user')
export class UserController extends OrmController {

public constructor(private readonly userService: UserService) {
super(userService);

this.options = {
dto: {
read: UserReadDto,
create: UserCreateDto,
update: UserUpdateDto,
},
};

}

}
```

## DTO

Data Transfer Objects definem a estrutura de como os dados são transferidos.

Ao criar controllers que recebem dados, você pode injetá-los em argumentos via decorators `@Body()`, `@Query()` e `@Param()`.

Uma vez que definir o tipo deste argumentos como uma classe DTO decorada com opções da biblioteca `class-validator`, suas requisições serão automaticamente validadas.

Crie-os em pastas `*.dto` em arquivos sobre a que método se referem, por exemplo `*.create.dto.ts` ou `*.update.dto.ts`.

Refira-se a [Nest JS - Auto Validation](https://docs.nestjs.com/techniques/validation#auto-validation) para mais informações.

Lista completa de decorators disponíveis em [Class Validator - Decorator Reference](https://github.com/typestack/class-validator#validation-decorators).

### Exemplos

DTO para criação de usuário:

```ts
export class UserCreateDto {

@IsString() @IsNotEmpty()
public name: string;

@IsNumber() @Min(1)
public age: string;

}
```

Definição de tipo no controller:

```ts
@Controller('user')
export class UserController {

public constructor(private readonly userService: UserService) { }

@Post()
// O fato de definirmos UserCreateDto a seguir irá validar a requisição automaticamente
public async postUser(@Body() body: UserCreateDto): Promise {
return this.userService.createUser(body);
}

}
```

## Providers

Serviços são componentes que executam um ou mais dentre extração, transformação e carregamento de dados.

A instanciação de suas classes são gerenciadas pela injeção de depências do framwork, por isso devem ser decorados com `@Injectable()`.

Eles podem ter vários tipos includindo Middlewares, Interceptors, Filters, etc, explicados mais adiante.

Para o serviço principal do seu domínio, utilize o padrão `*.service.ts` de nomenclatura e importe-o na propriedade `providers` do módulo respectivo.

Caso deseje que outros módulos tenha acesso ao seu serviço, é necessário exportá-lo na propriedade `exports`

Refira-se a [Nest JS - Providers](https://docs.nestjs.com/providers) para mais informações.

### Predefinidos

Classe | Arquivo | Descrição
---|---|---
AppProvider | `abstract.provider.ts` | Extenda essa classe para já ter acesso ao logger, variáveis de ambiente e método de retry.
OrmService | `ormservice.ts` | Extenda essa classe para já ter acesso a métodos de manipulação de dados via ORM com gerenciamento de exeções nas queries.
HttpsService | `https.service.ts` | Wrapper sobre o Axios para padronizar exceções HTTP e adicionar amenidades.
LoggerService | `logger.service.ts` | Disponível via AppProvider, realiza integração com Sentry e imprime no console durante desenvolvimento.
RedisService | `redis.service.ts` | Wrapper sobre o redis para ler e persistir dados chave/valor em cloud.

### Exemplos

Customizado:

```ts
@Injectable()
export class UserService {

/** Implemente seus métodos */

public async userHello(): Promise {
this.logger.debug('hello world');
}

}
```

Baseado em serviço ORM:

```ts
@Injectable()
export class UserService extends OrmService {

public constructor(
@InjectRepository(UserEntity)
private readonly userRepository: EntityRepository,
) {
super(userRepository, {
uniqueKey: [ 'name' ],
populate: [ 'team' ],
});
}

}
```

## Middlewares

Executam procedimentos logo que uma requisição HTTP entra e antes de chegar nos interceptors ou controllers.

Devem ser decoradas com `@Injectable()`, implementar `NestMiddleware` e serem aplicadas via `consumer` a nível de módulo.

Refira-se a [Nest JS - Middleware](https://docs.nestjs.com/middleware) para mais informações.

### Predefinidos

Classe | Arquivo | Descrição
---|---|---
AppAuthMiddleware | `app.metadata.middleware.ts` | Extrai o IP e User Agent da requisição e os separa em uma propriedade de metadados

### Exemplos

Implementação:
```ts
import { Injectable, NestMiddleware } from '@nestjs/common';
import { Request, Response } from '../app.interface';

@Injectable()
export class AppMetadataMiddleware implements NestMiddleware {

public async use(req: Request, res: Response, next: any): Promise {
req['metadata'] = {
ip: requestIp.getClientIp(req) || null,
userAgent: req.headers ? req.headers['user-agent'] : null,
};
next();
}

}
```

Importação:

```ts
export class AppModule {

public configure(consumer: MiddlewareConsumer): void {
consumer
.apply(
// Aplique quantos quiser em ordem
AppMetadataMiddleware,
)
// Você pode escolher as rotas com:
// .forRoutes({ method, path }, { method, path });
.forRoutes('*');
}

}
```

## Interceptors

Permitem que manipule dados de uma mesma requisição antes de ela atingir um controller bem como após ser tratada por ele.

Úteis caso precise rastrear a trajetória de uma requisição ou aplicar de maneira unifica algo durante seu retorno.

Seguem o padrão `*.interceptor.ts`, e devem ser decorados por `Injectable()` bem como importados no respectivo módulo.

Também devem implementar a classe `NestInterceptor` e retornar um `Observable` o que permite rastrear a requisição e utilizar métodos `rxjs` em seu témino.

Refira-se a [Nest JS - Interceptors](https://docs.nestjs.com/interceptors) para mais informações.

### Predefinidos

Classe | Arquivo | Descrição
---|---|---
AppLoggerInterceptor | `app.logger.interceptor.ts` | Extrai o IP da requisição bem faz o log da entrada e saída da requisição.
AbstractEntityInterceptor | `ormentity.interceptor.ts` | Executa o método `toJSON()` das entidades antes de retorná-las. Aplicável apenas se utilizado o ORM.

### Exemplo

Implementação:

```ts
@Injectable()
export class AppLoggerInterceptor implements NestInterceptor {

public intercept(context: ExecutionContext, next: CallHandler): Observable {
const req: AppRequest = context.switchToHttp().getRequest();

this.logger.server(`Incoming ${req.url}`);

return next
.handle()
.pipe(
// Utilize dentro do .pipe qualquer método rxjs
finalize(() => {
const res: AppResponse = context.switchToHttp().getResponse();
this.logger.server(`Finished ${req.url} with status ${res.statusCode}`);
}),
);
}

}
```

Importação:

```ts
@Module({
providers: [
{ provide: APP_INTERCEPTOR, useClass: AppLoggerInterceptor },
],
})
export class AppModule { }
```

## Filters

Filtros são decorados com `@Catch()` e servem o propósito de capturar execeções durante execução do contexto.

Neste boilerplate utilizamos um a nível global para padronização de retorno HTTP, mas eles podem ser individualizados a nível de módulo.

Refira-se a [Nest JS - Exception Filters](https://docs.nestjs.com/exception-filters) para mais informações.

### Predefinidos

Classe | Arquivo | Descrição
---|---|---
AppFilter | `app.filter.ts` | Padroniza o retorno HTTP em caso de exceções.

## Testes

Os testes são baseados no framework Jest, todavia para total isolamento nas execuções, o NestJS provê um pacote para compilação individual de módulos temporários `@nestjs/testing`.

Para usufruir desta funcionalidade, configure seus testes com um método `beforeEach()` implementando `createTestingModule()` conforme exemplo a seguir:

```ts
import { Test } from '@nestjs/testing';
import { RedisService } from './redis.service';

describe('RedisService', () => {
const rng = Math.random();
let redisService: RedisService;

beforeAll(async() => {
const testModule = await Test.createTestingModule({
providers: [ RedisService ],
}).compile();

redisService = testModule.get(RedisService);
});

describe('setKey', () => {
it('should persist a random number', async() => {
expect(await redisService.setKey('TEST_KEY', { rng }, 10 * 1000))
.toBeUndefined();
});
});

describe('getKey', () => {
it('should read persisted random number', async() => {
expect(await redisService.getKey('TEST_KEY'))
.toMatchObject({ rng });
});
});
});
```

Agora basta executar `npm run test` e todos os testes descritos em arquivos `*.spec.ts` serão aplicados.

## Migrations

>Para utilizar migrations com versionamento e rollback refira-se a documentação oficial: [Mikro ORM - Migrations](https://mikro-orm.io/docs/migrations/)

Este boilerplate permite sincronizar a modelagem atual com as entidades definidas nos arquivos `*.entity.ts`.

Por padrão, é possível possuir até três ambientes configurados em arquivos `.env` separados sendo:

```bash
.env # Desenvolvimento
.env.staging # Homologação
.env.production # Produção
```

Para realizar a migração de sincronismo execute:

```bash
npm run orm:sync:dev # Utiliza arquivo .env
npm run orm:sync:stg # Utiliza arquivo .env.staging
npm run orm:sync:prd # Utiliza arquivo .env.production
```

Caso prefira apenas visualizar a migração e não executá-la, substitua o script `orm:sync` por `orm:dump`. As queries serão impressas no console.

## Templates

Criar serviços, controllers e DTOs de algo repetitivo pode ser bastante tedioso, sendo assim foi desenvolvido uma maneira mais simples para realizar estas implementações.

### API

Cria todos os arquivos recomendados para implementação de um serviço de API externo.

Dado um domínio de sua escolha, por exemplo: `gmaps`, execute o script:

```
npm run template:api -- -n gmaps
```

A seguinte estrutura de arquivos será criada dentro de `/source`:

```
gmaps
|- gmaps.dto
| - index.ts
|- gmaps.interface
| - index.ts
|- gmaps.service.ts
|- gmaps.config.ts
```

Agora, por padrão, as variáveis de ambiente `GMAPS_HOST` e `GMAPS_AUTH` serão mandatórias. Você pode configurar a validação no arquivo `gmaps.config.ts`.

A estratégia de autenticação do modelo é colocar o `*_AUTH` no header `Authorization`. Dependendo do serviço será necessário modificar.

### CRUD

Cria todos os arquivos recomendados para implementação de uma entidade com métodos HTTP de GET, GET:id, POST, PUT, PUT:id, e DELETE:id.

Dado um domínio de sua escolha, por exemplo: `user`, execute o script:

```
npm run template:crud -- -n user
```

A seguinte estrutura de arquivos será criada dentro de `/source`:

```
user
|- user.dto
| - index.ts
| - user.create.dto.ts
| - user.read.dto.ts
| - user.update.dto.ts
|- user.controller.ts
|- user.entity.ts
|- user.module.ts
|- user.service.ts
```

Agora é só definir a entidade em `user.entity.ts` e configurar a validação dos dtos que todos os métodos estarão implementados.