Skip to content

14. [TD]: Exposição web da camada [de negócios]

Palavras-chave: arquitetura multicamadas, Spring, injeção de dependências, serviço web / JSON, cliente / servidor.

Voltemos à arquitetura atual da aplicação TD:

Vamos evoluir esta arquitetura para o seguinte:

para expor a interface [IMetier] da camada de negócios na Web. Para tal, seguiremos a metodologia descrita na secção 13.5.

14.1. Suporte

  

Os projetos para este capítulo podem ser encontrados na pasta [support / chap-14].

14.2. O projeto Eclipse para a camada [business]

  

14.2.1. Configuração do Maven

O projeto da camada [business] é um projeto Maven configurado pelo seguinte ficheiro [pom.xml]:


<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
    <modelVersion>4.0.0</modelVersion>
    <groupId>istia.st.elections</groupId>
    <artifactId>elections-metier-dao-spring-data</artifactId>
    <version>0.1.0</version>
 
    <!-- dependencies -->
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.2.7.RELEASE</version>
    </parent>
    <dependencies>
        <!-- layer [DAO] -->
        <dependency>
            <groupId>istia.st.elections</groupId>
            <artifactId>elections-dao-spring-data-01</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>
        <!-- Spring Boot -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot</artifactId>
            <scope>test</scope>
        </dependency>
        <!-- Spring Boot Test -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>
 
    <properties>
        <!-- use UTF-8 for everything -->
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <java.version>1.8</java.version>
    </properties>
 
    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
                <version>2.18.1</version>
            </plugin>
        </plugins>
    </build>
</project>
  • linhas 18–22: a dependência da camada [DAO] criada no parágrafo 12;
  • linhas 23–34: as dependências necessárias para os testes;

14.2.2. Configuração do Spring

  

O projeto da camada [business] é um projeto Spring configurado pelo seguinte ficheiro [MetierConfig]:


package elections.metier.config;
 
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Import;
 
import elections.dao.config.DaoConfig;
 
@Import({ DaoConfig.class })
@ComponentScan({ "elections.metier.service" })
public class MetierConfig {
}
  • Não utilizamos aqui a anotação [@Configuration], o que tornaria a classe numa classe de configuração Spring. A presença das anotações [@Import] e [@ComponentScan] torna-a automaticamente numa classe de configuração;
  • Linha 8: Importamos o ficheiro de configuração da camada [DAO]. Passamos então a ter acesso a todos os beans definidos por este ficheiro;
  • Linha 9: Outros beans Spring encontram-se na pasta [elections.metier.service];

14.2.3. Implementação da camada [business]

  

A implementação da camada [de negócios] é a definida na Secção 8.5.

14.2.4. Testar a camada [de negócios]

  

A classe de teste é a descrita na Secção 8.6.


Tarefa: Implementar o projeto da camada [de negócios] e passar no seu teste unitário. Gerar o arquivo da camada no repositório Maven local (Executar como / Maven / Instalar).


14.3. O projeto Eclipse para a camada [web]

A camada web é uma camada Spring MVC:

O projeto Eclipse tem a seguinte estrutura:

  • [Boot.java] é a classe que inicia o serviço web;
  • [WebConfig.java] é a classe de configuração do serviço web;
  • [Response.java] é a resposta gerada pelos vários URLs do serviço web;
  • [ElectionsController] é a classe de implementação do serviço web;

14.4. Configuração do Maven

O projeto é um projeto Maven configurado pelo seguinte ficheiro [pom.xml]:


<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <groupId>istia.st.elections</groupId>
    <artifactId>elections-webjson-metier-dao-spring-data</artifactId>
    <version>0.0.1-SNAPSHOT</version>
 
    <name>elections-webjson-metier-dao-spring-data</name>
    <description>couche métier exposée comme un service web / jSON</description>
 
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.2.7.RELEASE</version>
    </parent>
 
    <dependencies>
        <!-- business layer -->
        <dependency>
            <groupId>istia.st.elections</groupId>
            <artifactId>elections-metier-dao-spring-data</artifactId>
            <version>0.1.0</version>
        </dependency>
        <!-- layer MVC -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
    </dependencies>
 
    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <java.version>1.8</java.version>
    </properties>
 
    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
                <version>2.18.1</version>
            </plugin>
        </plugins>
    </build>
 
</project>
  • linhas 19–23: a dependência do arquivo da camada [business]. Esta é a que criámos no parágrafo 14;
  • linhas 25–28: a dependência para uma aplicação Spring MVC;

