Projeto Tech Challenge - FIAP Alura - 3º Fase
Contextos das APIs que serão disponibilizadas para o controle de Parquimetro
* Condutor
* Veiculo
* Estacionamento
Neste momento as APIs irão fornecer todos os métodos necessários para um parquimetro, optamos em não disponibilizar o metodo de DELETE, pois entendemos que as informações não devem ser excluídas neste tipo de negócio.
Optamos em persistir os dados em uma base NOSQL e no caso o MONGODB , por ser Open Source e com muitas recomendações do mercado, além de garantir a escalabildiade horizontal quando a cidade receber mais visitantes, possibilitado através da criação de clusters, neste momento esta usando apenas apenas um de forma mais simples, mas esta aplicação nos permite criarmos rapidamente quantos cluster forem necessários.
Em um tópico abaixo iremos falar com amsi detalhes sobre o Cluster e tantos outros pontos do MongoDB e do Spring Data.
1) Baixar via GIT : git clone https://github.com/ArleiPacanaro/estacionamento.git
2) Disponibilizamos 3 formas de execução:
2.1 Cloud : Via cloud para evitarmos cobranças e para avaliarmos como MVP1, usamos para a aplicação o Render e para o banco de dados o Atlas MongoDB (Permite adiconar cluster, fazer backups e etc) , seguimos os seguintes passos: Geramos o .jar da aplicação: no diretório do projeto: mvn install -U Geramos a imagem com o docker da raiz: docker build -t arleilepiani/tech:3.0 . Conectamos no docker hub central (após cadastro de conta), criamos o respositório tech como publico Conectamos no docker hub central via linha de comando da maquina local com o projeto: docker login Vinculamos "tagueamos" docker tag arleilepiani/tech:3.0 arleilepiani/tech:3.0 Subimos a imagem no docker hub : docker push arleilepiani/tech:3.0 Após isto e criar uma conta no Render, optamos pela Free e existem 2 formas de deploys para Web Services, via Git ou ou repostporio de imagens como o dcker hub que foi o que seguimos. Usamos basicamente as configruações default , até pela questão de custos, mas temos opções de ajustar o dominio da aplicação e mais alguns detalhes, apontando no caso para nossa imagem: docker.io/arleilepiani/tech:4.0 Bom com estes passos a url esta disponivel: nossa aplicação na Web na cloud do Render e nosso banco na cloud da Atlas MongoDB, com isto facilmemnete podemos mudar para qualquer outra cloud , pois temos pouco acomplamento com alguma cloud em especifico e podemos mudar e evoluir nossa API com Gateway , Autenticação, Balance Loaders e etc, conforme for auymentando a demanda de requisições.url: https://estacionamentoarlei.onrender.com/swagger-ui/index.html Com as ferramentas de cloud que aplicamos, podemos acompanhar as métricas de aplicações e de banco de dados e avaiar a necessidade de escalabilidade, em termos de banco o objetivo e não tirar na Atlas mas na aplicação podemos implementar por exemplo na AWS , usando os reccursos Serverless do FARGATE que faz esta escalabilidade automática da aplicação e no banco de dados podemos ir adicionando clusters e criando indices conforme a necessidade. <b>2.2 Local pelo Intellij ou IDE de sua preferencia com um mongoDB Local:</b> Após a execução do item 1 que é clonar a aplicação do GIT, importar ou abrir o projeto na sua Ide de preferencia e executar o projeto, alterando ao aplication.properties para para a 1 conexão e comentando as demais. Neste caso não precisa do Docker Cliente Instalado. <b>2.3 Docker-Compose:</b> Executar em máquina local : na linha de comando ir até a pasta que clonou o projeto e no prompt da linha de comando, executar: ir até a pasta: TechChallenge e executar o comando: docker build -t tech:3.0 . ir até a pasta: TechChallenge e executar o comando: docker compose up -d Neste ponto usar a 2 conexão do aplication properties.
Tanto pelo sua Ide de preferencia, quanto pelo docker compose, poderá acessar o swagger pela url:
Premissa para execução via docker: Docker instalado na máquina client
Em nossa aplicação temos todas as Collections na pasta raiz para poderem realizar os testes.
Liguagem: Java versão 17
Arquitetura: MVC - Model View Controller na fase1 , na fase 2 partimos para o inicio de um modelo hexagonal, pois consideramos mais interessante diante dos relacionamentos e para o futuro do projeto nas demais fases
Frameworks e Bibliotecas: Spring Boot , LOMBOK versão default para a versão do Spring, Bean Validation versão default para a versão do Spring, Spring Data
Ferramentas Colaborativas: Docker,Postman,GIT,Maven e Swagger
Banco de Dados: MongoDB
O objetivo deste tópico é explicar as motivações e desafios das técnicas e tecnologias aplicadas nesta fase do Tech Challenge.
A Framework Spring é Open Source e foi criado pela PIVOTAL com o objetivo de facilitar o desenvolvimento de sistemas, descomplicando toda a parte de infraestrutura e algumas outras complexidades que existe na especificação JAVA EE.
A Framework Spring Boot é uma versão reduzida da Spring, ideal para o desevolvimento de aplicações WEB stand alone de forma rápida e simples, foi baseada no conceito de Injeção de Dependências e Inversão de Controle o que torna o código mais coeso e com menos acoplamento entre as classes, além de gerenciar bem todas as dependências iniciais internas e externas, ter de forma embracada o servidor de aplicação, sendo por padrão o Apache-Tomcat, mas podemos usar outros como o Jetty e Undertown , durante a descrição deste documento iremos registrar aonde são aplicados estes conceitos e facilidades com base em anotações e na estruturação das camadas, sendo estes o principais motivos de optarmos por esta framework.
Internamente o Spring Boot possui 4 componentes chaves na sua execução:
1 - Spring Boot Starter: Utiliza a combinação de várias dependências para formar uma unica
2 - Spring Boot CLI: software responsável por rodar e testar as aplicações na linha de comando para isto executando o Spring Boot Starter e o Spring Boot Autocofigurator
3 - Spring Boot Autocofigurator: Gerencia as configurações de uma aplicação
4 - Spring Boot Acutator: Prover o endpoints e as métricas da aplicação
APIs (Application Programming Interface) é o meio utilizado para que diversas aplicações possam se comunicar e para isto algumas regras são necessárias de acordo com cada tipo : SOAP, RPC, WebSocket e REST.
No nosso caso optamos pela padrão REST (Representational State Transfer), por ser hoje o modelo mais utilizado e simples, utilizando o protocolo HTTP e o formato JSON (JavaScript Object Notation - Notação de Objetos JavaScript) para trafegar as informações, sendo uma formatação leve e com muitas bibiliotecas para utilização nas mais diversas linguagens de progração
O Padrão REST é statelles, ou seja, enviamos uma request e recebemos uma response do servidor, sem a necessidade de gerenciar o estado que o cliente ou servidor possui.
Outro ponto importante do tipo REST é que podemos utilizar a mesma url para os diferentes verbos HTTPs (GEP,POST,DELETE,PUT e etc).
Utilizamos em nosso projeto da ferramenta Spring Initializer que facilita a configuração do nosso projeto e agiliza o desenvolvimento, adicionando todas as depedências que necessitamos e formatando toda a sequência inicial do nosso build.
Acessamos a ferramenta através do site: https://start.spring.io/, porém caso queira poderá instalar na sua máquina e fazer de forma local.
No momento que estamos utilizando a ferramenta, precisamos definir a linguagem: Java ou Kotlin, Versão da Linguagem, Padrão de Projeto e Gerenciador de Dependência, Versão do Spring Boot, Dependências (quase sempre iremos usar a MVC, pois já irá outras como a WEB) e Tipos de Empacotamento: .JAR (Não iremos fazer deploy em Conteiners) ou .WAR (Iremos fazer deploy em Conteiners)
Em nosso projeto estamos utilizando o padrão de deploy .JAR, mesmo o arquivo sendo um pouco maior, não iremos necessitar que o servidor tenha o Conteiner TOMCAT instalado e configurado
Com a abordagem de reutilização de códigos, frameworks, padrões e pacotes com a o qual o Java foi pioneiro, uma ferramenta que nos auxilie a gerenciar todos estes aspectos é de suma importância, sendo que as mais comuns no mercado são: Maven: Linguagens Java ou Kotlin, Gradle:Linguagens Java ou Kotlin, NPM: linguagem java Script, NuGet: aplicações .NET e Pip: Aplicações Phython
No nosso caso optamos pela ferramenta Maven pelas seguintes questões:
* Estamos usando a liguagem JAVA;
* Não é necessário aprender uma tecnologia de programação adicional como no caso do Gradle que precisa ser escrito em Groove;
* OPEN source da APACHE;
* Estimula e auxilia na adoção de boas práticas, utilizando-se do conceito de programação por convenção (Estrutura de diretórios por padrão);
* Gerencia todos os downloads das dependências das dependências que estamos utilizando (antes do Maven era muito trabalhoso importar os .jar e adicionar ao build path);
* Facilita e organiza o build da nossa aplicação;
* Pode incorporar novas dependências através do arquivo POM.XML, podemos pesquisar no site : https://mvnrepository.com/ que é um repositório central;
* Arquivo XML de fácil leitura;
* Usando a ferramenta spring initializer com algumas poucas configurações ficamos com todas as dependências já configuradas e com a definição do arquétipo para o padrão\framework do Spring Boot;
* Familiaridade do time de desenvolvimento.
Citamos acima os principais motivações para a utilização do Maven no nosso projeto, mas existem algumas desvantagens nos seguintes aspectos:
* Necessita de Instalação;
* Arquivo XML de fácil leitura mas pode ser de dificil gravação.
O Maven é uma ferramenta OPEN Source da Apache Software Foundation e pode ser baixado pelo site: http://maven.apache.org/download.html, seguindo os seguintes passos:
* Escolher a ultima versão de acordo com o seu sistema operacional
git remote remove origin* Fazer o download do arquivo .zip
* Escolher um diretório para descompacta-lo
* Adicionar uma variável de ambiente: M2_HOME apontando para o diretório que descompactou o arquivo até chegar no diretório /bin
* Podemos confirmar a instalação, executando o comando no prompt do sistema operacional : mvn - version
Observação: Na primeira utilização do Maven alguns diretórios são criados na máquina local, sendo o principal deles a pasta M2 que é aonde ficam armazenadas fisicamente os arquivos .jar que estamos utilizando em nossos projetos
Quando optamos por usar o Maven em nosso projeto, além de gerenciar as dependências e o build, ele também como foi mencionado aplica conceito de programação por convenção (Estrutura de diretórios por padrão) e facilita a vida do desenvolvoder padronizando também os projetos,seguindo no padrão do arquétipo Spring Boot teremos as principais estruturas listadas abaixo (o que foi abordado até fase atual do curso):
src/main/java - Diretório onde está o(s) código(s) fonte(s) Java da Aplicação e/ou Biblioteca(s)
src/main/resources - Arquivos de configuração e outros arquivos devem ficar nesta pasta, como o application property (utilziada na configuração de acesso aos bancos de dados)
src/test/java - Pasta que contém os arquivos de testes unitários
src/test/resources - Pasta com arquivos que serão utilizados pelas classes de testes unitários
target - As classes javas compiladas do nosso projeto
pom.xml - Arquivo de Configuração do Maven com dados sobre o build e dependências, abaixo iremos detalhar.
README.md - Arquivo que fala de como configurar e os recursos utilizados no projeto
O que é o arquivo POM mencionado no item acima, é um arquivo XML que contém informações sobre o projeto e detalhes de configuração usados pelo Maven para construir o projeto.
Ele contém valores padrão para a maioria dos projetos. Exemplos disso são o diretório de construção, que é target; o diretório de origem, que é src/main/java; o diretório de origem do teste, que é src/test/java; e assim por diante seguindo o conceito de programação por convenção.
Ao executar uma tarefa ou objetivo, o Maven procura o POM no diretório atual. Ele lê o POM, obtém as informações de configuração necessárias e executa.
Algumas das configurações que podem ser especificadas no POM são as dependências do projeto (exemplo: LOMBOK e Validation que usamos em nosso projeto), os plugins ou objetivos que podem ser executados, os perfis de construção e assim por diante.
Outras informações como a versão do projeto, descrição, desenvolvedores, listas de discussão e outras também podem ser especificadas.
As principais tags são:
groupId: nome do pacote e comum em quase todas as demais tags como dependências e build
artifactId: nome do projeto e comum em quase todas as demais tags como dependências e build
version: versão e comum em quase todas as demais tags como dependências e build
parent: Arquétipo Pai de nosso projeto
properties-java.version: com a versão do Java utilizada em nosso projeto
dependencies-dependency: com as dependências que serão caregadas em nosso projeto, aqui vale mencionar que o arquivo settings.xml irá consultar o repositório para buscar estas dependências, por padrão estamos neste projeto usando o default que é o repositório oficial do Maven e quando baixadas os arquivos ficarão na pasta M2 do usuário na nossa máquina local.
build-builds-plugins: Informações para o build de nossa aplicação que segue um conceito de fases, podemos informar quais fases serão executadas ou serão ignoradas. As fases seguem uma ordem predefinida, você informa apenas quem será a última fase do build. Também é possível adicionar novas fases utilizando plugins, que precisam apenas ser configurados
Nas configurações abaixo, que foram geradas pela ferramenta Spring Initializer, informamos: que estamos herdando as funcionalidades de um Arquétipo de um projeto padrão Spring Boot e a sua versão.
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>pring-boot-starter-parent</artifactId>
<version>2.7.12</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
Seguem as dependências que também adicionamos durante a utilização da ferramenta Spring Initializer, lembrando que como explicado anteriormente, caso nossas dependências possuam outras dependências o Maven já faz todo este gerenciamento
Outro aspecto interessante é que na configuração de nosso projeto nas 1º e 2º fases optamos inicialmente em utilizar 4 dependências: Spring MVC (Que ja traz todas as dependências necessárias para um API Rest - Web Stand Alone ), Validation (para utilizarmos o Validation Bean), Lombok (reutilização de códigos comuns) e Swagger(Documentação), porém nesta 3º fase para trabalharmos efetivamente com um banco de dados e a framework mais adequada para isto, acrescentamos a depedência do Spring Data e do MongoDB.
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-validation</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
</dependency>
<dependency>
<groupId>io.springfox</groupId>
<artifactId>springfox-swagger2</artifactId>
<version>3.0.0</version>
</dependency>
<p><dependency></p>
<p><groupId>io.springfox</groupId></p>
<p><artifactId>springfox-boot-starter</artifactId></p>
<p><version>3.0.0</version></p>
<p></dependency></p>
<p><dependency></p>
<p><groupId>io.springfox</groupId></p>
<p><artifactId>springfox-swagger-ui</artifactId></p>
<p><version>3.0.0</version></p>
<p></dependency></p>
</dependencies>
Também temos as configurações de Plugin para o Build:
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
O Maven possui uma vasta lista de comandos para facilitar a nosa vida:
Para compilar: mvn compile
Para compilar off-line:mvn –o compile
Executar todas as fases do Maven até o build: mvn clean install
Para criar os pacotes: mvn package
Instalar o pacote nos respectivos repositórios:mvn install
Limpar a pasta target:mvn clean
Executar os testes unitários:mvn test
Gerar site de documentação para o projeto:mvn site
Gerar Javadoc:mvn javadoc
Startar uma aplicação: mvn spring-boot:run
Em nosso projeto, pensando na separação de responsabilidades, melhores práticas e na arquitetura MVC e agora na Hexagonal (a partir da 2º e 3º fases), criamos os pacotes\pastas (algumas padrões do Maven para o arquétipo escolhido) abaixo em negrito:
Pasta: Config e Controller Excpetion:Criamos esta package como boa prática para termos as classes de infra-estrutura da nossa aplicação bem identificadas, nesta fase do projeto, estamos utilizando duas:
MongoTransactionConfig: Esta classe possui apenas um método com a anotação @Bean que retorna uma factory (padrão de software) para trabalharmos com transações na evolução de nossa aplicação quando necessário.
SpringFoxConfig: Esta classe possui apenas um método com a anotação @Bean retorna uma factory (padrão de software) de Docket para que possamos documentar e executar nossa API com a ferramenta Swagger
CunstomExceptionHandler: Esta classe captura as validações que definimos em nossos DTOs e são validadas nas demais classes de Controller na entrada dos dados com a anotação @Validate.
As principais Classes e Anotações que estamos utilizando nesta package são:
Anotação @Configuration: anotação no nível de classe que indica que uma classe declara um ou mais @Bean-métodos e pode ser processada pelo contêiner Spring para gerar definições de bean e solicitações de serviços para esses beans em tempo de execução
Anotação @Bean: Essa anotação é no nível do Método e indica que esse cria e retorna um “bean” que pode ser usado como dependência em outras classes.
Sobre o Swagger nesta fase do projeto não iremos detalhar ainda.
Pasta Dominio, padrão hexagonal (cada API com uma package com seu nome e dentro de uma delas as demais packages abaixo):
Pasta: Dominio - Controller: Contêm as classes CondutorController,VeiculoController e EstacionamentoController que serão as responsáveis por atender as requisições da camada View e direcionar as ações para as classes de Negócio, nesta camada que é o C - Controller da arquitetura MVC, estamos fazendo a injeção de dependência da nossa classe de Serviço que esta no pacote Service, na classe da Controller temos a exposição dos serviços disponiveis no servidor com base no verbos HTTP
As principais Classes e Anotações que estamos utilizando nesta package são:
Anotação @RestController: anotação no nível de classe que indica que esta é camada de controller do tipo Rest com dados no padrão JSON.
Anotação @RequestMapping: anotação no nível de classe\ atributo que irá indicar a URL principal da API que no nosso casos teremos: Pessoa, Eletrodomestico e Endereço.
Anotação @Autowired: Que fará a injeção de dependência da classe Service, usamos a injeção de dependência para evitarmos o Acoplamento e separarmos bem as responsabilidades, um aspecto importante podemos apenas fazer a injeção nas classes que possuem as anotações: @Component, @Service, @Repository, @Controller e @RestController que serão mapeadas para serem instanciadas na inicialização da aplicação pelo Spring Boot
Anotação @PostMapping: anotação no nível de método que indica qual método irá executar a chamada de uma requisição do verbo POST na URL definida na anotação @RequestMapping da classe, mas podemos também nesta anotação definir uma URL caso tenhamos necessidade.
Anotação @RequestBody: anotação no nível de parâmetros dos métodos que indica que o valor do objeto virá do corpo da requisição, em nosso projeto usamos como parâmetro de entrada para o método que estamos vinculando ao verbo POST da requisição
Classe ResponseEntity: representa toda a resposta HTTP: código de status, cabeçalhos e corpo, em nosso projeto estamos utilizando no retorno do método que estamos vinculado ao método POST mas poderia na ser resposta de qualquer outro tipo de requisição
Nesta 2º Fase com as necessidades de termos um CRUD completo, acrescetamos mais algumas anotações e classes, explicados logo abaixo:
@GetMapping:anotação no nível de método que indica qual método irá executar a chamada de uma requisição do verbo GET, aqui podemos acrecentar um atributo que irá recuperar o(s) parametro(s) ou variaveis para uma chamada que necessite de uma pesquisa mais especifica, recuperadas através das anotações: @PathVariable entendemos a forma mais adequada para a semantica do padrão Rest ou através @RequestParam que viria como QueryString e no casos mais indicada para tratarmos com o solicitante quer paginar uma resposta com muitos registros
@PutMapping:anotação no nível de método que indica qual método irá executar a chamada de uma requisição do verbo PUT, recebendo normalmente a variável que sofrerá alteração, conforme explicado no verbo GET.
@DeleteMapping:anotação no nível de método que indica qual método irá executar a chamada de uma requisição do verbo DELETE, recebendo normalmente a variável que sofrerá aa exclusão, conforme explicado no verbo GET.
@RequestParam:anotação no nível de parametro, podemos ter vários parametros passados por url mas que não são parte da url em sí, conceito da QueryString, podemos também definir se o parametro é obrigatório ou não.
@PathVariable:anotação no nível de parametro, é utilizada quando o valor da variável é passada diretamente na URL, mas não como um parametro que você passa após o sinal de interrogação (?) mas sim quando o valor faz parte da url.
Classe PageRequest: Classe que define os atributos para paginação dentro de um Objeto Page
Classe Page: é uma sublista de uma lista de objetos. Permite obter informações sobre a posição do mesmo na lista inteira que o contém, usamos para termos um interface adequada com o front end
Pasta: Dominio - DTO
Nesta pasta temos as classes que irão representar as Entidades que irão ser utilizadas nas interações entra a camada View e Controller, tendo apenas esta responsabilidade de trafegar os dados (Seguindo o que é conhecido como padrão de software DTO) e aqui podemos falar que aplicamos a 1º especificação do SOLID (Single Responsability Principle (Princípio da Responsabilidade Única); Open/Closed Principle (Princípio do “Aberto para Extensão/Fechado para Implementação); Liskov Substitution Principle (Princípio da Substituição de Liskov)), estas classes são muito similares as classes de entidade do negócio nos atributos, mas aqui podemos tirar alguns atributos que não devem ser expostos por questões de segurança, aplicamos as validações e no nosso caso utilizando as anotações do bean validation
Para realizar o mapeamento dos atributos das classes DTO e das classes de Dominio que estão na nossa pasta MODEL, podemos utilizar diversas Classes, dentre elas a mais recomendada pela questão de performance e com o JMapper mas em nosso projeto pela simplicidade, criamos uma método na classe de DTO que retorna um objeto da nossa classe de Dominio\Negócio (Padrão de software Creator)
As principais Classes e Anotações que estamos utilizando nesta package são:
Anotação @Getter: Iremos falar sobre o Lombok que é a blblioteca que nos fornece esta anotação, em um capítulo mais abaixo, mas a priori esta anotação implementa os métodos Getters (Encapsulamento OO) , podemos usar esta anotação na classe ou nos atributos.
Anotação @ApiModelProperty: nível de atributo ou classe para descrever a especificação, exemplos de utilização na documentação do swagger
Anotação @NotBlank: anotação no nível de atributo do bean validation que não permite espaços em brancos ou não preenchimento do atributo, teremos também uma explicação mais detalhada sobre o Bean validation em um novo tópico abaixo
Anotação @Past: anotação no nível de atributo do bean validation que não permite que uma data seja menor que a data atual, ideal para preenchimento de datas de nascimento
Anotação @JsonFormat: anotação em nível de atributo do jackson que já vem de maneira deafult em projetos Spring Boot que podemos usar parametrizações de padrões como no nosso caso de data
Anotação @Email: anotação no nível de atributo do bean validation que valida se o email é válido
Anotação @NotNull: anotação no nível de atributo do bean validation que valida se o campo esta nulo, cuidado na utilização em conjunto com a anotação @NotBlank, que irá apresentar uma exceção
Pasta: Dominio - Model
Nesta pasta temos as classes que irão representar as Classes e Entidades de negócio, dentro da arquitetura MVC estamos falando que faz parte do M-Model, que dentro dos sistemas corporativos deveria ser a camada mais importante e que não deveria ter implementações de classes relacionadas a infra-estrutura, seguindo os conceitos inciais que estamos apredendo de DDD (Domain Drive Design) devemos padronizar classes e metódos com nomes que fazem referência ao negócio com linguagem ubíqua, em nosso projeto estamos usando nomes que facilitam o entendimento do que representa como Condutor, Veiculo, Estacionamento e etc
As principais Classes e Anotações que estamos utilizando nesta package são:
Anotação @Getter: Iremos falar sobre o Lombok que é a blblioteca que nos fornece esta anotação, em um capítulo mais abaixo, mas a priori esta anotação implementa os métodos Getters (Encapsulamento OO) , podemos usar esta anotação na classe ou nos atributos.
Anotação @AllArgsConstructor: Iremos falar sobre o Lombok que é a blblioteca que nos fornece esta anotação, em um capítulo mais abaixo, mas a priori esta anotação implementa um construtor com todos os atributos da classe, podemos usar esta anotação no nível da classe.
Anotação @EqualsAndHashCode: Iremos falar sobre o Lombok que é a biblioteca que nos fornece esta anotação, em um capítulo mais abaixo, mas a priori esta anotação implementa os métodos equals e hashcode sobrescritos de Object (podemos definir quais atributos devem ou não ser utilizados, através de parâmetros), podemos usar esta anotação no nível da classe
Nesta 3º Fase com as necessidades de termos um CRUD completo e em especial a persistência a camada de dados, acrescentamos mais algumas anotações e classes, explicados logo abaixo de forma sucinta, pois temos um novo tópico mais especifico sobre MongoDB e Spring Data
@Document: anotação a nível de classe, utilizamos para declarar que a classe é uma documento no banco MongoDB ou NOSQL A partir disso o Spring Data estabelecerá a ligação entre a entidade e uma tabela de mesmo nome no banco de dados, onde os dados de objetos desse tipo poderão ser persistidos
@Ref: anotação a nível de atributo, para podermos incorporar outro documento\tabela.
@ID : anotação a nível de atributo, definrrmos as chaves de nosso documento.
@Version : anotação a nível de atributo, porém é um atributo na classe que garante que nosso dado não será atualizado por outro requisição\sistema, pois o baco NoSQL não tem as regras de ACID como os padrão SQL e temos que garantir via aplicação.
@Indexed : anotação a nível de atributo, para indicar que o campo possui um indice mas devemos criar fisicamente no banco de dados também para ter efeito.
Pasta: Dominio - Repository
Nesta pasta temos as classes que irão representar as Classes de persistências e relacionamento com o banco de dados, seguindo os conceitos de DDD devemos padronizar classes e metódos com nomes que fazem referência ao negócio com linguagem ubíqua, em nosso projeto estamos usando nomes que facilitam o entendimento do que representa como CondutorRepository, Métodos : Salvar, Buscar e etc. Na começamos a persistir os dados em um banco de dados confiável do tipo NOSQL e um pouco mais abaixo, estamos expondo a utilização de uma nova classe.
As principais Classes e Anotações que estamos utilizando nesta package são:
Anotação @Repository: Indica que uma classe\interface anotada é um Repositório, originalmente definido por DDD como um mecanismo para encapsular armazenamento, recuperação e comportamento de pesquisa que representa uma coleção de objetos, com base nesta anotação e seguindo o objetivo de evitarmos acomplamento pela injeção de dependências e inversão de controle, injetamos estas classes nas classes da pasta Service.
MongoRespository: módulo que contém uma camada de Repositórios que dá suporte à criação de repositórios estendendo interfaces do Spring Data. A infraestrutura é compartilhada pelos bancos , aonde ja temos implementados diversos metodos como por exemplo de Crud que recebem um Generic da Entidade que queremos persistir.
Pasta: Dominio - Service: Pasta com as classes de serviço que contêm as regras de negócios comuns e são responsáveis em delegar para as classes de Negócios as requisições da Controller, em nosso projeto ela é injetada na Controller e recebe a injeção das classes de negócio.
As principais Classes e Anotações que estamos utilizando nesta package são:
Anotação @Service: Indica que uma classe anotada é um Serviço
Anotação @Autowired: Que fará a injeção de dependência da classe Repositório e da Config - Bean validation, usamos a injenção de dependência para evitarmos o Acoplamento.
Em tópicos anteriores, falamos sobre algumas das anotações que utilizamos desta biblioteca, sendo que esta surgiu da necessidade de reutilizarmos códigos comuns as mais diversas aplicações, chamados de boilerplate (alusão a contratos juridicos com clausulas comuns) como a criação de construtores, metodos getters e setters (polimorfismos OO).
Porém devemos tomar cuidados em usar sempre todas as anotações de forma indiscriminada, vendo a real necessidade sempre.
Para utilizarmos esta biblioteca, devemos adicionar a sua dependência no arquivo POM do gerenciador Maven.
Principais anotações:
@NoArgsConstructor: nível de classe para criar construtores sem nenhum parâmetro
@AllArgsConstructor:nível de classe para criar construtores com todos os parâmetros da classe
@RequiredArgsConstructor:nível de classe para criar construtores com alguns parâmetros da classe
@NonNull:nível de atributo, em conjunto com @RequiredArgsConstructor, para determinar qual parâmetro deve ser utilizado no construtor, o Lombok pede para que você o declare como tipo final sem inicialização ou com a annotation @NonNull sem inicialização.
@Getter:nível de classe (todos os atributos) e atributo para implementação de métodos getters, neste caso temos o nível de permissão como parâmetro: PUBLIC,PROTECTED,PACKAGE ou PRIVATE
@Setter:nível de classe (todos os atributos) e atributo para implementação de métodos setters, neste caso temos o nível de permissão como parâmetro: PUBLIC,PROTECTED,PACKAGE ou PRIVATE
@EqualsAndHashCode: nivel de classe, com essa annotation, o Lombok cria os métodos equals() e hashCode() que podem ser utilizados em comparação. Podemos [email protected] e @EqualsAndHashCode.Exclude para adicionar ou excluir algum atributo da classe conforme nossa necessidade
@Data:nível de classe é para implementarmos as seguintes anotações de forma implicita : @Getter, @Setter, @ToString, @EqualsAndHashCode e @RequiredArgsConstructor na classe (exceto que nenhum construtor será gerado se já existir algum construtor explicitamente escrito).
@ToString:nível de class(todos os atributos) e atributo para implementação do método ToString
@Value:nível de classe é a variante imutável de @Data; todos os campos são privados e finais por padrão, e os setters não são gerados. A própria classe também é finalizada por padrão, porque a imutabilidade não é algo que pode ser forçado em uma subclasse. Como @Data, métodos úteis toString(), equals() e hashCode() também são gerados, cada campo recebe um método getter e um construtor que cobre todos os argumentos (exceto os campos finais que são inicializados na declaração do campo) também é gerado .
O Bean Validation é uma biblioteca para apoiar na validação dos valores preenchidos em campos de entrada de dados e é extremamente importante em aplicações. O procedimento evita que armazenemos sujeira em nossas bases de dados e, dependendo do caso, pode até mesmo ter impacto na segurança do sistema.
Um dos mecanismos que podemos utilizar para realizar esta verificação surgiu com a liberação da plataforma Java EE 6, na qual foi introduzida a especificação Bean Validation 1.0.
O objetivo principal da biblioteca foi auxiliar os programadores nesta tarefa, que muitas vezes toma bastante tempo durante o desenvolvimento.
As boas práticas indicam que devemos inserir as anotações de validações desta biblioteca na camada de DTO evitando assim que tenhamos em diversas partes do código estas regras, auxiliando até na manutenção do projeto
Em nosso projeto usamos esta biblioteca e suas anotações nos atribuitos das classes da camada de DTO, passando também o parâmetro da mensagem de erro a ser retornada na response
Principais anotações que utulizamos:
@NotNull: valida se o valor é diferente de nulo.
@AssertTrue: valida se o valor é verdadeiro.
@Size: valida se a variável possui o tamanho entre valores mínimo (parâmetro min) e máximo (parâmetro max). Você pode utilizar essa annotation em campos de tipo String, Collection, Map e array.
@Min: valida se o valor da variável é maior que o passado no parâmetro value.
@Email: valida se o valor da variável é um e-mail válido.
@NotEmpty valida se o valor do campo não está vazio; pode ser utilizado em variáveis de tipo o String, Collection, Map ou Array.
@NotBlank: valida se o valor do campo não é nulo ou com espaços em branco; pode ser utilizado em variáveis de tipo de texto.
@Positive e @PositiveOrZero: valida se o valor do campo é positivo ou positivo incluindo o 0; pode ser utilizado em variáveis numéricas.
@Negative e @NegativeOrZero: valida se o valor do campo é negativo ou negativo incluindo o 0; pode ser utilizado em variáveis numéricas.
@Past e @PastOrPresent: valida se o valor do campo de tipo de data está somente no passado ou no passado e no presente.
@Future e @FutureOrPresent: valida se o valor do campo de tipo de data está somente no futuro ou no futuro ou no presente.
O MongoDB é um sistema de banco de dados NoSQL (Não Relacional), utiliza um modelo de documentos, onde os dados são armazenados em documentos no formato JSON(JavaScript Object Notation) ou BSON (Binary JSON).
A vantagem do MongoDB é o fato de fornecer alta disponibilidade e tolerância a falhas. Ele suporta replicação automática, garantindo que, se um servidor falhar, osoutros continuarão funcionando normalmente. Além disso, o MongoDB possui recursos de particionamento, permitindo que os dados sejam distribuídos em vários nós de um cluster para melhor desempenho e escalabilidadeComo o parquimetro pode ter um aumento exponecial nos periodos de férias, é importante termos esta capaciadade para irmos aumentando a escalabilidade
Optamos também em usar um modelo de desnromalizado que facilita e agiliza nossas pesquisas e é o mais indicado para bancos NoSQL, sequindo os modelos de Incorporação e duplicação seletiva de dados, mas cotrolando bem as transações já que dferente do SQL o NOSQL não possui ACID (Atomicidade, Consistência, Isolamento e Durabilidade), tendo que contralar em nosso caso na aplicação, para issso podemos usar recuros de lógica d eprogração e mesmo algumas anotações como @Version.
O Spring Data pertence a Spring Framework com o objetivo de facilitar o acesso e a manipulação de dados em aplicações Java. Ele fornece uma camada de abstração sobre diferentes tecnologias de persistência de dados, como bancos de dados relacionais, bancos de dados NoSQL, serviços web RESTful e até mesmo sistemas de arquivos.
Usamos em nosso projeto o Spring Data MongoDB para interação com o banco de dados NoSQL MongoDB.
O uso do Spring Data com o MongoDB é vantajoso pois simplifica a integração entre o MongoDB e aplicações Java, reduzindo código boilerplate e abstraindo detalhes da conexão.
Com o desenvolvimento do projeto e a necessidade de distribuição do projeto para que demais pessoas possam executa-lo, vimos a necessidade de criar um Docker para facilitar este ponto sem que quem desejar utiliza-lo tenha todas as instalações necessárias.
O que é Docker: é um sistema para desenvolvimento, distribuição ou execução de aplicações. Ele possui um conjunto de funcionalidades que permite que você separe sua aplicação da dependência de infraestrutura, e para isso utiliza-se do conceito de virtualização.
Grupo 39 : RM350113 , RM349894 , RM350459 e 350127