Skip to content

JackS1o/Delivery-Challenge

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

51 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Boas-vindas ao repositório do projeto ENAFOOD!

O que foi desenvolvido

Neste projeto, foi desenvolvido uma API para um sistema de delivery de comida. O sistema possui um banco de dados com informações de produtos e usuários, e permite que usuários façam pedidos de produtos cadastrados no banco de dados.

A API permite obter produtos disponíveis para compra e manipular a sacola do usuário, incluindo a inserção, escolha de quantidades e remoção de itens. Para a Fase 1, onde o EnaFood possuía cerca de 100 usuários, cada um fazendo em média 5 pedidos por mês, a API foi desenvolvida com a capacidade de gerenciar pedidos com 1 a 5 produtos.

Na Fase 2, que contava com cerca de 10.000 usuários e uma média de 10 pedidos por mês, a API foi atualizada para lidar com pedidos que continham de 1 a 15 produtos. Na Fase 3, que contava com cerca de 1.000.000 de usuários e uma média de 25 pedidos por mês, a API foi novamente atualizada para lidar com pedidos que continham de 1 a 20 produtos.

Por fim, na Fase 4, onde o EnaFood havia se tornado a principal rede de delivery do Brasil, com cerca de 50.000.000 de usuários, a API passou por uma última atualização para lidar com pedidos que também eram feitos por empresas, com um grande volume de produtos e preços mais em conta.

Como executar o projeto
  1. Clone o repositório git clone git@github.com:JackS1o/Delivery-Challenge.git

  2. Na raíz do projeto, execute o comando docker-compose up -d para subir o container da aplicação e do banco de dados. O servidor estará disponível na porta 3001.

  3. Na raíz do projeto, execute o comando npm run products:import para popular o banco de dados com os produtos.

Rotas desenvolvidas
  • POST /register - Cria um novo usuário. O corpo da requisição deve conter o email e password desejados. Exemplo de corpo de requisição:
{
  "email": "jack@gmail.com",
  "password": "123456",
}

a resposta será um objeto com o um token de autenticação no seguinte formato:

{
  "token": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJlbWFpbCI6ImphQGdtYWlsLmNvbSIsImlhdCI6MTY3ODY2NTU3NiwiZXhwIjoxNjc4NjY5MTc2fQ.svDiQTwItv4FG9_0OZNrxmKWb0EWRPsk8LFbYAtUIAc"
}
  • POST /login - Realiza o login de um usuário. O corpo da requisição deve conter o email e password que existam no banco de dados. O retorno dessa requisição tabém gera um token. Exemplo de corpo de requisição:
{
  "email": "jack@gmail.com",
  "password": "123456",
}
  • GET /products - Retorna todos os produtos cadastrados no banco de dados. Exemplo de resposta:
[
  {
    "_id": "640ebdb605aaccc707ab1507",
    "name": "Coca-Cola",
    "price": 5,
    "description": "Coca-Cola is a carbonated soft drink manufactured by The Coca-Cola Company."
  },
  {
    "_id": "640ebdb605aaccc707ab1508",
    "name": "Pepsi",
    "price": 5,
    "description": "Pepsi is a carbonated soft drink manufactured by PepsiCo."
  }
]
  • GET /orders - Retorna todas as ordens de compra cadastradas no banco de dados. Exemplo de resposta:
[
  {
    "_id": "640ebdb605aaccc707ab1507",
    "user": "jack@gmail.com",
    "paymentMethod": "cartão",
    "address": "Rua São Miguel",
    "order": [
      {
        "_id": "640ebdb605aaccc707ab1507",
        "name": "Coca-Cola",
        "price": 5,
        "description": "Coca-Cola is a carbonated soft drink manufactured by The Coca-Cola Company.",
        "quantity": 2
      },
      {
        "_id": "640ebdb605aaccc707ab1508",
        "name": "Pepsi",
        "price": 5,
        "description": "Pepsi is a carbonated soft drink manufactured by PepsiCo.",
        "quantity": 2
      }
    ]
  }
]
  • GET /user/orders - Retorna todas as ordens de compra de um usuário específico. Exemplo de resposta:
{
  "_id": "640ebdb605aaccc707ab1507",
  "user": "jack@gmail.com",
  "paymentMethod": "cartão",
  "address": "Rua São Miguel",
  "order": [
    {
      "_id": "640ebdb605aaccc707ab1507",
      "name": "Coca-Cola",
      "price": 5,
      "description": "Coca-Cola is a carbonated soft drink manufactured by The Coca-Cola Company.",
      "quantity": 2
    }
  ]
}
  • GET /order/:id - Retorna uma ordem de compra específica. Exemplo de resposta:
{
  "_id": "640ebdb605aaccc707ab1507",
  "user": "jack@gmail.com",
  "paymentMethod": "cartão",
  "address": "Rua São Miguel",
  "order": [
    {
      "_id": "640ebdb605aaccc707ab1507",
      "name": "Coca-Cola",
      "price": 5,
      "description": "Coca-Cola is a carbonated soft drink manufactured by The Coca-Cola Company.",
      "quantity": 2
    }
  ]
}
  • DELETE /order/:id - Deleta uma ordem de compra específica. Exemplo de resposta:
{
  "message": "Sale deleted successfully!",
  "order": {
    "_id": "640e204bff9bfc2adbe8933f",
    "user": "jack@gmail.com",
    "paymentMethod": "cartão",
    "address": "Rua São Miguel",
    "order": [
      {
        "_id": "640ebdb605aaccc707ab1507",
        "name": "Coca-Cola",
        "price": 5,
        "description": "Coca-Cola is a carbonated soft drink manufactured by The Coca-Cola Company.",
        "quantity": 2
      }
    ]
  }
}
  • POST /mvp/order - é possível criar uma nova ordem de compra, porém, nessa rota, o usuário consegue criar uma ordem com no máximo 5 produtos.
  • POST /earlyadopters/order - é possível criar uma nova ordem de compra, porém, nessa rota, o usuário consegue criar uma ordem com no máximo 15 produtos.
  • POST /earlymajority/order - é possível criar uma nova ordem de compra, porém, nessa rota, o usuário consegue criar uma ordem com no máximo 20 produtos.
  • POST /latemajority/order - é possível criar uma nova ordem de compra sem limite de produtos.
  • Todas as rotas de criação de ordem de compra recebem o seguinte corpo de requisição:
{
  "paymentMethod": "cartão",
  "address": "Rua São Miguel",
    "order": [
    {
      "_id": "640ebdb605aaccc707ab1507",
      "name": "Coca-Cola",
      "price": 5,
      "description": "Coca-Cola is a carbonated soft drink manufactured by The Coca-Cola Company.",
      "quantity": 2
    }
  ]
}
  • As rotas PATCH /mvp/order/:id, PATCH /earlyadopters/order/:id, PATCH /earlymajority/order/:id e PATCH /latemajority/order/:id são responsáveis por atualizar uma ordem de compra específica. Todas as rotas recebem o seguinte corpo de requisição:
{
  "paymentMethod": "cartão",
  "address": "Rua São Miguel",
    "order": [
    {
      "_id": "640ebdb605aaccc707ab1507",
      "name": "Coca-Cola",
      "price": 5,
      "description": "Coca-Cola is a carbonated soft drink manufactured by The Coca-Cola Company.",
      "quantity": 2
    }
  ]
}
Testes
  • Para executar os testes, execute o comando npm test na raíz do projeto.
  • Os testes foram desenvolvidos utilizando Mocha, Jest, Chai e Sinon.
  • Os testes foram desenvolvidos para as camadas de Controller do projeto.
Desenvolvimento
  • O projeto foi desenvolvido utilizando arquitetura MSC (Model, Service e Controller) para uma melhor organização do código e também middlewares para verificar se os dados enviados pelo usuário estão corretos, se o usuário está autenticado, se o produto enviado pelo usuário existe, se a quantidade de produtos enviados pelo usuário é válida, entre outros.

  • errorHandler - Middleware que captura os erros e retorna uma mensagem de erro.

  • authToken - Middleware que verifica se o usuário está autenticado. Caso não esteja, retorna um erro.

  • validateLogin - Middleware que verifica se os dados enviados pelo usuário ao realizar o login ou se registrar estão corretos. Caso não estejam, retorna um erro.

  • validateOrderFields - Middleware que verifica se os dados enviados pelo usuário ao realizar uma nova ordem de compra estão corretos. Caso não estejam, retorna um erro.

  • productExists - Middleware que verifica se o produto enviado pelo usuário ao realizar uma nova ordem de compra existe. Caso não exista, retorna um erro.

  • invalidQuantityMVP - Middleware que verifica se a quantidade de produtos enviados pelo usuário ao realizar uma nova ordem de compra na rota /mvp/order é válida. Caso não seja, retorna um erro.

  • invalidQuantityEarlyAdop - Middleware que verifica se a quantidade de produtos enviados pelo usuário ao realizar uma nova ordem de compra na rota /earlyadopters/order é válida. Caso não seja, retorna um erro.

  • invalidQuantityEarlyMajor - Middleware que verifica se a quantidade de produtos enviados pelo usuário ao realizar uma nova ordem de compra na rota /earlymajority/order é válida. Caso não seja, retorna um erro.

  • invalidUpdatedQuantityMVP - Middleware que verifica se a quantidade de produtos enviados pelo usuário ao atualizar uma ordem de compra na rota /mvp/order/:id é válida. Caso não seja, retorna um erro.

  • invalidUpdatedQuantityEarlyAdop - Middleware que verifica se a quantidade de produtos enviados pelo usuário ao atualizar uma ordem de compra na rota /earlyadopters/order/:id é válida. Caso não seja, retorna um erro.

  • invalidUpdatedQuantityEarlyMajor - Middleware que verifica se a quantidade de produtos enviados pelo usuário ao atualizar uma ordem de compra na rota /earlymajority/order/:id é válida. Caso não seja, retorna um erro.

  • invalidUpdatedFields - Middleware que verifica se os dados enviados pelo usuário ao atualizar uma ordem de compra estão corretos. Caso não estejam, retorna um erro.

Tecnologias utilizadas
  • O projeto foi desenvolvido utilizando o Docker para a criação de containers.
  • Linguagem Node.js e banco de dados MongoDB.
  • Express para a criação das rotas.
  • Mongoose para a conexão com o banco de dados.
  • JWT para a autenticação de usuários.
  • Mocha, Jest, Chai e Sinon para a realização dos testes.
  • Joi para a validação de dados.
  • Express Async Errors para a tratativa de erros.
  • Dotenv para a utilização de variáveis de ambiente.