14.5. Configuração do Spring

 

A classe [WebConfig] configura o serviço web:


package elections.webjson.config;
 
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.config.ConfigurableBeanFactory;
import org.springframework.boot.context.embedded.EmbeddedServletContainerFactory;
import org.springframework.boot.context.embedded.ServletRegistrationBean;
import org.springframework.boot.context.embedded.tomcat.TomcatEmbeddedServletContainerFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Import;
import org.springframework.context.annotation.Scope;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.servlet.DispatcherServlet;
import org.springframework.web.servlet.config.annotation.EnableWebMvc;

import com.fasterxml.jackson.databind.ObjectMapper;
 
import elections.metier.config.MetierConfig;
 
@EnableWebMvc
@Import({ MetierConfig.class })
@ComponentScan({ "elections.webjson.service" })
public class WebConfig {
    // -------------------------------- layer configuration [web]
    @Autowired
    private ApplicationContext context;
 
    @Bean
    public DispatcherServlet dispatcherServlet() {
        DispatcherServlet servlet = new DispatcherServlet((WebApplicationContext) context);
        return servlet;
    }
 
    @Bean
    public ServletRegistrationBean servletRegistrationBean(DispatcherServlet dispatcherServlet) {
        return new ServletRegistrationBean(dispatcherServlet, "/*");
    }
 
    @Bean
    public EmbeddedServletContainerFactory embeddedServletContainerFactory() {
        return new TomcatEmbeddedServletContainerFactory("", 8080);
    }
    // mapper jSON
    @Bean
    @Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    public ObjectMapper jsonMapper() {
        return new ObjectMapper();
    }
 
}
  • O significado desta configuração foi explicado na Secção 13.5.3.1. Iremos explicar apenas as novas funcionalidades:
  • linha 22: importamos o ficheiro de configuração da camada [business] para utilizar todos os seus beans;
  • linha 23: especificamos que outros beans serão encontrados na pasta [elections.webjson.server.service];

14.6. A classe de inicialização do serviço web

 

A classe [Boot] inicia o serviço web da seguinte forma:


package elections.webjson.boot;
 
import org.springframework.boot.SpringApplication;
 
import elections.webjson.config.WebConfig;
 
public class Boot {
 
    public static void main(String[] args) {
        SpringApplication.run(WebConfig.class, args);
    }
}
  • Linha 10: O método estático [SpringApplication.run] utilizará o ficheiro de configuração [WebConfig]. Devido à anotação [@EnableAutoConfiguration], o Spring Boot irá iniciar o servidor Tomcat e implementar o serviço web no mesmo;

14.7. A resposta das URLs do serviço web

 

Todas as URLs do serviço web / JSON enviam o mesmo tipo de resposta:


package elections.webjson.service;
 
import java.util.List;
 
public class Response<T> {
 
    // ----------------- properties
    // operation status
    private int status;
    // any error messages
    private List<String> messages;
    // the body of the reply
    private T body;
 
    // manufacturers
    public Response() {
 
    }
 
    public Response(int status, List<String> messages, T body) {
        this.status = status;
        this.messages = messages;
        this.body = body;
    }
 
    // getters and setters
...
}

Esta classe foi apresentada e discutida na Secção 13.5.5.3.

14.8. Implementação do serviço web / JSON

 

O serviço web /jSON é implementado pela seguinte classe [ElectionsController]:


package elections.webjson.service;
 
import java.util.ArrayList;
import java.util.List;
 
import javax.servlet.http.HttpServletRequest;
 
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.ResponseBody;
 
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
 
import elections.dao.entities.ElectionsConfig;
import elections.dao.entities.ElectionsException;
import elections.metier.service.IElectionsMetier;
 
@Controller
public class ElectionsController {
 
    // spring dependencies
    @Autowired
    private ObjectMapper jsonMapper;
 
    @Autowired
    private IElectionsMetier metier;
 
    @RequestMapping(value = "/getElectionsConfig", method = RequestMethod.GET, produces = "application/json; charset=UTF-8")
    @ResponseBody
    public String getElectionsConfig() throws JsonProcessingException {
        // answer
        Response<ElectionsConfig> response;
        try {
            response = new Response<>(0, null,
                    new ElectionsConfig(metier.getNbSiegesAPourvoir(), metier.getSeuilElectoral()));
        } catch (ElectionsException e1) {
            response = new Response<>(e1.getCode(), e1.getErreurs(), null);
        } catch (RuntimeException e2) {
            response = new Response<>(1000, getErreursForException(e2), null);
        }
        // answer
        return jsonMapper.writeValueAsString(response);
    }
 
