Skip to content

ArleiPacanaro/TechChallenge

Repository files navigation

Tech Challenge - Painel de Consumo de Energia

Portal de Consumo de Energia

Descrição do projeto

Projeto Tech Challenge - FIAP Alura - 2º Fase

APIs que serão disponibilizadas para um portal que irá demonstrar ao usuário o consumo de energia de seus eletrodomésticos.

* Endereços

* Pessoas

* Eletrodomésticos

Neste momento as APIs irão fornecer todos os métodos de um CRUD, Relacionamentos entre si, Filtros diversos de pesquisas sobre os dados de Endereços, Pessoas e Eletrodomésticos.

Após construimos na 1º Fase do projeto 3 APIs apenas com ações do método\verbo POST, sendo que 2 estavam persistindo em banco de memória H2 e uma em collection do tipo SET, nesta 2º Fase estamos agora persistindo em um banco de dados relacional confiável, criamos em todas as APIs os demais métodos que compõe um CRUD (GET,POST,UPDATE,DELETE), relacionamentos entre as entidades utlizando as framework de implemnetações da especificação JPA e o ambiente de execução com Docker, para estes temas adicionais acrescentamos em nosso README os capitulos\tópicos abaixo com os detalhes do que utilizamos, motivações e dificuldades :

* Banco de Dados

* JPA\Hibernate

* Docker

Funcionalidades e Demonstração da Aplicação - 2º Item do Entregável

1) Baixar via GIT : git clone https://github.com/ArleiPacanaro/TechChallenge.git

2) 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:

2.1)ir até a pasta: TechChallenge e executar o comando: docker build -t tech:2.0 . ** o detalhe do ponto muito importante, bem como manter o nome tech:2.0 para uso em nosso compose .yml

Poderiamos através de uma conta docker hub ja deixar a imagem pronta, porém por não ser requisito e ainda não termos esta conta cadastrada, iremos criar uma imagem localmente.

2.2) ir até a pasta: TechChallenge e executar o comando: docker compose up -d

Este commando irá executar o conteiner da imagem que criamos no item 1 da nossa aplicação , mas antes irá executar a criação do banco da nossa aplicação que chame-se dbEnergia em Postgres, com usuário = user e senha = 123456, isto é importante caso for analisar o resultado dos processamentos no banco de dados.

** As vezes se torna necessário executar 2 vezes o compose, colocamos a dependência para subir a aplicação após o banco, mas as vezes acabam indo em conjunto e a aplicação fica pendente.

Premissa: Docker instalado na máquina client.

Após os passos acima, poderá acessar o nosso swagger para testes e ver nossa documentação.

URL principal do nosso swagger: http://localhost:8080/swagger-ui/#/

API Usuários

O nosso projeto o ideal é que siga a sequência abaixo das URLs para realizar o ciclo completo do negócio, pois:

1) Devemos cadastrar os usuários

2) As pessoas referenciando o ID do usuário que possui aquela pessoa

3) Os endereços referenciando o ID do usuário que possui aqueles Endereços

4) Os Eletrodomésticos referenciando o ID do Endereços que possui aqueles Eletros

5) O cálculo do consumo do Eletrodoméstico, informando o tempo em minutos de uso do equipamento

Funcionalidades: CRUD de Usuários com os GETs relacionando as pessoas e endereços que o usuário possue.

Header da Requisição: ApplicattionType/JSON demais atributos vazios

Body da Requisição: Formato JSON, exemplo no swagger da aplicação

Swagger da aplicação local: http://localhost:8080/swagger-ui/#/usuario-controller

API Pessoas

Funcionalidades: CRUD de Pessoas relacionadas ao usuário e filtros: de Nome ou Parentesco ou Sexo

Header da Requisição: ApplicattionType/JSON demais atributos vazios

Body da Requisição: Formato JSON, exemplo no swagger da aplicação

Swagger da aplicação local: http://localhost:8080/swagger-ui/#/pessoa-controller

API Endereços

Funcionalidades: CRUD de Endereços relacionados aos usuários, filtros de : Bairro ou Municipio ou Rua, além de no GET recuperar qual é o usuário dono do endereço e a lista de eletrodomésticos.

Header da Requisição: ApplicattionType/JSON demais atributos vazios

