Skip to content

15. [TD]: Criação de um cliente para o serviço web

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

15.1. Suporte

  

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

15.2. A arquitetura cliente/servidor

Queremos criar a seguinte arquitetura cliente/servidor:

A camada [ui] será a que já foi desenvolvida nas secções 9 e 10. Isto será possível porque a camada [business] acima implementará a mesma interface [IElectionsMetier] que a camada [business] da secção 8:


package elections.client.metier;
 
import elections.client.entities.ListeElectorale;

public interface IElectionsMetier {
 
    // get the lists in competition
    public ListeElectorale[] getListesElectorales();
 
    // the number of seats to be filled
    public int getNbSiegesAPourvoir();
 
    // the electoral threshold
    public double getSeuilElectoral();
 
    // recording results
    public void recordResultats(ListeElectorale[] listesElectorales);
 
    // calculating seats
    public ListeElectorale[] calculerSieges(ListeElectorale[] listesElectorales);
 
}

Na Secção 7, a camada [DAO] trocava dados com um SGBD. Aqui, a camada [DAO] troca dados com um servidor web / JSON.

Inicialmente, iremos concentrar-nos na seguinte arquitetura:

15.3. O Projeto Eclipse

O projeto Eclipse é o seguinte:

Esta estrutura é idêntica à do projeto de exemplo da Secção 13.6.1. Seguiremos a mesma abordagem.

15.4. Configuração do Maven

Esta é a descrita na Secção 13.6.2.

15.5. Implementação da camada [DAO]

  
  • O pacote [elections.client.config] contém a configuração Spring para a camada [DAO];
  • O pacote [elections.client.dao] contém a implementação da camada [DAO];
  • o pacote [elections.client.entities] contém os objetos trocados com o serviço web / JSON;
  • O pacote [elections.client.business] contém a camada [business]
  • O pacote [elections.client.ui] contém a camada [UI]

15.5.1. Configuração da camada [business]

  

A classe [MetierConfig] gere a configuração Spring da camada [business]. O seu código é o seguinte:


package elections.client.config;
 
import org.springframework.beans.factory.config.ConfigurableBeanFactory;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Scope;
import org.springframework.http.client.HttpComponentsClientHttpRequestFactory;
import org.springframework.web.client.RestTemplate;
 
import com.fasterxml.jackson.databind.ObjectMapper;
 
@ComponentScan({ "elections.client.dao","elections.client.metier" })
public class MetierConfig {
 
    // constants
    static private final int TIMEOUT = 1000;
    static private final String URL_WEBJSON = "http://localhost:8080";
 
    @Bean
    public RestTemplate restTemplate(int timeout) {
        // creation of the RestTemplate component
        HttpComponentsClientHttpRequestFactory factory = new HttpComponentsClientHttpRequestFactory();
        RestTemplate restTemplate = new RestTemplate(factory);
        // exchange timeout
        factory.setConnectTimeout(timeout);
        factory.setReadTimeout(timeout);
        // result
        return restTemplate;
    }
 
    @Bean
    public int timeout() {
        return TIMEOUT;
    }
 
    @Bean
    public String urlWebJson() {
        return URL_WEBJSON;
    }
 
    // mapper jSON
    @Bean
    @Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    public ObjectMapper jsonMapper() {
        return new ObjectMapper();
    }
}

Este código foi explicado na Secção 13.6.3.1. É mais simples porque não há filtros JSON para gerir aqui.

15.5.2. As entidades

  

As entidades tratadas pelas camadas [DAO] e [business] são aquelas que são trocadas com o serviço web / JSON. Trata-se de objetos do tipo [ElectionsConfig] e [VoterList]. No lado do servidor, estas entidades tinham anotações de persistência JPA. Aqui, essas anotações foram removidas. Estamos a incluir novamente o código da entidade para referência:

[Entidade abstrata]


package spring.webjson.client.entities;
 
 
public abstract class AbstractEntity {
    // properties
    protected Long id;
    protected Long version;
 
    // manufacturers
    public AbstractEntity() {
 
    }
 
    public AbstractEntity(Long id, Long version) {
        this.id = id;
        this.version = version;
    }
 
    // redefine [equals] and [hashcode]
    @Override
    public int hashCode() {
        return (id != null ? id.hashCode() : 0);
    }
 
    @Override
    public boolean equals(Object entity) {
        if (!(entity instanceof AbstractEntity)) {
            return false;
        }
        String class1 = this.getClass().getName();
        String class2 = entity.getClass().getName();
        if (!class2.equals(class1)) {
            return false;
        }
        AbstractEntity other = (AbstractEntity) entity;
        return id != null && this.id == other.id.longValue();
    }
 
    // getters and setters
    ...
}

[ElectionsConfig]


package elections.webjson.client.entities;
 
 
public class ElectionsConfig extends AbstractEntity {
 
    // fields
    private int nbSiegesAPourvoir;
    private double seuilElectoral;
 
    // manufacturers
    public ElectionsConfig() {
 
    }
 
    public ElectionsConfig(int nbSiegesAPourvoir, double seuilElectoral) {
        this.nbSiegesAPourvoir = nbSiegesAPourvoir;
        this.seuilElectoral = seuilElectoral;
    }
 
    // getters and setters
    ...
}

[ListaDeVotantes]


package elections.webjson.client.entities;
 
 
public class ListeElectorale extends AbstractEntity {
 
    // fields
    private String nom;
    private int voix;
    private int sieges;
    private boolean elimine;
 
    // manufacturers
    public ListeElectorale() {
    }
 
    public ListeElectorale(String nom, int voix, int sieges, boolean elimine) {
        setNom(nom);
        setVoix(voix);
        setSieges(sieges);
        setElimine(elimine);
    }
 
    // getters and setters
    ...
}

15.5.3. A interface da camada [DAO]

  

A camada [DAO] possui a seguinte interface [IClientDao]:


package elections.client.dao;
 
public interface IClientDao {
 
    // generic request
    String getResponse(String url, String jsonPost);
 
}

A interface tem apenas um método [getResponse]:

  • o primeiro parâmetro é a URL do servidor a consultar;
  • o segundo parâmetro é o valor JSON a ser enviado, nulo se não houver nada para enviar;
  • o resultado é a cadeia JSON de um objeto [Response<T>], em que a classe [Response] foi descrita na Secção 14.7;

15.5.4. Implementação da comunicação com o serviço web / JSON

  

A classe [ClientDao] implementa a interface [IClientDao] da seguinte forma:


package elections.client.dao;
 
import java.net.URI;
import java.net.URISyntaxException;
 
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.ParameterizedTypeReference;
import org.springframework.http.MediaType;
import org.springframework.http.RequestEntity;
import org.springframework.stereotype.Component;
import org.springframework.web.client.RestTemplate;
 
import elections.client.entities.ElectionsException;
 
@Component
public class ClientDao implements IClientDao {
 
    // data
    @Autowired
    protected RestTemplate restTemplate;
    @Autowired
    protected String urlServiceWebJson;
 
    // generic request
    @Override
    public String getResponse(String url, String jsonPost) {
 
        try {
            // url : URL to contact
            // jsonPost: the jSON value to be posted
 
            // request execution
            RequestEntity<?> request;
            if (jsonPost != null) {
                // query POST
                request = RequestEntity.post(new URI(String.format("%s%s", urlServiceWebJson, url)))
                        .header("Content-Type", "application/json").accept(MediaType.APPLICATION_JSON).body(jsonPost);
            } else {
                // query GET
                request = RequestEntity.get(new URI(String.format("%s%s", urlServiceWebJson, url)))
                        .accept(MediaType.APPLICATION_JSON).build();
            }
            // execute the query
            return restTemplate.exchange(request, new ParameterizedTypeReference<String>() {
            }).getBody();
        } catch (URISyntaxException e1) {
            throw new ElectionsException(200, e1);
        } catch (RuntimeException e2) {
            throw new ElectionsException(201, e2);
        }
    }
}

Este código é descrito na secção 13.6.3.6.

15.6. Implementação da camada [de negócios]

 

Como mencionado, a camada [de negócios] tem a mesma interface [IElectionsMetier] que na secção 8.4:


package elections.client.metier;
 
import elections.client.entities.ListeElectorale;
 
public interface IElectionsMetier {
 
    // get the lists in competition
    public ListeElectorale[] getListesElectorales();
 
    // the number of seats to be filled
    public int getNbSiegesAPourvoir();
 
    // the electoral threshold
    public double getSeuilElectoral();
 
    // recording results
    public void recordResultats(ListeElectorale[] listesElectorales);
 
    // calculating seats
    public ListeElectorale[] calculerSieges(ListeElectorale[] listesElectorales);
 
}

Esta interface é implementada pela seguinte classe [ElectionsMetier]:


package elections.client.metier;
 
import java.util.ArrayList;
import java.util.List;
 
import javax.annotation.PostConstruct;
 
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Component;
 
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;
 
import elections.client.dao.IClientDao;
import elections.client.entities.ElectionsConfig;
import elections.client.entities.ElectionsException;
import elections.client.entities.ListeElectorale;
 