    @RequestMapping(value = "/getListesElectorales", method = RequestMethod.GET, produces = "application/json; charset=UTF-8")
    @ResponseBody
    public String getListesElectorales() throws JsonProcessingException {
        throw new UnsupportedOperationException("Not supported yet");
    }
 
    @RequestMapping(value = "/setListesElectorales", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8", produces = "application/json; charset=UTF-8")
    @ResponseBody
    public String setListesElectorales(HttpServletRequest request) throws JsonProcessingException {
        throw new UnsupportedOperationException("Not supported yet");
    }
 
    @RequestMapping(value = "/calculerSieges", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8", produces = "application/json; charset=UTF-8")
    @ResponseBody
    public String calculerSieges(HttpServletRequest request) throws JsonProcessingException {
        throw new UnsupportedOperationException("Not supported yet");
    }
 
    // private methods -----------------------------
    // list of RuntimeException error messages
    private List<String> getErreursForException(Exception e) {
        // retrieve the list of exception error messages
        Throwable cause = e;
        List<String> erreurs = new ArrayList<>();
        while (cause != null) {
            // the message is retrieved only if it is !=null and not blank
            String message = cause.getMessage();
            if (message != null) {
                message = message.trim();
                if (message.length() != 0) {
                    erreurs.add(message);
                }
            }
            // next cause
            cause = cause.getCause();
        }
        return erreurs;
    }
 
}

Tarefa: Seguindo os passos da Secção 13.5.5, complete o código para a classe [ElectionsController].


Notas:

  • Não existem filtros JSON aqui porque as tabelas [CONF] e [LISTES] não estão ligadas por uma relação de chave estrangeira, o que simplifica significativamente o código do serviço web;
  • Não se esqueça das várias anotações Spring necessárias;
  • Os URLs receberão nomes baseados nos métodos associados;
  • O método [setListeElectorales] é chamado através de uma operação [POST]. O valor enviado é a matriz de listas concorrentes (do tipo ListeElectorale[]) com os seus atributos [lugares, votos, eliminados], que devem ser guardados na base de dados. Este método devolve uma [Response<Void>] com o campo [status=0] se não houver erros; caso contrário, devolve outro valor;
  • O método [calculateSeats] é chamado com uma operação [POST]. O valor enviado é a matriz de listas concorrentes (do tipo ElectoralList[]) com os seus atributos [nome, votos]. Este método retorna um [Response<ElectoralList[]>] com, como corpo, as listas eleitorais com os seus campos [lugares, eliminados] inicializados;

14.9. Testes

Após iniciar o serviço web, execute os seguintes testes para garantir que o serviço web está a funcionar corretamente utilizando o utilitário [Advanced Rest Client]:

 

A resposta JSON ao pedido anterior é a seguinte [1]:

1
2

Em [2], copie a resposta para a área de transferência e, em seguida, cole-a em qualquer editor de texto [3]:

Isole o valor do campo [body] e altere, por exemplo, os votos das listas. Abaixo [4], definimos os votos de todas as listas como 100:

Verifique se a sua cadeia JSON começa com [ e termina com ]. Estes caracteres são utilizados para delimitar uma matriz JSON. Em [5], cole a cadeia JSON acima. Este será o valor enviado para o URL seguinte. Para tal, selecione o método HTTP [POST] [7].

  • Em [6], solicite a URL [setListesElectorales]. Esta URL é solicitada através de um pedido POST. O valor enviado é a matriz JSON das listas concorrentes, cujos resultados devem ser guardados na base de dados;

O resultado obtido é o seguinte:

 

O campo [status=0] indica que não ocorreram erros. Para verificar isso, solicite novamente as listas concorrentes e verifique se as alterações que efetuou nas listas foram aplicadas:

Fazemos outra solicitação [POST] para calcular os lugares conquistados pelas listas:

  • em [1]: o URL para calcular os lugares;
  • em [2]: enviamos um pedido [POST];
  • em [3]: as listas concorrentes. Definimos o campo [votes] com os valores do tutorial, todos os [seats] são definidos como 0 e todos os campos [eliminate] são definidos como false;

O resultado obtido é o seguinte: