Hexafood

Índice

⛵ Sobre o projeto

Esse projeto faz parte do trabalho "Tech Challenge - Fase 02", ministrado no segundo módulo do curso de Pós Graduação Software Architecture da FIAP em parceria com a Alura.

Para exercitar os conceitos apresentados nas matérias do curso, sendo elas sobre Kubernentes Clean Architecture, a aplicação backend desenovlvida em Typescript no framework NestJS, foi adaptada a fim de representar os conceitos da Arquitetura Limpa, e também foi criado scripts de configuração de um cluster kubernetes.

🔨 Tecnologias:

🚀 Como rodar esse projeto

Se você estiver usando Windows, vai precisar do WSL para rodar esse projeto de forma prática. Para isso, você pode instalá-lo seguindo o seguinte tutorial. Também será necessário uma distribuição linux para utilizar o WSL. Recomendo o Ubuntu que pode ser baixando na própria Microsoft Store no link. Depois, vai precisar do Docker, o qual a versão de Windows pode ser encontrada aqui. Então, clone o projeto dentro do WSL, vá para pasta dele e execute o comando:

docker compose build --no-cache

Após a construção da imagem, basta executar o comando:

docker compose up

O projeto estará executando no endereço http://localhost:3000/.

Para limpar o volume db do docker, execute o comando: docker-compose down -v

🔌 Infraestrutura K8S

O Kubernetes(K8S) é uma plataforma de orquestração de contêineres open-source que automatiza o deployment, o escalonamento e a gestão de aplicações contêinerizadas. Ele agrupa contêineres que compõem uma aplicação em unidades lógicas para fácil gestão e descoberta de serviços, oferecendo recursos poderosos como balanceamento de carga, armazenamento persistente e monitoramento, tudo isso de forma declarativa através de arquivos de configuração.

Atendendo aos critérios do Tech Challenge - Fase 02, foi aprovisionado um cluster K8S com as seguintes configurações:

  • Deployment da aplicação com ao menos 2 pods;
  • Service Load Balancer do tipo NLB ou ALB
  • Configurações de acesso aos servicos parametrizados via Secrets

Representação visual de arquitetura hexagonal


Além dos critérios do desafio, foi criado um Job para execução das migrations. Dessa forma, como boa prática separamos essa responsabilidade do ciclo de vida de aprovisionamento dos pods da API.

metadata:
  name: api-migration
spec:
  template:
    spec:
      containers:
        - name: api-migration
          image: marayza/hexafoodk8s:v15
          command: ['/bin/sh', '-c']
          args:
            - './wait-for-it.sh postgres-service:5432 -- yarn prisma migrate dev && yarn prisma db seed'
          envFrom:
            - secretRef:
                name: postgres-secret
      restartPolicy: OnFailure

Para testar o cluster, supondo que você tenha minikube e kubectl configurados na sua máquina, navegue até a pasta "k8s"e execute:

kubectl apply -f .

Depois só aguardar o aprovisionamento do ambiente e testar API, com a URL gerada pelo load balancer:

minikube service api-lb-service --url

📂 Clean Architecture na prática

A Clean Architecture é um conjunto de princípios de design de software que busca promover a separação de preocupações e a criação de sistemas desacoplados e testáveis. Concebida por Robert C. Martin ("Uncle Bob"), essa arquitetura prioriza a independência de frameworks, interfaces de usuário e bancos de dados, colocando as regras de negócio no centro do design. Assim, permite uma maior flexibilidade e facilidade de manutenção, tornando o sistema mais robusto e adaptável a mudanças.

Durante a Fase 01, foi utilizado a Arquitetura Hexagonal (Ports and Adpaters), o que facilitou a conversão da aplicação para Clean Architecture. Pois ambas pregam a cerca do isolamento do coração da aplicação com o mundo exterior. Tudo que for inerente a lógica do negócio e do domínio, deve ser considerado um detalhe, e por isso o design de código deve se guiar pelo desacoplamento.

Conforme a modelagem do fluxo de realização de pedido obtida através da técnica do Event Storming do DDD (disponível para melhor visualização aqui), foram identificados três agregados os quais consideramos também potenciais candidatos a serem Contextos Delimitados:

  • Identificação
  • Pedido
  • Pagamento

Event Storming do fluxo de pedidos do sistema


Dessa forma, visando aproveitar a estrutura modular do NestJS, os três foram separados em módulos:
.
├── src (separação do código fonte por módulos)
│   ├── identificacao
│   ├── pedido
│   └── pagamento
├── test
│   └── [arquivos de teste]
├── package.json
├── package-lock.json
└── README.md

Adentrando no módulo Identificação para exemplificação, temos a representação da arquitetura partindo de dois diretórios:

  • Infrastructure: onde estarão todos os detalhes de implementação, como gateways que se comunicam com mundo exterior e recursos inerentes a regra de negócio;
  • Core: onde estará o coração do software, isolando o que concerne o domínio da aplicação (domain), nesse caso as entidades que representam o negócio, e a lógica da aplicação (application), a qual visando atender ao Princípio da Responsabilidade Única (SRP) da Clean Architecture, é separada Casos de Uso (Use Cases).
.
├── src
│   ├── identificacao
│   │   ├── infrastructure
│   │   │   ├── controller
│   │   │   │   └── clientes.controller.ts
│   │   │   ├── filter
│   │   │   └── gateway
│   │   └── core
│   │       ├── application
│   │       │   ├── exceptions
│   │       │   └── use-cases
│   │       │       ├── create.cliente.usecase.ts
│   │       │       ├── find.cliente.usecase.ts
│   │       │       ├── identify.cliente.usecase.ts
│   │       │       └── dto
│   │       └── domain
│   │           ├── entities
│   │           │   └── cliente.entity.ts
│   │           ├── factory
│   │           └── repository
│   │               └── clientes.repository.ts

Para se comunicar com o mundo exterior, assim como na Arquitetura Hexagonal a camada interna deve continuar utilizando "Ports". Com isso, pode-se traçar uma "fronteira" que separa o que é lógica do domínio e detalhe de implementação, o que representa essa separação de fato é o uso de interfaces, como no caso do IClienteResository que é utilizado para abstrair a necessidade da lógica de aplicação interagir com banco de dados (mundo externo). Essa separação fica melhor descrita visualmente na figura a seguir:

Event Storming do fluxo de pedidos do sistema


Além da organização de diretórios, foi adotado outras boas práticas a fim de reforçar os conceitos da Clean Architecture. Cada Use Case é representado por um arquivo diferente, e todos seguem o design Command Pattern, tendo um método único "execute" para representara execução do caso de uso. Dessa forma, o código se torna mais organizado estimulando a coesão e o Princípio da Responsabilidade Única.

export class CreateClienteUseCase {

  ...

  async execute(data: InputClienteDto): Promise<OutputClienteDto> {
    const exists = await this.clientesRepository.existsByCpf(data.cpf);
    if (exists) {
      throw new ClienteException('Cliente já cadastrado');
    }
    const cliente = await this.clientesRepository.create(
      new Cliente(data.nome, data.cpf),
    );
    return {
      id: cliente.id,
      nome: cliente.nome,
      cpf: cliente.cpf,
    };
  }
}

A fim de isolar cada camada da arquitetura, cada use case deve trabalhar com DTOs (Data Transfer Objects). Dessa forma as camadas internas não ficam expostas a cada mais externas, a fim de diminuir o acoplamento do código.

Outra boa prática adotada foi o uso Factories para abstração de complexidades referentes a criação de objetos. Por exemplo, a entidade Cliente pode ser criada sem ter um ID e sem ter a data de criação, pois quem fornece essas informações é o repositório onde a entidade é registrada (como o banco de dados). Em outra situação, quando a entidade é carregada do repositório, ela já terá essas informações.

Desta forma, para abstrair essa complexidade na hora de instanciar uma entidade Cliente no sistema, de ter que verificar se os valores estão presentes ou não, foi criado uma Factory com o objetivo de instanciar entidades "Cliente" a partir de uma conjunto bruto de dados:

export class ClienteFactory {
  static create(data: any): Cliente {
    const cliente = new Cliente(data.nome, data.cpf);
    cliente.id = data.id || null;
    cliente.createdAt = data.createdAt || null;
    return cliente;
  }
}

📓 Lógica de negócio (domínio) aplicada

Esse projeto consistem uma API para atender as necessidades de uma lanchonete que deseja automatizar os seus pedidos através de dispositivos de autoatendimento. Baseando-se na modelagem do Event Storming, foram identidicados os seguintes casos de uso:

Identificação

Primeiramente tem-se a fase da identificação, na qual o cliente poderá ou não estar se identificando. Caso ele seja um novo cliente, ele poderá ser cadastrar conforme endpoint (cada um já está linkado com a referência no swagger para testes, considerando que ele está sendo executado em http://localhost:3000):

/clientes

{
  "nome": "Senhor Teste 1",
  "cpf": "12345678910"
}

O sistema efetuará as seguintes validações:

  • O CPF não pode conter menos de 11 dígitos
  • O CPF não pode ter sido cadastrado em outro cliente

Caso o cliente já seja cadastrado, ele poderá se identificar pelo endpoint:

/clientes/{cpf}

Também é possível que o cliente prossiga sem se identificar. Dessa forma, o autoatendimento apenas não informará a identificação do mesmo na hora de efetuar o pedido.

Pedido

Para composição do pedido, primeiramente é necessário consultar as categorias cadastradas no sistema:

/categorias O cliente poderá escolher entre umas das categorias do sistema:

[
  {
    "id": 1,
    "nome": "Lanche"
  },
  {
    "id": 2,
    "nome": "Acompanhamento"
  },
  {
    "id": 3,
    "nome": "Bebida"
  },
  {
    "id": 4,
    "nome": "Sobremesa"
  }
]

Com a categoria escolhida, o sistema deverá listar os produtos filtrados, os quais o usuário poderá estar adicionando aos itens do pedido, na sequência e quantidade que desejar:

/produtos/{id_categoria}

[
  {
    "id": 4,
    "nome": "Hexa Dog",
    "id_categoria": 1,
    "valor": 12.9,
    "descricao": "Cachorro quente minimalista",
    "imagem": "https://images.unsplash.com/photo-1612392062422-ef19b42f74df?ixlib=rb-4.0.3&ixid=M3wxMjA3fDB8MHxwaG90by1wYWdlfHx8fGVufDB8fHx8fA%3D%3D&auto=format&fit=crop&w=1770&q=80",
    "createdAt": "2023-07-05T01:39:43.422Z",
    "updatedAt": "2023-07-05T01:39:43.422Z"
  },
  ...
]

Após escolher todos itens necessário, o autoatendimento deve submeter o pedido completo no seguinte endpoint:

/pedidos

{
  "id_cliente": 1,
  "itens": [
    {
      "quantidade": 1,
      "id_produto": 4
    },
    {
      "quantidade": 3,
      "id_produto": 7
    },
    {
      "quantidade": 2,
      "id_produto": 9
    },
    {
      "quantidade": 3,
      "id_produto": 11
    }
  ]
}

O pedido consiste na identificação do cliente (se estiver identificado), e os itens escolhidos com a quantidade de cada um.

O pedido será submetido a uma etapa de processamento do pagamento que será explicada no próximo tópico. Por hora, nesse momento será exemplificado a etapa pós pagamento.

Após o processamento do pagamento, o pedido tem o seu STATUS alterado para "RECEBIDO". Dessa forma, ele deve estar disponível para consulta de pedidos pendentes no painel da cozinha, através do seguinte endpoint:

/pedidos/consultar_pedidos_pendentes

[
  {
      "codigo_pedido": "ZG53O3",
      "status": "RECEBIDO",
      "id": 9,
      "createdAt": "2023-07-08T23:28:13.810Z",
      "updatedAt": "2023-07-08T23:28:13.810Z",
      "cliente": {
        "id": 1,
        "nome": "Sammy Abshire",
        "cpf": "50808605066",
        "createdAt": "2023-07-08T19:25:03.451Z"
      },
      "valor_total": 153.9,
      "itens": [
        {
          "id": 25,
          "quantidade": 3,
          "valor": 14.5,
          "id_produto": 11,
          "produto": {
            "id": 11,
            "nome": "Hexa Brownie",
          }
        },
        ...
      ]
    }
  ...
]

Dessa forma, o pedido entra numa fila e fica disponível de acordo com a ordem de criação (o pedido mais antigo para o mais recente) para que algum usuário atuante da Cozinha, inicie a preparação. Ele deve dar algum comando para painel, para que o status do pedido seja atualizado no seguinte endpoint:

/pedidos/{id}/iniciar_preparacao

{
  "id": 9,
  "codigo_pedido": "ZG53O3",
  "valor_total": 153.9,
  "status": "EM_PREPARACAO",
  "createdAt": "2023-07-08T23:28:13.810Z",
  "updatedAt": "2023-07-08T23:28:13.810Z",
  "id_cliente": 1
}

Dessa forma o STATUS do pedido passa para a "EM_PREPARACAO". Ao finalizar a preparação, o usuário da Cozinha deve chamar o endpoint:

/pedidos/{id}/finalizar_preparacao

{
  "id": 9,
  "codigo_pedido": "ZG53O3",
  "valor_total": 153.9,
  "status": "PRONTO",
  "createdAt": "2023-07-08T23:28:13.810Z",
  "updatedAt": "2023-07-08T23:28:13.810Z",
  "id_cliente": 1
}

O STATUS do pedido passa a ser "PRONTO", o que significa ele já pode ser retirado pelo cliente. Quando isso acontecer, deve ser dado o comando para atualizar o STATUS do pedido novamente:

/pedidos/{id}/finalizar_pedido

{
  "id": 9,
  "codigo_pedido": "ZG53O3",
  "valor_total": 153.9,
  "status": "FINALIZADO",
  "createdAt": "2023-07-08T23:28:13.810Z",
  "updatedAt": "2023-07-08T23:28:13.810Z",
  "id_cliente": 1
}

O pedido passa para "FINALIZADO" e se encerra o fluxo.

Em todo processo é possível consultar pedido atual através do código gerado pelo sistema, através do seguinte endpoint:

/pedidos/{codigo_pedido}

Pagamento

Antes do pedido prosseguir para cozinha para preparação, ele deve ter seu pagamento processado. Como isso é feito por um sistema externo, foi optado por fazer isso de forma assíncrona.

Quando um pedido é criado, primeiramente é definido um STATUS de "INICIADO", e disparado um evento no sistema. Foi utilizado um recurso do nestjs chamado "event-emitter", para diminuir acoplamento e separar responsabilidades. O funcionamento se dá seguinte forma:

async createNewPedido(pedidoDto: PedidoDTO) {
  //logica de validação e criação de um novo pedido
  ...
  pedido.status = StatusPedido.INICIADO;
  ...
  
  console.log('Novo pedido criado: ', pedido);
  this.eventEmitter.emit('novo.pedido', new NovoPedidoEvent(pedido));

  return pedido;
}

Na criação do pedido é disparado o evento NovoPedidoEvent que recebe o pedido criado como parâmetro. Dessa forma, é possível implementar uma outra classe que ficará escutando esses eventos, e dessa forma executará uma lógica adicional. Nesse caso, definimos o NovoPedidoListener:

@Injectable()
export class NovoPedidoListener {
  constructor(
    private createPagamentoUseCase: CreatePagamentoUseCase,
  ) {}

  @OnEvent('novo.pedido')
  async handle(event: NovoPedidoEvent) {
    const pedido = event.pedido;
    //lógica de criação de DTO para criação do pagametno

    if (pedido.cliente) {
      pagamentoDto.cliente = {
        id: pedido.cliente.id,
        nome: pedido.cliente.nome,
        cpf: pedido.cliente.cpf,
      };
    }
    //chama o usse case para criar pagamento
      return this.createPagamentoUseCase.execute(
          pagamentoDto,
        );
  }
}

Diferente a primeira implementação, na Fase 02 o listener NovoPedidoListener passa a ser responsável apenas para chamar o fluxo de criação de pagamentos, mas deixa de ser responsável por atualizar o pedido. A fim de representar o funcionamento de um Webhook, foi criado um outro evento chamado "PagamentoProcessado". Esse evento deve ser disparado a partir de um retorno da API de pagamento, na qual é chamado o "UpdatePagamentoUseCase", verifica o status do pagamento e chama o use case de atualização do pedido:

export class UpdatePagamentoUseCase {
  ...

  async execute(data: Pick<PagamentoDto, 'id'|'id_pedido' | 'status'>) {
    const { status, id_pedido, id } = data
    const pedido = await this.FindPedidoByIdUseCase.findById(id_pedido)
    if (!pedido) {
      throw new PagamentosException('O Pedido informado não existe.');
    }
    if(status === 'APROVADO'){
      pedido.status = StatusPedido.RECEBIDO
      await this.updatePedidoUseCase.execute(pedido);
    }else{
      pedido.status = StatusPedido.CANCELADO
      await this.updatePedidoUseCase.execute(pedido);
    }
    return this.pagamentosRepository.update(id, status)
  }
}

O restante da lógica se mantém, sendo caso o pagamento seja aprovado, o pedido passa a ter o status "RECEBIDO" ficando disponível para preparação não seja aprovado, e caso não seja aprovado, o pedido tem o status "CANCELADO" e não avança para fila de preparação. Uma representação visual desse fluxo consta na figura a seguir:

Fluxo de pagamentos


```

Estrutura de dados

Nosso banco de dados consiste em cinco tabelas. A primeira é a tabela de categorias de produtos, como lanches, refrigerantes, etc. Em seguida, temos a tabela de clientes, na qual não gerenciamos as informações de login, por isso não armazenamos senhas. Temos também a tabela de produtos em si, e a tabela de pedidos, que relaciona os itens dos produtos com os clientes. Por fim, temos a tabela de valores e status dos produtos.

Estrutura dos dados

Banco de dados

Optamos pelo banco de dados SQL Postgres devido a alguns motivos. Primeiro, a familiaridade dos membros da equipe com essa tecnologia. Além disso, desejávamos estabelecer relacionamentos entre tabelas, como clientes e pedidos, categorias de produtos e itens que relacionam pedidos e produtos. Também levamos em consideração que não teríamos um grande volume de requisições diárias, evitando sobrecarregar o banco de dados. Com base nesses aspectos, a escolha pelo Postgres foi a mais adequada.

Pipeline

A nossa pipeline é configurada para ser executada automaticamente em cada push que realizamos, bem como em cada pull request direcionado para a branch principal (main). Essa pipeline é executada no ambiente do ubuntu-latest e utiliza a versão do Node 16.14.x para realizar os processos de instalação de dependências durante a etapa de run: yarn, para compilar o projeto com o comando run: yarn build e também para executar os testes por meio do comando run: yarn test.

É importante destacar que a branch main possui restrições de merge caso alguma etapa da nossa pipeline falhe. Não é permitido fazer o merge se uma etapa não for concluída com êxito, e não é possível realizar commits diretamente na branch main.

Uma vez concluídas as etapas de compilação e testes, procedemos com o processo de implantação da imagem gerada no Docker para o Docker Hub. Para garantir a segurança, evitamos incluir informações sensíveis, como credenciais, diretamente no código YAML. Em vez disso, utilizamos o painel de secrets fornecido pelo próprio GitHub para armazenar e utilizar essas credenciais no código YAML, por exemplo, ${{ secrets.DOCKERHUB_USERNAME }}.

Fluxo da pipeline

📑 Licença

Este projeto esta sobe a licença MIT. Veja a LICENÇA para saber mais.

😉 Autores

Feito com ❤️ por:

Voltar ao topo