Skip to content

17. Exposição de uma base de dados na Web

17.1. Arquitetura do serviço Web/JSON

Iremos implementar a seguinte arquitetura:

  • em [1], as camadas [DAO, [JPA], JDBC] são implementadas utilizando uma das 24 configurações apresentadas nos capítulos anteriores, nomeadamente no parágrafo 15;
  • a camada [DAO] [3] do cliente remoto implementa a mesma interface que a camada [DAO] [1], o que nos permite utilizar a mesma camada de teste que nos capítulos anteriores. É como se as camadas [2-3] fossem transparentes para a camada [4];

Iremos basear-nos nos seguintes projetos:

  • o projeto [sgbd-config-jdbc], que configura a camada JDBC de um dos seis SGBDs;
  • o projeto [sgbd-config-jpa-*], que configura a camada JPA do SGBD selecionado para uma das três implementações JPA estudadas (Hibernate, EclipseLink, OpenJpa);
  • o projeto genérico [spring-jdbc-04], que implementa a camada [DAO] [1];
  • o projeto genérico [spring-jpa-generic] que implementa a camada [DAO] [2];
  • o projeto genérico [spring-webjson-server-jdbc-generic], que implementa um serviço web baseado no projeto [spring-jdbc-04];
  • o projeto genérico [spring-webjson-server-jpa-generic], que implementa um serviço web baseado no projeto [spring-jpa-generic];
  • o cliente genérico [spring-webjson-client-generic], que será o único cliente para todas as 24 configurações de serviço web;

17.2. Configurar o ambiente de desenvolvimento

Iremos trabalhar com os seguintes componentes:

  • Base de dados MySQL 5.6.25;
  • Implementação do Hibernate JPA;

Importe os seguintes projetos para o STS:

  
  • Os projetos [spring-webjson-*] podem ser encontrados na pasta [<examples>\spring-database-generic\spring-webjson];
  • Prima [Alt-F5] e, em seguida, regenere todos os projetos acima;

Para verificar se o ambiente de desenvolvimento está instalado corretamente, proceda da seguinte forma:

  • Inicie o serviço web utilizando a configuração de tempo de execução [spring-webjson-server-jpa-generic-hibernate], que depende de uma implementação JPA/Hibernate;

depois:

  • inicie o cliente para este serviço web utilizando a configuração de tempo de execução [spring-webjson-client-generic], que é um teste JUnit:

O teste deve ser aprovado:

  • Em [1], pare o serviço web e, em seguida, inicie-o com a configuração de tempo de execução [spring-webjson-server-jdbc-generic], que depende de uma implementação JDBC:

depois inicie o cliente para este serviço web com a configuração de tempo de execução [spring-webjson-client-generic]:

O teste deve ser aprovado:

 

17.3. Implementação de Serviço Web / JSON / JDBC

Vamos começar por nos concentrar na seguinte arquitetura:

onde a camada [DAO] [1] comunica diretamente com a camada JDBC do SGBD.

17.3.1. O projeto Eclipse para o serviço web

O projeto Eclipse para o serviço web / JSON / JDBC é o seguinte:

  

Trata-se de um projeto Maven cujo ficheiro [pom.xml] é o seguinte:


<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>dvp.spring.database</groupId>
    <artifactId>spring-webjson-server-jdbc-generic</artifactId>
    <version>0.0.1-SNAPSHOT</version>
 
    <name>spring-webjson-server-jdbc-generic</name>
    <description>démo spring mvc</description>
 
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.2.3.RELEASE</version>
    </parent>
 
    <dependencies>
        <!-- web layer -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <!-- layer [DAO] -->
        <dependency>
            <groupId>dvp.spring.database</groupId>
            <artifactId>spring-jdbc-generic-04</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>
    </dependencies>
    <!-- plugins -->
    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
                <version>2.18.1</version>
            </plugin>
        </plugins>
    </build>
 
</project>
  • linhas 11–15: o projeto Maven pai;
  • linhas 24–28: a dependência da camada [DAO / JDBC] implementada pelo projeto [spring-jdbc-generic-04];
  • linhas 19–22: a dependência do artefacto [spring-boot-starter-web]. Este artefacto inclui todas as dependências necessárias para criar um serviço web / JSON. Também inclui bibliotecas desnecessárias. Seria, portanto, necessária uma configuração mais precisa, mas esta configuração é útil para começar.

As dependências introduzidas por esta configuração são as seguintes:

1
  • Em [1], podemos ver que o Eclipse detetou a dependência do arquivo do projeto [spring-jdbc-generic-04];

As dependências acima são tanto da camada [DAO] como da camada [web].

17.3.2. Configuração da camada [web]

A camada [web] é configurada por dois ficheiros de configuração do Spring:

  

17.3.2.1. A classe [WebConfig]

A principal função da classe [WebConfig] é configurar:

  • o servidor Tomcat no qual o serviço web será implementado;
  • os filtros JSON para serializar e deserializar objetos [Product] e [Category]:

package spring.webjson.server.config;
 
import java.util.List;
 
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.Configuration;
import org.springframework.context.annotation.Scope;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.converter.json.MappingJackson2HttpMessageConverter;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.servlet.DispatcherServlet;
import org.springframework.web.servlet.config.annotation.EnableWebMvc;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;
 
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.ser.impl.SimpleBeanPropertyFilter;
import com.fasterxml.jackson.databind.ser.impl.SimpleFilterProvider;
 
@Configuration
@EnableWebMvc
public class WebConfig extends WebMvcConfigurerAdapter {
 
    // -------------------------------- 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("", 8081);
    }
 
    // -------------------------------- configuration filters [json]
    ...
}
  • linha 25: a classe é uma classe de configuração Spring;
  • linha 26: a anotação [@EnableWebMvc] indica que a camada web é implementada utilizando o Spring MVC. Isto irá acionar configurações implícitas que não teremos de definir;
  • linhas 30-31: injeção do contexto Spring da aplicação;
  • linhas 33–37: definição do bean [dispatcherServlet], que, em aplicações Spring MVC, atua como o [FrontController], cuja função é encaminhar os pedidos do cliente para o controlador capaz de os processar;
  • linhas 39–42: o servlet do serviço web é registado juntamente com os URLs que trata. Aqui escrevemos [/*], o que significa todos os URLs;
  • Linhas 44–47: Definição do bean [embeddedServletContainerFactory], que especifica o servidor web a utilizar. Neste caso, será o servidor web Tomcat [http://tomcat.apache.org/]. Também pode utilizar o servidor Jetty [http://www.eclipse.org/jetty/]. Ambos são servidores incorporados incluídos nas dependências do Maven. Quando o [Spring Boot] inicia o lançamento do projeto, inicia automaticamente o servidor web especificado na configuração e implementa o serviço ou a aplicação web nesse servidor;

Os filtros JSON estão configurados da seguinte forma:


package spring.webjson.server.config;
 
import java.util.List;
...
 
@Configuration
@EnableWebMvc
public class WebConfig extends WebMvcConfigurerAdapter {
 
    // -------------------------------- layer configuration [web]
...
    // -------------------------------- configuration filters [json]
    // mapping jSON
    @Bean
    public MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter() {
        final MappingJackson2HttpMessageConverter converter = new MappingJackson2HttpMessageConverter();
        final ObjectMapper objectMapper = new ObjectMapper();
        converter.setObjectMapper(objectMapper);
        return converter;
    }
 
    @Override
    public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
        converters.add(mappingJackson2HttpMessageConverter());
        super.configureMessageConverters(converters);
    }
 
    // filters jSON
    @Bean
    public ObjectMapper jsonMapper(MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter) {
        return mappingJackson2HttpMessageConverter.getObjectMapper();
    }
 
    @Bean
    @Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    ObjectMapper jsonMapperShortCategorie(MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter) {
        ObjectMapper jsonMapper = jsonMapper(mappingJackson2HttpMessageConverter);
        jsonMapper.setFilters(new SimpleFilterProvider().addFilter("jsonFilterCategorie",
                SimpleBeanPropertyFilter.serializeAllExcept("produits")));
        return jsonMapper;
    }
 
    @Bean
    @Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    ObjectMapper jsonMapperLongCategorie(MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter) {
        ObjectMapper jsonMapper = jsonMapper(mappingJackson2HttpMessageConverter);
        jsonMapper.setFilters(new SimpleFilterProvider().addFilter("jsonFilterCategorie",
                SimpleBeanPropertyFilter.serializeAllExcept()).addFilter("jsonFilterProduit",
                SimpleBeanPropertyFilter.serializeAllExcept("categorie")));
        return jsonMapper;
    }
 
    @Bean
    @Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    ObjectMapper jsonMapperShortProduit(MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter) {
        ObjectMapper jsonMapper = jsonMapper(mappingJackson2HttpMessageConverter);
        jsonMapper.setFilters(new SimpleFilterProvider().addFilter("jsonFilterProduit",
                SimpleBeanPropertyFilter.serializeAllExcept("categorie")));
        return jsonMapper;
    }
 
    @Bean
    @Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    ObjectMapper jsonMapperLongProduit(MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter) {
        ObjectMapper jsonMapper = jsonMapper(mappingJackson2HttpMessageConverter);
        jsonMapper.setFilters(new SimpleFilterProvider().addFilter("jsonFilterProduit",
                SimpleBeanPropertyFilter.serializeAllExcept()).addFilter("jsonFilterCategorie",
                SimpleBeanPropertyFilter.serializeAllExcept("produits")));
        return jsonMapper;
    }
}
  • Linha 8: A classe [WebConfig] estende a classe [WebMvcConfigurerAdapter]. Esta última classe configura a aplicação web com valores predefinidos. Quando se pretende personalizar esta configuração, é necessário substituir determinados métodos desta classe. Aqui, pretendemos substituir o método [configureMessageConverters] nas linhas [22–26] (repare na anotação @Override), que define uma lista de «conversores». O serviço web/JSON e o seu cliente trocam linhas de texto. Um conversor é uma ferramenta capaz de criar um objeto a partir de uma linha de texto recebida (deserialização) e de criar uma linha de texto a partir de um objeto ( ização). Aqui, as linhas de texto serão cadeias de caracteres JSON. Referir-nos-emos, portanto, à serialização/deserialização JSON;
  • linha 23: o método [configureMessageConverters] recebe uma lista de conversores como parâmetro;
  • Linhas 24–25: O conversor JSON [MappingJackson2HttpMessageConverter] das linhas [14–20] é adicionado a esta lista. Isto permitirá trocas de JSON entre o cliente e o servidor;
  • linhas [14-20]: definem um conversor JSON implementado pela classe [MappingJackson2HttpMessageConverter]. Esta classe (linha 10) pode ser encontrada nas dependências Maven do projeto;
  • linhas [17-18]: é criado um mapeador JSON e atribuído ao [MappingJackson2HttpMessageConverter];
  • linhas [29-32]: definem o mapeador JSON criado na linha 17 como um bean Spring. Isto coloca-o no contexto Spring e torna-o disponível para ser injetado noutros beans ou utilizado no código da aplicação web;
  • linhas 34-41: definem um filtro JSON para o mapeador JSON anterior;
  • linha 35: a anotação [@Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)] garante que o bean definido aqui não seja um singleton. Cada vez que for solicitado a partir do contexto, o método [jsonMapperCategoryWithoutProducts] será reexecutado. Isto é necessário aqui porque estamos a definir quatro filtros JSON. No entanto, apenas um deve estar ativo em qualquer momento. Ao atribuir ao bean o escopo [ConfigurableBeanFactory.SCOPE_PROTOTYPE], garantimos que o método será reexecutado e que o filtro anterior será substituído pelo novo;
  • Para compreender estes filtros, lembre-se de que na camada [DAO]:
    • a entidade [Product] foi anotada com a anotação [jsonFilterProduct];
    • a entidade [Category] foi anotada com [jsonFilterCategory];

Devemos, portanto, definir filtros com estes nomes.

  • linhas [34-41]: definem um filtro chamado [jsonMapperShortCategorie] que fornece a representação JSON de uma categoria sem os seus produtos;
  • linhas [43-51]: definem um filtro chamado [jsonMapperLongCategory] que fornece a representação JSON de uma categoria juntamente com os seus produtos;
  • linhas [53-60]: definem um filtro chamado [jsonMapperShortProduct] que fornece a representação JSON de um produto sem a sua categoria;
  • linhas [62-70]: definem um filtro chamado [jsonMapperLongProduct] que fornece a representação JSON de um produto com a sua categoria;

17.3.2.2. A classe [AppConfig]

A classe [AppConfig] configura toda a aplicação, ou seja, as camadas [web] e [DAO]:


package spring.webjson.server.config;
 
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Import;
 
@Configuration
@ComponentScan(basePackages = { "spring.webjson.server.service" })
@Import({ spring.jdbc.config.AppConfig.class, WebConfig.class })
public class AppConfig {
 
}
  • Linha 7: A classe é uma classe de configuração Spring;
  • linha 9: importamos os beans da camada [DAO / JDBC], bem como aqueles definidos pela classe [WebConfig]. Todos os beans da camada [DAO] estarão, portanto, disponíveis na aplicação web/JSON;
  • linha 8: especifica os pacotes onde outros beans Spring podem ser encontrados;

17.3.3. A [ServerException]

  

Tal como nos capítulos anteriores, em que a camada [DAO] lançou uma [DaoException] não capturada, a camada [web] lançará uma [ServerException] não capturada:


package spring.webjson.server.infrastructure;
 
import generic.jdbc.infrastructure.UncheckedException;
 
public class ServerException extends UncheckedException {
 
    private static final long serialVersionUID = 1L;
 
    // manufacturers
    public ServerException() {
        super();
    }
 
    public ServerException(int code, Throwable e, String simpleClassName) {
        super(code, e, simpleClassName);
    }
}
  • linha 5: a classe [ServerException] estende a classe [UncheckedException] definida no projeto que configura a camada JDBC (linha 3);

17.3.4. Os controladores

  

Aqui teremos dois controladores:

  • [CategoryController] irá tratar dos pedidos relacionados com categorias;
  • [CategorieController] irá tratar das solicitações relacionadas com produtos;

Os URLs expostos pelos controladores correspondem, numa relação de um para um, aos métodos das interfaces [DaoCategorie] e [DaoProduit] na camada [DAO]:

Assim, como mostrado acima:

  • o método web [deleteAllCategories] chamará o método [deleteAllEntities] da classe [DaoCategorie];
  • O método web [getShortCategoriesById] irá chamar o método [getShortEntitiesById] da classe [DaoCategorie];

O mesmo se aplica aos produtos:

17.3.4.1. Os URLs expostos pelo [CategoryController]

URL
Método

@RequestMapping(value = "/saveCategories",
 método = RequestMethod.POST, consome =
 "application/json; charset=UTF-8")

public Response<List<CoreCategory>>
 saveCategories
(HttpServletRequest request)
====
O método recebe as categorias a serem guardadas através de um pedido POST.
Estas estão acessíveis no [HttpServletRequest
 request]. As categorias são persistidas pelo
método [saveEntities] da camada [DaoCategorie]. Apenas
 dos objetos persistidos (categorias / produtos) são
 devolvidos ao cliente.

@RequestMapping(value = "/deleteAllCategories",
 method = RequestMethod.GET)

public Response<Void> deleteAllCategories()
====
A URL não tem parâmetros. As categorias são eliminadas pelo
 o método [deleteAllEntities] na camada [DaoCategorie].

@RequestMapping(value = "/deleteCategoriesById",
 method = RequestMethod.POST, consumes =
 "application/json; charset=UTF-8")

public Response<Void> deleteCategoriesById
(HttpServletRequest request)
====
O método recebe as chaves primárias das
 categorias a serem eliminadas. Estas estão acessíveis no
 objeto [HttpServletRequest request]. As categorias são
 eliminadas utilizando o método [deleteEntitiesById] da
 camada [DaoCategorie].

@RequestMapping(value = "/deleteCategoriesByName",
 method = RequestMethod.POST, consumes =
 "application/json; charset=UTF-8")

public Response<Void> deleteCategoriesByName
(HttpServletRequest request)
====
O método recebe os nomes das categorias a serem
eliminadas. Estes estão acessíveis no
 [HttpServletRequest request]. As categorias são eliminadas
 usando o método [deleteEntitiesByname] da
camada [DaoCategorie].

@RequestMapping(value = "/getAllShortCategories",
 method = RequestMethod.GET)

public Response<List<Category>> getAllShortCategories()
====
A URL não tem parâmetros. As categorias curtas são
recuperadas utilizando o método [getAllShortEntities] do
 [DaoCategorie].

@RequestMapping(value = "/getAllLongCategories",
 method = RequestMethod.GET)

public Response<List<Category>> getAllLongCategories()
====
A URL não tem parâmetros. As categorias longas são
 recuperadas utilizando o método [getAllLongEntities] do
[DaoCategorie].

@RequestMapping(value = "/getLongCategoriesById",
 method = RequestMethod.POST)

public Response<List<Category>> getLongCategoriesById(HttpServletRequest request)
====
O método recebe as chaves primárias das
 categorias pretendidas através de um pedido POST. Estas estão acessíveis no
 [HttpServletRequest request]. As categorias longas são
 recuperadas utilizando o método [getLongEntitiesById] da
 [DaoCategorie].

@RequestMapping(value = "/getLongCategoriesByName",
 method = RequestMethod.POST)

public Response<List<Category>> getLongCategoriesByName
(HttpServletRequest request)
====
O método recebe os nomes das categorias desejadas através de um pedido POST
 . Estes estão acessíveis no
 [HttpServletRequest request]. As categorias longas são
recuperadas utilizando o método [getLongEntitiesByName] do
 [DaoCategorie].

@RequestMapping(value = "/getShortCategoriesByName",
 method = RequestMethod.POST)

public Response<List<Category>> getShortCategoriesByName
(HttpServletRequest request)
====
O método recebe os nomes das categorias desejadas através de um pedido POST
 . Estes estão acessíveis no
 [HttpServletRequest request]. As categorias curtas são
 recuperadas pelo método [getShortEntitiesByName] da
 camada [DaoCategorie].

@RequestMapping(value = "/getShortCategoriesById",
 method = RequestMethod.POST)

public Response<List<Category>> getShortCategoriesById
(HttpServletRequest request)
====
O método recebe as chaves primárias das
 categorias pretendidas através de um pedido POST. Estas estão acessíveis no
 [HttpServletRequest request]. Os nomes abreviados das categorias são
 recuperadas através do método [getShortEntitiesById] da
 [DaoCategorie].

17.3.4.2. Os URLs expostos pelo [ProductController]

URL
Método

@RequestMapping(value = "/saveProducts", method =
 RequestMethod.POST, content-type = "application/json;
 charset=UTF-8")

public Response<List<CoreProduct>> saveProducts
(HttpServletRequest request)
====
O método recebe os produtos a serem guardados através de um pedido POST. Estes
estão acessíveis na [HttpServletRequest
 ]. Os produtos são persistidos pelo
 método [saveEntities] na camada [DaoProduit]. Apenas os
 são devolvidos ao cliente.

@RequestMapping(value = "/deleteAllProducts",
 method = RequestMethod.GET)

public Response<Void> deleteAllProducts()
====
A URL não tem parâmetros. Os produtos são eliminados pelo
 método [deleteAllEntities] na camada [DaoProduit] do .

@RequestMapping(value = "/deleteProductsById",
 method = RequestMethod.POST, consumes =
 "application/json; charset=UTF-8")

public Response<Void> deleteProductsById
(HttpServletRequest request)
====
O método recebe as chaves primárias dos produtos
 a serem eliminados através de um pedido POST. Estas estão acessíveis no
 [HttpServletRequest request]. Os produtos são eliminados
 utilizando o método [deleteEntitiesById] do
[DaoProduit].

@RequestMapping(value = "/deleteProductsByName",
 method = RequestMethod.POST, consumes =
 "application/json; charset=UTF-8")

public Response<Void> deleteProductsByName
(HttpServletRequest request)
====
O método recebe os nomes dos produtos a serem
 eliminados. Estes estão acessíveis no
 [HttpServletRequest request]. Os produtos são eliminados
 utilizando o método [deleteEntitiesByname] da
 camada [DaoProduit].

@RequestMapping(value = "/getAllShortProducts",
 method = RequestMethod.GET)

public Response<List<Product>> getAllShortProducts()
====
O URL não tem parâmetros. As categorias curtas são
 recuperadas utilizando o método [getAllShortEntities] da
 camada [DaoProduct].

@RequestMapping(value = "/getAllLongProducts",
 method = RequestMethod.GET)

public Response<List<Product>> getAllLongProducts()
====
A URL não tem parâmetros. Os produtos longos são recuperados
 usando o método [getAllLongEntities] da
 camada [DaoProduct].

@RequestMapping(value = "/getLongProductsById",
 method = RequestMethod.POST)

public Response<List<Product>> getLongProductsById
(HttpServletRequest request)
====
O método recebe as chaves primárias dos produtos desejados através de um pedido POST
 . Estas estão acessíveis no
 [HttpServletRequest request]. Os produtos longos são
recuperados utilizando o método [getLongEntitiesById] do

 [DaoProduit].

@RequestMapping(value = "/getLongProductsByName",
 method = RequestMethod.POST)

public Response<List<Product>> getLongProductsByName
(HttpServletRequest request)
====
O método recebe os nomes dos produtos desejados através de um pedido POST.
 Estes estão acessíveis no [HttpServletRequest
 request]. Os produtos longos são recuperados pelo
 método [getLongEntitiesByName] na camada [DaoProduit].

@RequestMapping(value = "/getShortProductsByName",
 method = RequestMethod.POST)

public Response<List<Product>> getShortProductsByName
(HttpServletRequest request)
====
O método recebe os nomes dos produtos desejados através de um pedido POST.
 Estes estão acessíveis no [HttpServletRequest
 request]. Os produtos resumidos são obtidos através do
método [getShortEntitiesByName] na camada [DaoProduit].

@RequestMapping(value = "/getShortProductsById",
 method = RequestMethod.POST)

public Response<List<Product>> getShortProductsById
(HttpServletRequest request)
====
O método recebe as chaves primárias dos produtos desejados através de um pedido POST
 . Estas estão acessíveis no
[HttpServletRequest request]. Os produtos resumidos são
recuperados utilizando o método [getShortEntitiesById] do
 [DaoProduit].

17.3.5. Implementação genérica do serviço web

  

A lista de URLs expostas pelo serviço web mostra que oferecemos os mesmos tipos de URLs para a gestão de categorias e produtos. Em vez de escrever dois controladores muito semelhantes, faremos com que derivem de uma classe que irá tratar de todo o trabalho comum a ambos os controladores. Esta será a classe [AbstractController] acima. Esta classe irá implementar a seguinte interface [Iws]:


package spring.webjson.server.service;
 
import java.util.List;
 
import javax.servlet.http.HttpServletRequest;
 
import spring.jdbc.entities.AbstractCoreEntity;
 
public interface Iws<T extends AbstractCoreEntity> {
 
    // list of all T entities
    public Response<List<T>> getAllShortEntities();
 
    public Response<List<T>> getAllLongEntities();
 
    // special entities - short version
    public Response<List<T>> getShortEntitiesById(HttpServletRequest request);
 
    public Response<List<T>> getShortEntitiesByName(HttpServletRequest request);
 
    // special entities - long version
    public Response<List<T>> getLongEntitiesById(HttpServletRequest request);
 
    public Response<List<T>> getLongEntitiesByName(HttpServletRequest request);
 
    // update of several entities
    public Response<List<T>> saveEntities(HttpServletRequest request);
 
    // delete all entities
    public  Response<Void> deleteAllEntities();
 
    // deletion of multiple entities
    public  Response<Void> deleteEntitiesById(HttpServletRequest request);
 
    public  Response<Void> deleteEntitiesByName(HttpServletRequest request);
}

Esta interface implementa os métodos da interface da camada [DAO] que serão utilizados:


package spring.jdbc.dao;
 
import java.util.List;
 
import spring.jdbc.entities.AbstractCoreEntity;
 
public interface IDao<T extends AbstractCoreEntity> {
 
    // list of all T entities
    public List<T> getAllShortEntities();
 
    public List<T> getAllLongEntities();
 
    // special entities - short version
    public List<T> getShortEntitiesById(Iterable<Long> ids);
 
    public List<T> getShortEntitiesById(Long... ids);
 
    public List<T> getShortEntitiesByName(Iterable<String> names);
 
    public List<T> getShortEntitiesByName(String... names);
 
    // special entities - long version
    public List<T> getLongEntitiesById(Iterable<Long> ids);
 
    public List<T> getLongEntitiesById(Long... ids);
 
    public List<T> getLongEntitiesByName(Iterable<String> names);
 
    public List<T> getLongEntitiesByName(String... names);
 
    // update of several entities
    public List<T> saveEntities(Iterable<T> entities);
 
    public List<T> saveEntities(@SuppressWarnings("unchecked") T... entities);
 
    // delete all entities
    public void deleteAllEntities();
 
    // deletion of multiple entities
    public void deleteEntitiesById(Iterable<Long> ids);
 
    public void deleteEntitiesById(Long... ids);
 
    public void deleteEntitiesByName(Iterable<String> names);
 
    public void deleteEntitiesByName(String... names);
 
    public void deleteEntitiesByEntity(Iterable<T> entities);
 
    public void deleteEntitiesByEntity(@SuppressWarnings("unchecked") T... entities);
}

A transição da interface [IDao<T>] na camada [DAO] para a interface [Iws<T>] no serviço web seguiu estas regras:

  • os métodos da interface [Iws<T>] não lançarão exceções. Se ocorrer uma exceção, esta será encapsulada no objeto [Response];
  • as variações de parâmetros, tais como as linhas 45 e 47 [Iterable<String> names, String... names], são removidas. Os métodos obtêm os seus parâmetros a partir do pedido HTTP do cliente do tipo [HttpServletRequest request];

Todas as respostas do serviço web serão encapsuladas no seguinte objeto [Response]:

  

package spring.webjson.server.service;
 
 
public class Response<T> {
 
    // ----------------- properties
    // operation status
    private int status;
    // an error message
    private String exception;
    // the body of the reply
    private T body;
 
    // manufacturers
    public Response() {
 
    }
 
    public Response(int status, String exception, T body) {
        this.status = status;
        this.exception = exception;
        this.body = body;
    }
 
    // getters and setters
...
}
  • linha 4: a resposta encapsula um tipo T;
  • linha 12: a resposta do tipo T;
  • linhas 7–10: um método pode encontrar uma exceção. Neste caso, irá devolver uma resposta com:
    • linha 8: status!=0;
    • linha 10: uma mensagem de erro;

A classe [AbstractController] é a seguinte:


package spring.webjson.server.service;
 
import java.util.List;
 
import javax.annotation.PostConstruct;
import javax.servlet.http.HttpServletRequest;
 
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
 
import spring.jdbc.dao.IDao;
import spring.jdbc.entities.AbstractCoreEntity;
import spring.jdbc.infrastructure.DaoException;
import spring.webjson.server.infrastructure.ServerException;
 
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.common.io.CharStreams;
 
public abstract class AbstractController<T extends AbstractCoreEntity> implements Iws<T> {
 
    @Autowired
    protected ApplicationContext context;
 
    // layer DAO
    private IDao<T> dao;
 
    abstract protected IDao<T> getDao();
 
    // local
    private String simpleClassName = getClass().getSimpleName();
 
    @PostConstruct
    public void init(){
        dao=getDao();
    }
    
    @Override
    public Response<List<T>> getAllShortEntities() {
        try {
            // answer
            return new Response<List<T>>(0, null, dao.getAllShortEntities());
        } catch (DaoException e) {
            return new Response<List<T>>(1, e.toString(), null);
        } catch (Exception e) {
            return new Response<List<T>>(2, new ServerException(1007, e, simpleClassName).toString(), null);
        }
    }
 
    @Override
    public Response<List<T>> getAllLongEntities() {
        try {
            // answer
            return new Response<List<T>>(0, null, dao.getAllLongEntities());
        } catch (DaoException e) {
            return new Response<List<T>>(1, e.toString(), null);
        } catch (Exception e) {
            return new Response<List<T>>(2, new ServerException(1008, e, simpleClassName).toString(), null);
        }
    }
 
    @Override
    public Response<List<T>> getShortEntitiesById(HttpServletRequest request) {
    ...
    }
 
    @Override
    public Response<List<T>> getShortEntitiesByName(HttpServletRequest request) {
    ...
    }
 
    @Override
    public Response<List<T>> getLongEntitiesById(HttpServletRequest request) {
    ...
    }
 
    @Override
    public Response<List<T>> getLongEntitiesByName(HttpServletRequest request) {
    ...
    }
 
        @Override
    public Response<List<T>> saveEntities(HttpServletRequest request) {
        return new Response<List<T>>(2, new ServerException(1013, new RuntimeException("[saveEntities] not implemented"), simpleClassName).toString(), null);
    }
 
 
    @Override
    public Response<Void> deleteAllEntities() {
        try {
            // we delete
            dao.deleteAllEntities();
            // answer
            return new Response<Void>(0, null, null);
        } catch (DaoException e) {
            return new Response<Void>(1, e.toString(), null);
        } catch (Exception e) {
            return new Response<Void>(2, new ServerException(1014, e, simpleClassName).toString(), null);
        }
    }
 
    @Override
    public Response<Void> deleteEntitiesById(HttpServletRequest request) {
        ...
    }
 
    @Override
    public Response<Void> deleteEntitiesByName(HttpServletRequest request) {
    ...
    }
 
}

Todos os métodos são implementados da mesma forma:

  1. se precisam de informações, obtêm-nas a partir do objeto [HttpServletRequest request];
  2. chamam o método na camada [DAO] que tem o mesmo nome que eles;
  3. tratam quaisquer exceções que possam ocorrer, seja na operação 1 (recuperação de parâmetros) ou na operação 2 (chamada à camada [DAO]);

Vamos primeiro examinar como a camada [DAO] é injetada na classe [AbstractController]:


public abstract class AbstractController<T extends AbstractCoreEntity> implements Iws<T> {
 
    @Autowired
    protected ApplicationContext context;
 
    // layer DAO
    private IDao<T> dao;
 
    abstract protected IDao<T> getDao();
 
    @PostConstruct
    public void init(){
        dao=getDao();
}

  • linha 1: a classe é abstrata e implementa a interface genérica [Iws<T>];
  • linhas 3-4: injeção do contexto Spring;
  • linha 7: a referência ainda desconhecida à camada [DAO] a ser utilizada;
  • linha 9: o método abstrato [getDao] que irá devolver a referência à camada [DAO] a ser utilizada. Este método será substituído pela classe filha, pelo que é a classe filha que irá especificar qual a camada [DAO] a utilizar (DaoProduct ou DaoCategory);
  • linha 11: a anotação [@PostConstruct] marca um método a ser executado assim que a instanciação do objeto estiver concluída. Uma vez concluída a instanciação, as injeções do Spring já foram realizadas. A classe filha terá então obtido a referência à sua camada [DAO] e poderá, portanto, passá-la à sua classe pai;

O método [getShortEntitiesById] é o seguinte:


    @Override
    public Response<List<T>> getShortEntitiesById(HttpServletRequest request) {
        try {
            // retrieve the posted value
            String body = CharStreams.toString(request.getReader());
            // we deserialize it
            ObjectMapper mapper = context.getBean("jsonMapper", ObjectMapper.class);
            List<Long> ids = mapper.readValue(body, new TypeReference<List<Long>>() {
            });
            // answer
            return new Response<List<T>>(0, null, dao.getShortEntitiesById(ids));
        } catch (DaoException e) {
            return new Response<List<T>>(1, e.toString(), null);
        } catch (Exception e) {
            return new Response<List<T>>(2, new ServerException(1009, e, simpleClassName).toString(), null);
        }
}
  • linha 5: o valor enviado pelo cliente será uma cadeia JSON. É aqui que é recuperado;
  • linhas 7–9: A cadeia JSON contém a lista de chaves primárias das entidades cuja versão resumida é pretendida;
  • linha 11: chamamos o método [DAO] com o mesmo nome. A resposta do tipo [List<T>] é encapsulada num objeto [Response];
  • linha 13: caso em que a camada [DAO] lançou uma exceção;
  • linha 15: caso para outras exceções, em particular a possível exceção durante a deserialização do parâmetro JSON, linha 8;

O método [getShortEntitiesByName] é semelhante:


@Override
    public Response<List<T>> getShortEntitiesByName(HttpServletRequest request) {
        try {
            // retrieve the posted value
            String body = CharStreams.toString(request.getReader());
            // we deserialize it
            ObjectMapper mapper = context.getBean("jsonMapper", ObjectMapper.class);
            List<String> noms = mapper.readValue(body, new TypeReference<List<String>>() {
            });
            // answer
            return new Response<List<T>>(0, null, dao.getShortEntitiesByName(noms));
        } catch (DaoException e) {
            return new Response<List<T>>(1, e.toString(), null);
        } catch (Exception e) {
            return new Response<List<T>>(2, new ServerException(1010, e, simpleClassName).toString(), null);
        }
}
  • linhas 4–9: aqui, o parâmetro jSON é a lista de nomes de categorias para as quais queremos a versão curta;

O método [saveEntities] não foi implementado porque depende fortemente do tipo de entidade a ser persistida, [Category] ou [Product]. Há pouco código para refatorar. Esta tarefa é, portanto, deixada para as classes filhas.


        @Override
    public Response<List<T>> saveEntities(HttpServletRequest request) {
        return new Response<List<T>>(2, new ServerException(1013, new RuntimeException("[saveEntities] not implemented"), simpleClassName).toString(), null);
    }

17.3.6. O [CategoryController]

  

O controlador [CategorieController] lida com URLs relacionadas com categorias:


package spring.webjson.server.service;
 
import java.util.ArrayList;
import java.util.List;
 
import javax.servlet.http.HttpServletRequest;
 
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
 
import spring.jdbc.dao.IDao;
import spring.jdbc.entities.Categorie;
import spring.jdbc.entities.Produit;
import spring.jdbc.infrastructure.DaoException;
import spring.webjson.server.entities.CoreCategorie;
import spring.webjson.server.entities.CoreProduit;
import spring.webjson.server.infrastructure.ServerException;
 
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.common.io.CharStreams;
 
@RestController
public class CategorieController extends AbstractController<Categorie> {
 
    @Autowired
    private IDao<Categorie> daoCategorie;
 
    @Override
    protected IDao<Categorie> getDao() {
        return daoCategorie;
    }
 
    // local
    private String simpleClassName = getClass().getSimpleName();
 
    @RequestMapping(value = "/getAllShortCategories", method = RequestMethod.GET)
    public Response<List<Categorie>> getAllShortCategories() {
        // parent
        Response<List<Categorie>> response = super.getAllShortEntities();
        // serialization filters jSON
        context.getBean("jsonMapperShortCategorie", ObjectMapper.class);
        // answer
        return response;
    }
 
    @RequestMapping(value = "/getAllLongCategories", method = RequestMethod.GET)
    public Response<List<Categorie>> getAllLongCategories() {
        // parent
        Response<List<Categorie>> response = super.getAllLongEntities();
        // serialization filters jSON
        context.getBean("jsonMapperLongCategorie", ObjectMapper.class);
        // answer
        return response;
    }
 
    @RequestMapping(value = "/getShortCategoriesById", method = RequestMethod.POST)
    public Response<List<Categorie>> getShortCategoriesById(HttpServletRequest request) {
        // parent
        Response<List<Categorie>> response = super.getShortEntitiesById(request);
        // serialization filters jSON
        context.getBean("jsonMapperShortCategorie", ObjectMapper.class);
        // answer
        return response;
    }
 
    @RequestMapping(value = "/getShortCategoriesByName", method = RequestMethod.POST)
    public Response<List<Categorie>> getShortCategoriesByName(HttpServletRequest request) {
        // parent
        Response<List<Categorie>> response = super.getShortEntitiesByName(request);
        // serialization filters jSON
        context.getBean("jsonMapperShortCategorie", ObjectMapper.class);
        // answer
        return response;
    }
 
    @RequestMapping(value = "/getLongCategoriesById", method = RequestMethod.POST)
    public Response<List<Categorie>> getLongCategoriesById(HttpServletRequest request) {
        // parent
        Response<List<Categorie>> response = super.getLongEntitiesById(request);
        // serialization filters jSON
        context.getBean("jsonMapperLongCategorie", ObjectMapper.class);
        // answer
        return response;
    }
 
    @RequestMapping(value = "/getLongCategoriesByName", method = RequestMethod.POST)
    public Response<List<Categorie>> getLongCategoriesByName(HttpServletRequest request) {
        // parent
        Response<List<Categorie>> response = super.getLongEntitiesByName(request);
        // serialization filters jSON
        context.getBean("jsonMapperLongCategorie", ObjectMapper.class);
        // answer
        return response;
    }
 
    @RequestMapping(value = "/saveCategories", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<List<CoreCategorie>> saveCategories(HttpServletRequest request) {
    ...
    }
 
    @RequestMapping(value = "/deleteAllCategories", method = RequestMethod.GET)
    public Response<Void> deleteAllCategories() {
        return super.deleteAllEntities();
    }
 
    @RequestMapping(value = "/deleteCategoriesById", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<Void> deleteCategoriesById(HttpServletRequest request) {
        return super.deleteEntitiesById(request);
    }
 
    @RequestMapping(value = "/deleteCategoriesByName", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<Void> deleteCategoriesByName(HttpServletRequest request) {
        return super.deleteEntitiesByName(request);
    }
 
}
  • linha 26: a classe [CategorieController] estende a classe [AbstractController];
  • linha 25: a anotação [@RestController] torna a classe um componente Spring. Esta anotação também indica que a classe é um serviço web cujos métodos enviam as suas respostas diretamente ao cliente no formato JSON;
  • linhas 28–29: a referência à camada [DAO] é injetada aqui;
  • linhas 31–34: redefinição do método [getDao], declarado abstrato na classe pai, cujo objetivo é devolver uma referência à camada [DAO] a ser utilizada;

Os métodos são todos construídos com base no mesmo modelo:

  • delegação do processamento à classe pai;
  • inicialização do mapeador JSON que irá serializar a resposta;
  • envio da resposta;

Vamos dar uma olhada nas assinaturas de algumas URLs:


@RequestMapping(value = "/getAllShortCategories", method
 = RequestMethod.GET)
- a URL [/getAllShortCategories] é chamada com um GET

@RequestMapping(value = "/getShortCategoriesById",
 method = RequestMethod.POST, consumes =
 "application/json; charset=UTF-8")
- A URL [/getShortCategoriesById] é chamada com uma solicitação POST. O valor enviado é a string JSON contendo as chaves primárias das
categorias pretendidas;

@RequestMapping(value = "/getLongCategoriesByName",
 method = RequestMethod.POST, consumes =
 "application/json; charset=UTF-8")
- A URL [/getLongCategoriesByName] é chamada com uma
POST. O valor enviado é a cadeia JSON que contém os nomes das
categorias desejadas;

Agora, vamos analisar mais detalhadamente o método [saveCategories], que não segue o formato dos outros métodos:


    @RequestMapping(value = "/saveCategories", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<List<CoreCategorie>> saveCategories(HttpServletRequest request) {
        // we persist categories
        try {
            // retrieve the posted value
            String body = CharStreams.toString(request.getReader());
            // we deserialize it
            ObjectMapper mapper = context.getBean("jsonMapperLongCategorie", ObjectMapper.class);
            List<Categorie> categories = mapper.readValue(body, new TypeReference<List<Categorie>>() {
            });
            // we persist categories
            categories = daoCategorie.saveEntities(categories);
            // we return the result
            List<CoreCategorie> coreCategories = new ArrayList<CoreCategorie>();
            for (Categorie categorie : categories) {
                CoreCategorie coreCategorie = new CoreCategorie(categorie.getId());
                coreCategories.add(coreCategorie);
                List<Produit> produits = categorie.getProduits();
                if (produits != null) {
                    List<CoreProduit> coreProduits = new ArrayList<CoreProduit>();
                    for (Produit produit : categorie.getProduits()) {
                        coreProduits.add(new CoreProduit(produit.getId()));
                    }
                    coreCategorie.setCoreProduits(coreProduits);
                }
            }
            // result
            return new Response<List<CoreCategorie>>(0, null, coreCategories);
        } catch (DaoException e) {
            return new Response<List<CoreCategorie>>(1, e.toString(), null);
        } catch (Exception e) {
            return new Response<List<CoreCategorie>>(2, new ServerException(1020, e, simpleClassName).toString(), null);
        }
}
  • linha 1: o URL [/saveCategories] é acompanhado por um valor enviado via POST. Trata-se da string JSON que contém as versões completas das categorias a serem persistidas;
  • linhas 5–10: as categorias a serem persistidas são recriadas a partir da string JSON. A ligação [product.category] que liga um [Product] à sua [Category] é nula porque, na versão longa de uma [Category], cada [Product] está na sua versão curta, sem o campo [category]. Isto não constitui um problema, uma vez que a camada [DAO] implementada com JDBC não requer esta informação;
  • linha 12: as categorias são persistidas. A lista de categorias recebida foi enriquecida com as chaves primárias dos elementos persistidos, categorias e produtos. Nada mais mudou. Em vez de devolver toda a lista recebida, o que é dispendioso, devolveremos apenas as chaves primárias dos elementos desta lista. Para o fazer, utilizamos as seguintes classes [CoreCategory] e [CoreProduct]:
  

package spring.webjson.server.entities;
 
import java.util.List;
 
public class CoreCategorie {
 
    // primary key
    private Long id;
    
    // manufacturers
    public CoreCategorie() {
 
    }
 
    public CoreCategorie(Long id) {
        this.id=id;
    }
 
    // list of products
    private List<CoreProduit> coreProduits;
 
    // getters and setters
    ...
}
  • linha 8: a chave primária de um produto;
  • linha 20: as chaves primárias dos seus produtos;

package spring.webjson.server.entities;
 
public class CoreProduit {
 
    // primary key
    private Long id;
 
    // manufacturers
    public CoreProduit() {
 
    }

    public CoreProduit(Long id) {
        this.id = id;
    }
 
    // getters and setters
...
}
  • linha 6: a chave primária de um produto;

Voltemos ao código do método [saveCategories]:


...            
// on persiste les catégories
            categories = daoCategorie.saveEntities(categories);
            // on rend le résultat
            List<CoreCategorie> coreCategories = new ArrayList<CoreCategorie>();
            for (Categorie categorie : categories) {
                CoreCategorie coreCategorie = new CoreCategorie(categorie.getId());
                coreCategories.add(coreCategorie);
                List<Produit> produits = categorie.getProduits();
                if (produits != null) {
                    List<CoreProduit> coreProduits = new ArrayList<CoreProduit>();
                    for (Produit produit : categorie.getProduits()) {
                        coreProduits.add(new CoreProduit(produit.getId()));
                    }
                    coreCategorie.setCoreProduits(coreProduits);
                }
            }
            // résultat
            return new Response<List<CoreCategorie>>(0, null, coreCategories);
...
  • linhas 5–17: criamos a lista de objetos [CoreCategory] que iremos devolver ao cliente remoto;
  • linha 19: a resposta é devolvida e serializada para JSON;

17.3.7. Tratamento de filtros JSON

Para cada método de um controlador, existem dois pontos para a serialização/desserialização JSON:

  • deserialização do valor enviado: isto é tratado explicitamente aqui;
  • serialização do resultado: isto é tratado implicitamente aqui;

Vamos começar pela deserialização do valor enviado em [CategorieController.saveCategories]:


    @RequestMapping(value = "/saveCategories", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<List<CoreCategorie>> saveCategories(HttpServletRequest request) {
        // we persist categories
        try {
            // retrieve the posted value
            String body = CharStreams.toString(request.getReader());
            // we deserialize it
            ObjectMapper mapper = context.getBean("jsonMapperLongCategorie", ObjectMapper.class);
            List<Categorie> categories = mapper.readValue(body, new TypeReference<List<Categorie>>() {
});
  • Linha 8: Recuperamos um mapper configurado para lidar com o filtro JSON [jsonMapperLongCategorie] do contexto Spring. Voltemos à definição deste mapper na classe de configuração [WebConfig]:

// -------------------------------- configuration filters [json]
    // mapping jSON
    @Bean
    public MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter() {
        final MappingJackson2HttpMessageConverter converter = new MappingJackson2HttpMessageConverter();
        final ObjectMapper objectMapper = new ObjectMapper();
        converter.setObjectMapper(objectMapper);
        return converter;
    }
 
    @Override
    public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
        converters.add(mappingJackson2HttpMessageConverter());
        super.configureMessageConverters(converters);
    }
 
    // filters jSON
    @Bean
    public ObjectMapper jsonMapper(MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter) {
        return mappingJackson2HttpMessageConverter.getObjectMapper();
    }
 
    @Bean
    @Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    ObjectMapper jsonMapperShortCategorie(MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter) {
        ObjectMapper jsonMapper = jsonMapper(mappingJackson2HttpMessageConverter);
        jsonMapper.setFilters(new SimpleFilterProvider().addFilter("jsonFilterCategorie",
                SimpleBeanPropertyFilter.serializeAllExcept("produits")));
        return jsonMapper;
    }
 
    @Bean
    @Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    ObjectMapper jsonMapperLongCategorie(MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter) {
        ObjectMapper jsonMapper = jsonMapper(mappingJackson2HttpMessageConverter);
        jsonMapper.setFilters(new SimpleFilterProvider().addFilter("jsonFilterCategorie",
                SimpleBeanPropertyFilter.serializeAllExcept()).addFilter("jsonFilterProduit",
                SimpleBeanPropertyFilter.serializeAllExcept("categorie")));
        return jsonMapper;
    }
 
  • linhas 32–40: o mapeador JSON [jsonMapperLongCategorie] recuperado pela classe [CategorieController];
  • linha 35: este mapeador é devolvido pelo método [jsonMapper] nas linhas 18–21;
  • linhas 18–21: o método [jsonMapper] devolve o mapeador JSON do [MappingJackson2HttpMessageConverter] nas linhas 3–9;

Por outras palavras, o mapeador JSON recuperado pela linha 4 abaixo em [CategorieController.saveCategories]:


            // on récupère la valeur postée
            String body = CharStreams.toString(request.getReader());
            // on la désérialise
            ObjectMapper mapper = context.getBean("jsonMapperLongCategorie", ObjectMapper.class);
            List<Categorie> categories = mapper.readValue(body, new TypeReference<List<Categorie>>() {
});

é o conversor padrão utilizado pelo Spring MVC para deserializar o valor enviado pelo cliente e serializar o resultado que lhe é devolvido. Nas linhas acima, não houve deserialização implícita do valor enviado. Para o fazer, teria de escrever:


    @RequestMapping(value = "/saveCategories", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<List<CoreCategorie>> saveCategories(@RequestBody List<Categorie> categories) {

Neste caso, teria ocorrido uma deserialização automática do valor enviado no parâmetro [categories]. Mas havia a questão do filtro [jsonFilterCategorie] que as entidades [Categorie] possuem. Este precisa de ser configurado. É por isso que optámos pela deserialização explícita (linhas 4–5). O segundo ponto a destacar é que o mapeador na linha 4 (que é o utilizado por predefinição pelo Spring MVC) também é adequado para serializar o resultado [Response<List<CoreCategory>]. Na verdade, a entidade [CoreCategorie] não possui um filtro JSON. Portanto, não há necessidade de configurar o mapeador JSON resultante com um filtro adicional. Neste caso, a resposta enviada ao cliente será serializada implicitamente.

17.3.8. O [ProductController]

  

O controlador [ProduitController] gere as URLs relacionadas com produtos. O seu código é semelhante ao do controlador [CategorieController]:


package spring.webjson.server.service;
 
import java.util.ArrayList;
import java.util.List;
 
import javax.servlet.http.HttpServletRequest;
 
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
 
import spring.jdbc.dao.IDao;
import spring.jdbc.entities.Produit;
import spring.jdbc.infrastructure.DaoException;
import spring.webjson.server.entities.CoreProduit;
import spring.webjson.server.infrastructure.ServerException;
 
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.common.io.CharStreams;
 
@RestController
public class ProduitController extends AbstractController<Produit> {
 
    @Autowired
    private IDao<Produit> daoProduit;
 
    @Override
    protected IDao<Produit> getDao() {
        return daoProduit;
    }
 
    // local
    private String simpleClassName = getClass().getSimpleName();
 
    @RequestMapping(value = "/getAllShortProduits", method = RequestMethod.GET)
    public Response<List<Produit>> getAllShortProduits() {
        // parent
        Response<List<Produit>> response = super.getAllShortEntities();
        // serialization filters jSON
        context.getBean("jsonMapperShortProduit", ObjectMapper.class);
        // answer
        return response;
    }
 
    @RequestMapping(value = "/getAllLongProduits", method = RequestMethod.GET)
    public Response<List<Produit>> getAllLongProduits() {
        // parent
        Response<List<Produit>> response = super.getAllLongEntities();
        // serialization filters jSON
        context.getBean("jsonMapperLongProduit", ObjectMapper.class);
        // answer
        return response;
    }
 
    @RequestMapping(value = "/getShortProduitsById", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<List<Produit>> getShortProduitsById(HttpServletRequest request) {
        // parent
        Response<List<Produit>> response = super.getShortEntitiesById(request);
        // serialization filters jSON
        context.getBean("jsonMapperShortProduit", ObjectMapper.class);
        // answer
        return response;
    }
 
    @RequestMapping(value = "/getShortProduitsByName", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<List<Produit>> getShortProduitsByName(HttpServletRequest request) {
        // parent
        Response<List<Produit>> response = super.getShortEntitiesByName(request);
        // serialization filters jSON
        context.getBean("jsonMapperShortProduit", ObjectMapper.class);
        // answer
        return response;
    }
 
    @RequestMapping(value = "/getLongProduitsById", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<List<Produit>> getLongProduitsById(HttpServletRequest request) {
        // parent
        Response<List<Produit>> response = super.getLongEntitiesById(request);
        // serialization filters jSON
        context.getBean("jsonMapperLongProduit", ObjectMapper.class);
        // answer
        return response;
    }
 
    @RequestMapping(value = "/getLongProduitsByName", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<List<Produit>> getLongProduitsByName(HttpServletRequest request) {
        // parent
        Response<List<Produit>> response = super.getLongEntitiesByName(request);
        // serialization filters jSON
        context.getBean("jsonMapperLongProduit", ObjectMapper.class);
        // answer
        return response;
    }
 
    @RequestMapping(value = "/saveProduits", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<List<CoreProduit>> saveProduits(HttpServletRequest request) {
        ...
    }
 
    @RequestMapping(value = "/deleteAllProduits", method = RequestMethod.GET)
    public Response<Void> deleteAllProduits() {
        return super.deleteAllEntities();
    }
 
    @RequestMapping(value = "/deleteProduitsById", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<Void> deleteProduitsById(HttpServletRequest request) {
        return super.deleteEntitiesById(request);
    }
 
    @RequestMapping(value = "/deleteProduitsByName", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<Void> deleteProduitsByName(HttpServletRequest request) {
        return super.deleteEntitiesByName(request);
    }
 
}

Apenas o método [saveProducts] tem uma estrutura diferente dos outros métodos:


@RequestMapping(value = "/saveProduits", method = RequestMethod.POST, consumes = "application/json; charset=UTF-8")
    public Response<List<CoreProduit>> saveProduits(HttpServletRequest request) {
        try {
            // retrieve the posted value
            String body = CharStreams.toString(request.getReader());
            // we deserialize it
            ObjectMapper mapper = context.getBean("jsonMapperShortProduit", ObjectMapper.class);
            List<Produit> produits = mapper.readValue(body, new TypeReference<List<Produit>>() {
            });
            // we persist products
            produits = daoProduit.saveEntities(produits);
            List<CoreProduit> coreProduits = new ArrayList<CoreProduit>();
            for (Produit produit : produits) {
                coreProduits.add(new CoreProduit(produit.getId()));
            }
            // we return the answer
            return new Response<List<CoreProduit>>(0, null, coreProduits);
        } catch (DaoException e) {
            return new Response<List<CoreProduit>>(1, e.toString(), null);
        } catch (Exception e) {
            return new Response<List<CoreProduit>>(2, new ServerException(1021, e, simpleClassName).toString(), null);
        }
}
  • linhas 4–9: A partir da cadeia JSON recebida, reconstruímos a lista de objetos [Product] a serem persistidos. Uma vez que a cadeia JSON recebida contém as versões curtas dos produtos, o seu campo [category] é nulo. Mais uma vez, a camada DAO/JDBC não necessita desta informação;
  • linha 11: os produtos são persistidos;
  • linhas 12–15: a lista de objetos [CoreProduct] a serem devolvidos é construída;
  • linha 18: a resposta é devolvida, sendo serializada (serialização implícita realizada pelo Spring MVC) pelo mapeador da linha 7 antes de ser enviada ao cliente remoto (ver discussão na secção 17.3.7);

17.3.9. O serviço web / classe de execução JSON

  

A classe [Boot] é a classe executável do projeto:


package spring.webjson.server.boot;
 
import org.springframework.boot.SpringApplication;
 
import spring.webjson.server.config.AppConfig;
 
public class Boot {
 
    public static void main(String[] args) {
        SpringApplication.run(AppConfig.class, args);
    }
}
  • Linha 10: O método estático [SpringApplication.run] é executado. A classe [SpringApplication] é uma classe do projeto [Spring Boot] (linha 3). São-lhe passados dois parâmetros:
    • [AppConfig.class]: a classe que configura toda a aplicação;
    • [args]: quaisquer argumentos passados para o método [main] na linha 9. Este parâmetro não é utilizado aqui;

Quando esta classe é executada, são gerados os seguintes registos:

  .   ____          _            __ _ _
 /\\ / ___'_ __ _ _(_)_ __  __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
 \\/  ___)| |_)| | | | | || (_| |  ) ) ) )
  '  |____| .__|_| |_|_| |_\__, | / / / /
 =========|_|==============|___/=/_/_/_/
 :: Spring Boot ::        (v1.2.3.RELEASE)

11:34:08.661 [main] INFO  spring.webjson.server.boot.Boot - Starting Boot on Gportpers3 with PID 6796 (started by ST in D:\data\istia-1415\spring data\dvp\dvp-spring-database-05\spring-database-generic\spring-webjson\spring-webjson-server-jdbc-generic)
11:34:08.700 [main] INFO  o.s.b.c.e.AnnotationConfigEmbeddedWebApplicationContext - Refreshing org.springframework.boot.context.embedded.AnnotationConfigEmbeddedWebApplicationContext@2df32bf7: startup date [Mon Jun 08 11:34:08 CEST 2015]; root of context hierarchy
11:34:08.916 [main] INFO  o.s.b.f.s.DefaultListableBeanFactory - Overriding bean definition for bean 'jsonMapper': replacing [Root bean: class [null]; scope=; abstract=false; lazyInit=false; autowireMode=3; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=generic.jdbc.config.ConfigJdbc; factoryMethodName=jsonMapper; initMethodName=null; destroyMethodName=(inferred); defined in class generic.jdbc.config.ConfigJdbc] with [Root bean: class [null]; scope=; abstract=false; lazyInit=false; autowireMode=3; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=spring.webjson.server.config.WebConfig; factoryMethodName=jsonMapper; initMethodName=null; destroyMethodName=(inferred); defined in class spring.webjson.server.config.WebConfig]
11:34:08.917 [main] INFO  o.s.b.f.s.DefaultListableBeanFactory - Overriding bean definition for bean 'jsonMapperLongCategorie': replacing [Root bean: class [null]; scope=prototype; abstract=false; lazyInit=false; autowireMode=3; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=generic.jdbc.config.ConfigJdbc; factoryMethodName=jsonMapperLongCategorie; initMethodName=null; destroyMethodName=(inferred); defined in class generic.jdbc.config.ConfigJdbc] with [Root bean: class [null]; scope=prototype; abstract=false; lazyInit=false; autowireMode=3; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=spring.webjson.server.config.WebConfig; factoryMethodName=jsonMapperLongCategorie; initMethodName=null; destroyMethodName=(inferred); defined in class spring.webjson.server.config.WebConfig]
11:34:08.918 [main] INFO  o.s.b.f.s.DefaultListableBeanFactory - Overriding bean definition for bean 'jsonMapperShortProduit': replacing [Root bean: class [null]; scope=prototype; abstract=false; lazyInit=false; autowireMode=3; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=generic.jdbc.config.ConfigJdbc; factoryMethodName=jsonMapperShortProduit; initMethodName=null; destroyMethodName=(inferred); defined in class generic.jdbc.config.ConfigJdbc] with [Root bean: class [null]; scope=prototype; abstract=false; lazyInit=false; autowireMode=3; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=spring.webjson.server.config.WebConfig; factoryMethodName=jsonMapperShortProduit; initMethodName=null; destroyMethodName=(inferred); defined in class spring.webjson.server.config.WebConfig]
11:34:08.919 [main] INFO  o.s.b.f.s.DefaultListableBeanFactory - Overriding bean definition for bean 'jsonMapperShortCategorie': replacing [Root bean: class [null]; scope=prototype; abstract=false; lazyInit=false; autowireMode=3; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=generic.jdbc.config.ConfigJdbc; factoryMethodName=jsonMapperShortCategorie; initMethodName=null; destroyMethodName=(inferred); defined in class generic.jdbc.config.ConfigJdbc] with [Root bean: class [null]; scope=prototype; abstract=false; lazyInit=false; autowireMode=3; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=spring.webjson.server.config.WebConfig; factoryMethodName=jsonMapperShortCategorie; initMethodName=null; destroyMethodName=(inferred); defined in class spring.webjson.server.config.WebConfig]
11:34:08.919 [main] INFO  o.s.b.f.s.DefaultListableBeanFactory - Overriding bean definition for bean 'jsonMapperLongProduit': replacing [Root bean: class [null]; scope=prototype; abstract=false; lazyInit=false; autowireMode=3; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=generic.jdbc.config.ConfigJdbc; factoryMethodName=jsonMapperLongProduit; initMethodName=null; destroyMethodName=(inferred); defined in class generic.jdbc.config.ConfigJdbc] with [Root bean: class [null]; scope=prototype; abstract=false; lazyInit=false; autowireMode=3; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=spring.webjson.server.config.WebConfig; factoryMethodName=jsonMapperLongProduit; initMethodName=null; destroyMethodName=(inferred); defined in class spring.webjson.server.config.WebConfig]
11:34:09.409 [main] INFO  o.s.b.c.e.t.TomcatEmbeddedServletContainer - Tomcat initialized with port(s): 8081 (http)
11:34:09.641 [main] INFO  o.a.catalina.core.StandardService - Starting service Tomcat
11:34:09.642 [main] INFO  o.a.catalina.core.StandardEngine - Starting Servlet Engine: Apache Tomcat/8.0.20
11:34:09.778 [localhost-startStop-1] INFO  o.a.c.c.C.[Tomcat].[localhost].[/] - Initializing Spring embedded WebApplicationContext
11:34:09.778 [localhost-startStop-1] INFO  o.s.web.context.ContextLoader - Root WebApplicationContext: initialization completed in 1081 ms
11:34:09.839 [localhost-startStop-1] INFO  o.s.b.c.e.ServletRegistrationBean - Mapping servlet: 'dispatcherServlet' to [/*]
11:34:10.558 [main] INFO  o.h.validator.internal.util.Version - HV000001: Hibernate Validator 5.1.3.Final
11:34:10.654 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerAdapter - Looking for @ControllerAdvice: org.springframework.boot.context.embedded.AnnotationConfigEmbeddedWebApplicationContext@2df32bf7: startup date [Mon Jun 08 11:34:08 CEST 2015]; root of context hierarchy
11:34:10.745 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/saveCategories],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.webjson.server.entities.CoreCategorie>> spring.webjson.server.service.CategorieController.saveCategories(javax.servlet.http.HttpServletRequest)
11:34:10.745 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/deleteCategoriesById],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.lang.Void> spring.webjson.server.service.CategorieController.deleteCategoriesById(javax.servlet.http.HttpServletRequest)
11:34:10.745 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/getShortCategoriesByName],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.jdbc.entities.Categorie>> spring.webjson.server.service.CategorieController.getShortCategoriesByName(javax.servlet.http.HttpServletRequest)
11:34:10.746 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/getAllLongCategories],methods=[GET],params=[],headers=[],consumes=[],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.jdbc.entities.Categorie>> spring.webjson.server.service.CategorieController.getAllLongCategories()
11:34:10.746 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/getLongCategoriesById],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.jdbc.entities.Categorie>> spring.webjson.server.service.CategorieController.getLongCategoriesById(javax.servlet.http.HttpServletRequest)
11:34:10.746 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/deleteCategoriesByName],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.lang.Void> spring.webjson.server.service.CategorieController.deleteCategoriesByName(javax.servlet.http.HttpServletRequest)
11:34:10.746 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/getShortCategoriesById],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.jdbc.entities.Categorie>> spring.webjson.server.service.CategorieController.getShortCategoriesById(javax.servlet.http.HttpServletRequest)
11:34:10.746 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/getAllShortCategories],methods=[GET],params=[],headers=[],consumes=[],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.jdbc.entities.Categorie>> spring.webjson.server.service.CategorieController.getAllShortCategories()
11:34:10.747 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/getLongCategoriesByName],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.jdbc.entities.Categorie>> spring.webjson.server.service.CategorieController.getLongCategoriesByName(javax.servlet.http.HttpServletRequest)
11:34:10.747 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/deleteAllCategories],methods=[GET],params=[],headers=[],consumes=[],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.lang.Void> spring.webjson.server.service.CategorieController.deleteAllCategories()
11:34:10.748 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/saveProduits],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.webjson.server.entities.CoreProduit>> spring.webjson.server.service.ProduitController.saveProduits(javax.servlet.http.HttpServletRequest)
11:34:10.749 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/getShortProduitsById],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.jdbc.entities.Produit>> spring.webjson.server.service.ProduitController.getShortProduitsById(javax.servlet.http.HttpServletRequest)
11:34:10.749 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/getAllLongProduits],methods=[GET],params=[],headers=[],consumes=[],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.jdbc.entities.Produit>> spring.webjson.server.service.ProduitController.getAllLongProduits()
11:34:10.749 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/getShortProduitsByName],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.jdbc.entities.Produit>> spring.webjson.server.service.ProduitController.getShortProduitsByName(javax.servlet.http.HttpServletRequest)
11:34:10.749 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/getAllShortProduits],methods=[GET],params=[],headers=[],consumes=[],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.jdbc.entities.Produit>> spring.webjson.server.service.ProduitController.getAllShortProduits()
11:34:10.749 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/deleteProduitsByName],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.lang.Void> spring.webjson.server.service.ProduitController.deleteProduitsByName(javax.servlet.http.HttpServletRequest)
11:34:10.750 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/getLongProduitsByName],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.jdbc.entities.Produit>> spring.webjson.server.service.ProduitController.getLongProduitsByName(javax.servlet.http.HttpServletRequest)
11:34:10.750 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/deleteProduitsById],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.lang.Void> spring.webjson.server.service.ProduitController.deleteProduitsById(javax.servlet.http.HttpServletRequest)
11:34:10.750 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/deleteAllProduits],methods=[GET],params=[],headers=[],consumes=[],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.lang.Void> spring.webjson.server.service.ProduitController.deleteAllProduits()
11:34:10.750 [main] INFO  o.s.w.s.m.m.a.RequestMappingHandlerMapping - Mapped "{[/getLongProduitsById],methods=[POST],params=[],headers=[],consumes=[application/json;charset=UTF-8],produces=[],custom=[]}" onto public spring.webjson.server.service.Response<java.util.List<spring.jdbc.entities.Produit>> spring.webjson.server.service.ProduitController.getLongProduitsById(javax.servlet.http.HttpServletRequest)
11:34:10.809 [main] INFO  o.a.coyote.http11.Http11NioProtocol - Initializing ProtocolHandler ["http-nio-8081"]
11:34:10.826 [main] INFO  o.a.coyote.http11.Http11NioProtocol - Starting ProtocolHandler ["http-nio-8081"]
11:34:10.860 [main] INFO  o.a.tomcat.util.net.NioSelectorPool - Using a shared selector for servlet write/read
11:34:11.733 [main] INFO  o.s.b.c.e.t.TomcatEmbeddedServletContainer - Tomcat started on port(s): 8081 (http)
11:34:11.934 [main] INFO  spring.webjson.server.boot.Boot - Started Boot in 3.533 seconds (JVM running for 4.137)
11:34:20.382 [http-nio-8081-exec-1] INFO  o.a.c.c.C.[Tomcat].[localhost].[/] - Initializing Spring FrameworkServlet 'dispatcherServlet'
11:34:20.384 [http-nio-8081-exec-1] INFO  o.s.web.servlet.DispatcherServlet - FrameworkServlet 'dispatcherServlet': initialization started
11:34:20.410 [http-nio-8081-exec-1] INFO  o.s.web.servlet.DispatcherServlet - FrameworkServlet 'dispatcherServlet': initialization completed in 26 ms
11:34:33.103 [http-nio-8081-exec-8] INFO  o.s.b.f.xml.XmlBeanDefinitionReader - Loading XML bean definitions from class path resource [org/springframework/jdbc/support/sql-error-codes.xml]
11:34:33.168 [http-nio-8081-exec-8] INFO  o.s.j.support.SQLErrorCodesFactory - SQLErrorCodes loaded: [DB2, Derby, H2, HSQL, Informix, MS-SQL, MySQL, Oracle, PostgreSQL, Sybase, Hana]
  • linhas 11-15: são detetados beans que definem filtros JSON. Estes substituem os beans com os mesmos nomes detetados no projeto de configuração da camada JDBC;
  • linhas 17-18: o servidor Tomcat é iniciado para executar o serviço web/JSON;
  • Linhas 19–21: O contexto Spring MVC é inicializado;
  • linhas 24–43: as URLs expostas são detetadas;

17.3.10. Testar o serviço web /jSON

Para realizar os testes, utilizamos o [Advanced Rest Client] (ver secção 23.11) para consultar as URLs expostas pelo serviço web /jSON (o serviço web /jSON deve estar em execução, assim como o SGBD, claro). Para preencher a base de dados, executamos a configuração de execução denominada [spring-jdbc-generic-04-fillDataBase], que preenche a base de dados com 5 categorias e 10 produtos:

 
  • Em [1-3], solicitamos o URL [/getAllLongCategories] através de uma solicitação HTTP GET;

Recebemos a seguinte resposta:

  • em [1], o pedido HTTP do cliente;
  • em [2], a resposta HTTP do servidor;
  • em [3], o estado [200 OK] indica que o servidor processou a solicitação com sucesso;
  • em [4], a resposta JSON do servidor;

A resposta JSON completa é a seguinte:


{"status":0,"exception":null,"body":[{"id":1880,"version":1,"nom":"categorie[0]","produits":[{"id":9072,"version":1,"nom":"produit[0,0]","idCategorie":1880,"prix":100.0,"description":"desc[0,0]"},{"id":9073,"version":1,"nom":"produit[0,1]","idCategorie":1880,"prix":101.0,"description":"desc[0,1]"},{"id":9074,"version":1,"nom":"produit[0,2]","idCategorie":1880,"prix":102.0,"description":"desc[0,2]"},{"id":9075,"version":1,"nom":"produit[0,3]","idCategorie":1880,"prix":103.0,"description":"desc[0,3]"},{"id":9076,"version":1,"nom":"produit[0,4]","idCategorie":1880,"prix":104.0,"description":"desc[0,4]"}]},{"id":1881,"version":1,"nom":"categorie[1]","produits":[{"id":9077,"version":1,"nom":"produit[1,0]","idCategorie":1881,"prix":110.00000000000001,"description":"desc[1,0]"},{"id":9078,"version":1,"nom":"produit[1,1]","idCategorie":1881,"prix":111.00000000000001,"description":"desc[1,1]"},{"id":9079,"version":1,"nom":"produit[1,2]","idCategorie":1881,"prix":112.00000000000001,"description":"desc[1,2]"},{"id":9080,"version":1,"nom":"produit[1,3]","idCategorie":1881,"prix":112.99999999999999,"description":"desc[1,3]"},{"id":9081,"version":1,"nom":"produit[1,4]","idCategorie":1881,"prix":114.00000000000001,"description":"desc[1,4]"}]}]}
  • status:0 significa que não houve erros do lado do servidor;
  • exception: null significa que não há mensagem de erro;
  • body: é o corpo da resposta, neste caso a lista de categorias com os seus produtos. Existem duas categorias, cada uma com 5 produtos;

Vamos adicionar o produto [product15] à categoria [category1]. Para isso, utilizaremos a URL [/saveProducts], que espera uma string JSON dos produtos a serem guardados (inserção/atualização). Esta string será a seguinte:

[{"id":null,"version":null,"nom":"produit15","idCategorie":1881,"prix":111.0,"description":"desc15"}]}]

O pedido ao serviço web / JSON é feito da seguinte forma:

  • em [1], o URL solicitado;
  • em [2], é solicitada através de uma operação POST;
  • em [3], a cadeia JSON enviada;
  • em [4], o servidor é notificado de que serão enviados dados JSON;

A resposta do servidor é a seguinte:

  • em [1], recebemos uma lista de objetos [CoreProduct] com as suas chaves primárias. Aqui, recebemos uma lista contendo um único item com a chave primária do produto que acabámos de inserir na base de dados;

Agora, vamos solicitar a versão completa da categoria denominada [category[1]:

  • Em [1], a URL solicitada;
  • em [2], fazemos um pedido POST;
  • em [3,4], o valor enviado é uma cadeia JSON. Isto representa a lista de nomes de categorias para as quais queremos a versão longa;

Obtemos o seguinte resultado:

  • em [5], a categoria [category[1]] tem agora um sexto produto;

Agora vamos eliminar este produto:

  • Em [1], o URL solicitado;
  • Em [2], fazemos uma solicitação POST;
  • em [3-4], enviamos uma string JSON que representa a lista de chaves primárias dos produtos que queremos eliminar;

O resultado é o seguinte:

 
  • [status:0] indica que a eliminação foi bem-sucedida;

Agora, vamos solicitar o produto [product[1,5]] para verificar se foi realmente eliminado:

 

Obtemos o seguinte resultado:

 
  • [status:0] indica que a operação foi concluída sem exceções;
  • [body:[0]] indica que [body] é uma lista vazia. A entidade [product[1,5]] foi, portanto, eliminada com sucesso;

Todas as operações [GET] podem ser realizadas num navegador web padrão:

Convidamos os leitores a testarem os outros URLs do serviço web / JSON.