Body da Requisição: Formato JSON, exemplo no swagger da aplicação

Swagger da aplicação: http://localhost:8080/swagger-ui/#/endereco-controller/

API Eletrodomesticos

Funcionalidade : CRUD de Eletrodomesticos relacionados aos endereços, cálculo do consumo do eletrodoméstico e o relacionamento a qual Endereço o Eletrodoméstico pertence.

Header da Requisição: ApplicattionType/JSON demais atributos vazios

Body da Requisição: Formato JSON, exemplo no swagger da aplicação

Swagger da aplicação: http://localhost:8080/swagger-ui/#/eletrodomestico-controller/

]

Tecnologias Utilizadas

Liguagem: Java versão 11

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 versão: 2.7.12 , LOMBOK versão default para a versão do Spring, Bean Validation versão default para a versão do Spring, JPA

Ferramentas Colaborativas: Docker,Postman,GIT,Maven e Swagger

Banco de Dados: POSTGRES

Relatório Técnico - 3º Item do Entregável

O objetivo deste tópico é explicar as motivações e desafios das técnicas e tecnologias aplicadas nesta fase do Tech Challenge.

Spring Boot E API REST

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).

Spring Initializer

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

Gestão de Dependências

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

Maven

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.

Instalação Maven

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

* 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

Estutura de pastas no Projeto Maven

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

Configuração do Projeto Maven - POM.xml

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

POM de nosso projeto - Principais configurações

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 na 1º fase 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 2º fase para trabalharmos efetivamente com um banco de dados e a framework mais adequada para isto, acrescentamos a depedência do JPA.

<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>

             <dependency>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-data-jpa</artifactId>
	</dependency>

</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>

Principais Comandos Maven

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

Separação de Camadas, Padrões de Projeto, Principais: Anotações e Classes

Em nosso projeto, pensando na separação de responsabilidades, melhores práticas e na arquitetura MVC e agora na Hexagonal (a partir da 2 fase), criamos os pacotes\pastas (algumas padrões do Maven para o arquétipo escolhido) abaixo em negrito:

Pasta: Config: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:

ValidatorBean: Esta classe possui apenas um método com a anotação @Bean que retorna uma factory (padrão de software) de Bean Validations para que possamos, validar os dados das requisições.

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

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.

Classe Validation: Classe que possui diversos métodos estáticos, sendo que usamos um deles no projeto para o retorno de Validators que iremos utilizar para a valdiação de nossos objetos do tipo DTO, centralizando assim na entrada dos dados as regras de validações antes do processamento e persistência.

Sobre o Swagger nesta fase do projeto não iremos detalhar ainda.

Pasta: Exception: Criamos esta package como boa prática para termos as classes de tratamentos de exceções da nossa aplicação mas por não ser o foco desta fase nas aulas, não iremos entrar em maiores detalhes deste tema.

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 PessoaController,EnderecoController e EletrodomesticoController 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, na fase 1 deste projeto estamos apenas expondo serviço com chamada via metódo POST, um pouco mais abaixo expomos novas necessidades da utilização de outras anotações e classes com a evolução das fases.

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 - Entitie

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 Pessoa, PessoaSexo, Endereco 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 2º 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 JPA\Hibernate:

@Entity: anotação a nível de classe, utilizamos para declarar que a classe é uma entidade. A partir disso o JPA 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

@Table: anotação a nível de classe, podemos especificar detalhes em seus 4 tipos de atributos (nome, sobrescrever seu catálogo, seu esquema e assegurar restrições de unicidade nas colunas) da tabela que serão utilizados para persistir as nossas entidades na base de dados. Caso essa anotação seja omitida, não teremos um erro como resultado, porém será utilizado o nome da classe como valor default. Dessa forma, apenas definimos a anotação se quisermos sobrescrever algo do foi mencionado na parametrização de seus atributos.

@Column: anotação a nível de atributo, podemosespecificar os detalhes da coluna que um campo ou propriedade, alguns detalhes são relacionados com o esquema e, portanto aplicados apenas quando um esquema for gerado. A anotação @Column é opcional, possuindo valores default já configurados.

@GeneratedValue: anotação a nível de atributo utilizada para indicar que a geração do valor do identificador único da entidade será gerenciada pelo provedor de persistência. Essa anotação deve ser adicionada logo após a anotação @Id. Quando não anotamos o campo com essa opção, significa que a responsabilidade de gerar e gerenciar as chaves primárias será da nossa aplicação

@Id: anotação a nível de atributo utilizada para indicar que será a chave primária de nossa entidade

@OneToOne: anotação a nível de atributo, que define uma associação com outra entidade que tenha a multiplicidade de um-para-um

@OneToMany: anotação a nível de atributo, que define uma associação com outra entidade que tenha a multiplicidade um-para-muitos

@ManyToOne:anotação a nível de atributo, que define uma associação com outra entidade que tenha a multiplicidade de muitos-para-um

@ManyToMany:anotação a nível de atributo, que define uma associação com outra entidade que tenha a multiplicidade de muitos-para-muitos

@JoinColumn: anotação a nível de atributo: é usado para especificar o mapeamento de uma coluna de chave estrangeira em um relacionamento entre duas entidades. A anotação @JoinColumn é aplicada no lado proprietário da associação para definir o nome da coluna de chave estrangeira e outros atributos relacionados à coluna de junção.

@JoinTable: anotação a nível de atributo, usada para mapear uma relação ManyToMany entre duas entidades

@Transactional: anotação a nível de classe, criar uma transação na persistencia de nosso dados para evitar erros de constraints

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 PessoaRepository, Métodos : Salvar, Buscar e etc. Na fase 1 do projeto ainda não usamos JPA e mesmo banco de dados não estava obrigatório, mas em algumas APIS deste projeto já estavmos persistindo em banco H2 e em outras seguimos a persistência em Collections do tipo Set. Porém após a fase 1 começamos a persistir os dados em um banco de dados confiável 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.

JPARespository: módulo que contém uma camada de Repositórios (JpaRepository) que dá suporte à criação de repositórios estendendo interfaces do Spring Data. A infraestrutura é compartilhada pelos bancos e o JPA Provider que implementa o JPA, aonde ja temos implementados diversos metodos como por exemplo de Crud que recebem um Generic da Entidade que queremos persistir.

Um aspecto importante é que para nossos Filtros usamos ja metodos implementados pela framework do Hibernate seguindo a especificação JPA, existem dezenas de metodos que fazerm as combinações dos atributos das nossas entidades, por exeplo em Pessoa para a listagem criamos a assinatura findByNomeOrParentescoOrSexo que facilitou muito nosso desenvolvimento dos filtros.

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.

LOMBOK

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 .

Bean Validation

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.

Banco de Dados

O nosso projeto a partir da 2º Fase apresentou a necessidade de persistirmos os dados em um banco de dados confiável, com isto fomos pesquisar através dos requisitos:

* Armazenamento de Objeto

* Possibilidade de implementação de comportamento no banco de dados

* Controle de concorrência

* Controle de Transação

* Aplicar integridade referencial

* Open Source

Qual seria um SGDB adequado para nosso projeto e optamos pelo POSTGRES que possui todas as caracteristicas mencionadas e pelo site do DockerHub também vimos a versão com menos vulnerabildiades e estável e decidimos pela 9.4

Com base nos requisitos apresentados e na modelagem do comportamento dos registros em anexo [Tabelas_atualizadas.xlsx](https://github.com/ArleiPacanaro/TechChallenge/files/12508633/Tabelas_atualizadas.xlsx) , começamos a estruturar nossas tabelas com os comandos DDL (Data Definition Language) abaixo:

Adicionamos comentários no bloco para explicarmos as motivações que nos indicaram a utilização de alguns comandos

--- Comentário: Tabela que irá representar os usuários do sistema e que conterá as pessoas --- São comanddos DDL e em nosso caso, como a criação das tabelas ocorrerá de forma automática através de um docker-compose (tema - --- que temos um tópico abaixo com mais detalhes), realizamos uma verificação se a tabela existe e apenas criamos senão existirem - --- no banco de dados , o tipo SERIAL indica que será autoincrementado valor do campo do tipo INT, a palavra PRIMARY KEY indica a --- coluna que será a chave primária e pode haver composição, NOT NULL siginifica que o campo não aceitará valores NULOS, ocasionando --- uma exceção

CREATE TABLE IF NOT exists tb_usuario( id SERIAL PRIMARY KEY, username VARCHAR(255) NOT NULL, senha VARCHAR(255) NOT NULL);

--- Comentário: Tabela que irá representar as pessoas do sistema e que possui uma relação de N para 1 com a tabela de usuários, por isso da informação das palavras FOREIGN KEY e --- REFERENCES quem indicam qual a tabela que ela irá relacionar e uma dependência --- mesmo, já que poderemos ter apenas pessoas com usuários que já existam no banco de dados. --- Também temos a informação que caso o registro na tabela Pai seja exlcuido o mesmo deverá ocorrer na tabela filha com o comando CASCADE ON DELETE, sem a necessidade da aplicação --- gestionar, isto é um ponto muito importate de sempre avaliarmos em qualquer sistema que formos trabalhar.

CREATE table IF NOT exists tb_pessoa( id SERIAL PRIMARY KEY, cpf VARCHAR(14) NOT NULL, nome VARCHAR(255) NOT NULL, telefone VARCHAR(30) , email VARCHAR(30) NOT NULL, senha VARCHAR(255) NOT NULL, dataNascimento DATE NOT NULL, sexo VARCHAR(9) NOT NULL, parentesco VARCHAR(30) NOT null, idUsuario BIGINT NOT null, FOREIGN KEY (idUsuario) REFERENCES tb_usuario (id) ON DELETE CASCADE );

-- Comentário: Tabela que irá representar endereços do sistema e que possui uma relação de N para N com a tabela de usuários CREATE TABLE IF NOT exists tb_endereco ( id SERIAL PRIMARY KEY, pais VARCHAR(255) NOT NULL, estado CHAR(2) NOT NULL, municipio VARCHAR(255), bairro VARCHAR(255) NOT NULL, cep VARCHAR(30) NOT NULL, complemento VARCHAR(30), rua VARCHAR(255) NOT NULL, idUsuario BIGINT NOT null, FOREIGN KEY (idUsuario) REFERENCES tb_usuario (id) ON DELETE CASCADE );

-- Comentário: Tabela que irá representar os eletrodomésticos de uma casa sendo N para 1 com o endereço (@ManyToOne) e que será ------- excluida de forma automática, caso o endereço seja excluído CREATE TABLE IF NOT exists tb_eletrodomestico ( id SERIAL PRIMARY KEY, nome VARCHAR(255) NOT NULL, modelo VARCHAR(255) NOT NULL, potencia INTEGER , serialNumber VARCHAR(30) NOT NULL, idEndereco SERIAL NOT null unique , FOREIGN KEY (idEndereco) REFERENCES tb_endereco (id) ON DELETE CASCADE );

JPA\Hibernate

A Java Persistence API (JPA) é uma especificação que possue 3 implentações de framework que foram desenvolvidas com o conceito de POJO (Plain Old Java Object ou Velho e Simples Objeto Java) para persistir os objetos Java. Em nosso caso usamos a implementação da Hibernate por ser a mais difundida e default no mercado.

Existem diversas anotações e propriedades no JPA, em nosso projeto com as necessidades que foram aparecendo, tivemos que usar as anotações\classes abaixo:

* @Entity: anotação a nível de classe, utilizamos para declarar que uma classe é uma entidade. A partir disso o JPA 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

* @Table: anotação a nível de classe, podemos especificar detalhes em seus 4 tipos de atributos (nome, sobrescrever seu catálogo, seu esquema e assegurar restrições de unicidade nas colunas) da tabela que serão utilizados para persistir as nossas entidades na base de dados. Caso essa anotação seja omitida, não teremos um erro como resultado, porém será utilizado o nome da classe como valor default. Dessa forma, apenas definimos a anotação se quisermos sobrescrever algo do foi mencionado na parametrização de seus atributos.

* @Column: anotação a nível de atributo, podemos especificar os detalhes da coluna que um campo ou propriedade, alguns detalhes são relacionados com o esquema e, portanto aplicados apenas quando um esquema for gerado. A anotação @Column é opcional, possuindo valores default já configurados.

* @GeneratedValue: anotação a nível de atributo utilizada para indicar que a geração do valor do identificador único da entidade será gerenciada pelo provedor de persistência. Essa anotação deve ser adicionada logo após a anotação @Id. Quando não anotamos o campo com essa opção, significa que a responsabilidade de gerar e gerenciar as chaves primárias será da nossa aplicação, um dos seus atributos indica a estratégia de como será gerada o idendificador e a mais comum é de auto-incremento.

* @Id: anotação a nível de atributo utilizada para indicar que será a chave primária de nossa entidade

* @OneToOne: anotação a nível de atributo, que define uma associação\relacionamento com outra entidade que tenha a multiplicidade de um-para-um, normalmente em um relacionamento unidirecional apenas usamos a anotação na classe PAI, por isso é sempre importante esta avaliação do direcionamento da relação.

* @OneToMany: anotação a nível de atributo, que define uma associação\relacionamento com outra entidade que tenha a multiplicidade um-para-muitos

* @ManyToOne:anotação a nível de atributo, que define uma associação\relacionamento com outra entidade que tenha a multiplicidade de muitos-para-um

* @ManyToMany:anotação a nível de atributo, que define uma associação\relacionamento com outra entidade que tenha a multiplicidade de muitos-para-muitos

* @JoinColumn: anotação a nível de atributo: é usado para especificar o mapeamento de uma coluna de chave estrangeira em um relacionamento entre duas entidades. A anotação @JoinColumn é aplicada no lado proprietário da associação para definir o nome da coluna de chave estrangeira e outros atributos relacionados à coluna de junção.

* @JoinTable: anotação a nível de atributo, usada para mapear uma relação ManyToMany entre duas entidades

* @Transactional: anotação a nível de classe, criar uma transação na persistencia de nosso dados para evitar erros de constraints

Docker

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.

Em nosso caso aplicamos os passos e configurações abaixo:

* Copiamos o arquivo Dockerfile (Padrão de nome que deve ser respeitado) na pasta raiz do nosso projeto.
* Geramos o . JAR da nossa aplicação.
* Fomos na pasta TARGET a partir da raiz do nosso projeto e executamos mvn packge -DskipTests = true pelo terminal no proprio Intelij
* No terminal executamos talbém o comando docker build -t tech:2.0  e o ponto . ** irá criar a imagem no docker local com as definições do 
Dockerfile
* Criamos 2 diretórios na pasta raiz do projeto ** mas poderia ser outra ** sendo: scripts e data, o criamos arquivo docker- 
compose.yml (Padrão de nome que deve ser respeitado).
* Copiamos o arquivo .sql  na pasta script (temos ele com detalhes no tópico de banco de dados).
* Fomos pelo terminal, até o diretório que esta o arquivo docker-compose.yml e executamos o comando: docker compose up -d
 Com isto o banco irá subir, criando as tabelas caso não existam e startar a nossa aplicação.

Abaixo nosso arquivo Dockerfile:

    FROM openjdk:11
    ADD target/*.jar app.jar
    WORKDIR /appp
    EXPOSE 8080
    COPY target/*.jar /appp/app.jar
    ENTRYPOINT ["java", "-jar", "/app.jar"]

Estamos fazendo o download da versão 11 do Java, adiconando o arquivo de extensão .JAR de nossa aplicação , definindo a nossa área de trabalho, a porta que nossa aplicação irá utilizar e o comando quando for executada a imagem e que irá startar a nossa aplicação.

Abaixo esta nosso arquivo docker-compose que irá executar nosso projeto, mas antes criando uma imagem do banco de dados:

version: '3'

services: db: container_name: Techchallenge image: postgres:9.4 restart: always volumes: - ./data:/var/lib/postgresql/data - ./scripts/createbanco.sql:/docker-entrypoint-initdb.d/init.sql environment: POSTGRES_USER: user POSTGRES_PASSWORD: 123456 POSTGRES_DB: dbEnergia ports: - 5432:5432 web: image: tech:2.0 depends_on: - db ports: - 8080:8080 environment: - POSTGRES_URL=Techchallenge - POSTGRES_USERNAME=user - POSTGRES_PASSWORD=123456

Criamos um serviço que executará a imagem de um banco POSTGRES, criando também através do nossos script no tópico de banco as tabelas caso não existam e na sequência executando nossa imagem da aplicação que esta dependente do Banco de dados.

Pessoas Desenvolvedoras do Projeto

Grupo 39 : RM350113 , RM349894 , RM350459 e 350127

Conclusão

About

TechChallenge

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 4

  •  
  •  
  •  
  •