@Component
public class ElectionsMetier implements IElectionsMetier {
 
    @Autowired
    private IClientDao dao;
    @Autowired
    private ApplicationContext context;
 
    // election configuration
    private ElectionsConfig electionsConfig;
 
    @PostConstruct
    public void init() {
        // mappers jSON
        ObjectMapper mapperResponse = context.getBean(ObjectMapper.class);
        try {
            // request
            Response<ElectionsConfig> response = mapperResponse.readValue(dao.getResponse("/getElectionsConfig", null),
                    new TypeReference<Response<ElectionsConfig>>() {
                    });
            // mistake?
            if (response.getStatus() != 0) {
                // 1 exception is thrown
                throw new ElectionsException(response.getStatus(), response.getMessages());
            } else {
                electionsConfig = response.getBody();
            }
        } catch (ElectionsException e1) {
            throw e1;
        } catch (Exception e2) {
            throw new ElectionsException(100, getMessagesForException(e2));
        }
    }
 
    @Override
    public ListeElectorale[] getListesElectorales() {
        ...
    }
 
    @Override
    public int getNbSiegesAPourvoir() {
        return electionsConfig.getNbSiegesAPourvoir();
    }
 
    @Override
    public double getSeuilElectoral() {
        return electionsConfig.getSeuilElectoral();
    }
 
    @Override
    public void recordResultats(ListeElectorale[] listesElectorales) {
    ...
    }
 
    @Override
    public ListeElectorale[] calculerSieges(ListeElectorale[] listesElectorales) {
...
    }
 
    // list of exception error messages
    private List<String> getMessagesForException(Exception exception) {
        // retrieve the list of exception error messages
        Throwable cause = exception;
        List<String> erreurs = new ArrayList<String>();
        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;
    }
}
 

O tipo [Resposta] utilizado na linha 37 é a resposta do servidor web/JSON descrita na secção 14.7;


Tarefa: Seguindo a secção 13.6.3.7, complete a classe [ElectionsMetier];


15.7. O teste JUnit

Voltemos à arquitetura cliente/servidor atualmente em construção:

A camada [JUnit] [1] comunica com a camada [Business] do servidor [5] através das camadas [2–4]. Ao garantir que as camadas [Business] [2] e [5] têm a mesma interface, tornamos as camadas [2–4] transparentes. A camada [1] parece comunicar diretamente com a camada [5]. O ponto interessante é que em [1] poderemos utilizar o teste JUnit que foi utilizado para testar a camada [Business] [5].

  

Tarefa: Execute o teste JUnit do projeto para verificar a sua implementação tanto do servidor como do cliente.


15.8. Implementação da camada [UI]

Voltemos à arquitetura que queremos construir:

Agora que a camada [Negócio] [2] foi construída e testada, podemos construir a camada [IU] [1].

Uma vez que a interface [IElectionsMetier] da camada [Business] é idêntica à do projeto descrito no parágrafo 8, podemos [3] copiar o projeto da camada [UI] do parágrafo 10. Este projeto era um projeto NetBeans. Basta copiar e colar as classes Java relevantes do NetBeans para o Eclipse. Depois de feito isto, há alguns ajustes de pacotes e importações a efetuar.

Faremos o mesmo para as classes executáveis no pacote [elections.client.boot] [4].

A classe [AbstractBootElections] é a seguinte:


package elections.client.boot;
 
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
 
import elections.client.config.UiConfig;
import elections.client.entities.ElectionsException;
import elections.client.ui.IElectionsUI;
 
public abstract class AbstractBootElections {
 
    // spring context retrieval
    protected AnnotationConfigApplicationContext ctx;
 
    public void run() {
        // instantiation layer [ui]
        IElectionsUI electionsUI = null;
        try {
            // spring context retrieval
            ctx = new AnnotationConfigApplicationContext(UiConfig.class);
            // ui] layer recovery
            electionsUI = getUI();
 
...
  • Linha 19: O contexto Spring definido pela classe de configuração [UiConfig] é instanciado. Esta classe é a seguinte:
  

A classe [UiConfig] é a seguinte:


package elections.client.config;
 
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Import;
 
@Import(MetierConfig.class)
@ComponentScan(basePackages = { "elections.client.ui" })
public class UiConfig {
}
  • Linha 6: Importa os beans da camada [business];
  • Linha 7: Indicamos que existem beans Spring no pacote [elections.client.ui];

Tarefa: Verifique se as versões de consola e Swing da camada [ui] estão a funcionar.