‘pesado' é antigo… • Frameworks web server-side mandaram por um tempo (Struts, Spring, MVC, JSF) • Ajax foi uma mudança suave para o client-side (GWT, Vaadin) • Rich clients estão voltando voltaram, graças ao JavaScript/HTML5 • Motores Javascript melhoraram muito • Melhores ferramentas desenvolvimento • Melhores padrões (CSS3, HTML5, Websocket)
responsável pela UI, input, validação, lógica e estado • Server responsável pela lógica de negócio, modelo de domínio, persistência • Web/HTTP é a cola que conecta client e server • Protocolos de comunicação comuns • REST na maioria dos casos • WebSocket quando precisa de comunicação full-duplex • Ferramentas Javascript suportam REST muito bem, mas ainda não WebSocket • O formato comum (talvez ideal?) de troca de dados é JSON
• ︎Operações bem definidas (GET, POST, PUT) • ︎Sintaxe universal para identificação de recursos (URL) • ︎Transferência de informações em formato padrão (XML, HTML, JSON) • Web Services que seguem a arquitetura REST são denominados RESTful
Clientes somente precisam saber a URI root da aplicação e os media types utilizados • Descrevem o estado atual da aplicação e como navegar para o próximo estado
• Programação declarativa • Abstrações para implementação no server e client • Serviços implementados via POJO • Configuração via anotações • @Path, @GET, @POST, @PUT, @DELETE, @PathParam, @QueryParam, @Produces, @Consumes, etc • Plugável e extensível • Providers, filters, interceptors, validators • Suporte a processamento assíncrono • Integrado com as tecnologias do Java EE
@GET Determina acesso ao serviço via HTTP GET @POST Determina acesso ao serviço via HTTP POST @PUT Determina acesso ao serviço via HTTP PUT @DELETE Determina acesso ao serviço via HTTP DELETE @HEAD Determina acesso ao serviço via HTTP HEAD @PathParam Define o mapeamento do valor informado na URI para um determinado parâmetro de método @QueryParam Define o mapeamento do valor informado na query string para um determinado parâmetro de método @Consumes Define um determinado MIME type para recebimento de dados pelo serviço @Produces Define um determinado MIME type para envio de dados pelo serviço @Provider Define um determinado componente para auxiliar no JAX- RS runtime. @ApplicationPath Determina o root path de uma aplicação JAX-RS
anotações que permitem extrair informação de um request • @QueryParam e @DefaultValue • Extraem dados de um query string (?nome=valor&nome=valor) • @FormParam • Extrai dados de um formulário (applicaton/x-www-form-urlencoded) • @CookieParam • Extrai dados de cookies (pares nome=valor) • @HeaderParam • Extrai dados de cabeçalhos HTTP • @MatrixParam • Extrai dados de segmentos de URL
classes de resource podem ser validados através da API Bean Validation, que é configurada via anotações @POST @Path("/criar") @Consumes(MediaType.APPLICATION_FORM_URLENCODED) public void criarFilme( @NotNull @FormParam("titulo") String titulo, @NotNull @FormParam("diretor") String diretor, @Min(1900) @FormParam("ano") int ano) { ... } @Pattern(regexp="tt[0-9]{5-7}") private String imdbCode;
a conversão de objetos customizados Object para String e vice-versa • Por exemplo, pode ser utilizado para produzir um objeto java.util.Date a partir de uma String formatada. @Provider public class MyBeanConverterProvider implements ParamConverterProvider { @Override public <T> ParamConverter<T> getConverter( Class<T> clazz, Type type, Annotation[] annotations) { if (clazz.getName().equals(MyBean.class.getName())) { return new ParamConverter<T>() { @Override public T fromString(String value) {...} @Override public String toString(T bean) {...} }; } return null; } }
• Exemplo de definição de exceção customizada public class CustomNotFoundException extends WebApplicationException { public CustomNotFoundException() { super(Responses.notFound().build()); } } @Path("items/{itemid}/") public Item getItem(@PathParam("itemid") String itemid) { Item i = getItems().get(itemid); if (i == null) { throw new CustomNotFoundException("Item, " + itemid + ", is not found"); } return i; }
// @Compress annotation is the name binding annotation @NameBinding @Retention(RetentionPolicy.RUNTIME) public @interface Compress {} @Path("helloworld") public class HelloWorldResource { @GET @Path("too-much-data") @Compress public String getVeryLongString() {...} } @Compress public class GZIPWriterInterceptor implements WriterInterceptor { @Override public void aroundWriteTo(WriterInterceptorContext context) throws IOException, WebApplicationException {...} }
por meio de suporte à hypermedia • Em cada mensagem de resposta, deve ser incluído os links para a próxima mensagem • Utilizando este suporte, a aplicação consegue definir todo o modelo de navegação via HTTP @POST @Consumes({"application/json", "application/xml"}) @Produces({"application/json", "application/xml"}) public Response create(Article article) { Article created = articleDao.create(article); return Response.ok(created) .link("link-URI", "link-rel") .links(produceLinks(created)) .build(); } private Link[] produceLinks(Article article) {...}
tecnologias da plataforma Java Enterprise, especialmente com os componentes EJBs e CDI. • CDI beans podem ser injetados diretamente nos resources. Providers e Application terão comportamento singleton ou @ApplicationScoped @Path("/cdibean") public class CdiBeanResource { @Inject MyOtherCdiBean bean; // CDI injected bean @GET @Produces("text/plain") public String getIt() { return bean.getIt(); } }
de responses HTTP por meio da classe CacheControl @GET @Path("{id}") public Response read(@PathParam("id") int id) { Article article = articleDao.findById(id); CacheControl cacheControl = new CacheControl(); cacheControl.setMaxAge(60); return Response.ok(article) .cacheControl(cacheControl) .build(); }
o seu throughput • Libera a thread do servidor para executar outras tarefas • @Suspended indica que o método será executado de maneira assíncrona • Possível configurar timeout @Path(“Async") @RequestScope public class AsyncResource { @GET public void asyncGet(@Suspended final AsyncResponse asyncResponse) { new Thread(new Runnable() {...}).start(); } }
conexão TCP • Inicia através de um hand-shake através do protocolo HTTP, mas as conversações utilizam o protocolo WebSockets. • Suporte requisições assíncronas • Perfeito para aplicações como chat e jogos • Utiliza as tecnologias web existentes
em String e processado com métodos como split(), substring(), indexOf() dentre outros • Java EE 7 disponibiliza um API para construir objetos JSON e para converter strings JSON em mapas • É um par de APIs de baixo nível (não é mapeamento objeto- JSON) • Existem várias implementações que fazem mapeamento (binding) objeto-JSON automático (não são parte do Java EE) • MOXy, Jettison, Jersey, Jackson, etc. {id:123, cidade:”Paris”, voos:[“M344”,”J919”]}
de objetos JSON definida pelo Java EE • Object Model API - javax.json • Análogo a DOM: estrutura em árvore; I/O streaming via decorators • JsonObject: representa um objeto JSON • JsonArray: representa um array JSON • Leitura e gravação usando JsonReader e JsonWriter • JSON Streaming API - javax.json.stream • Análogo a SAX: leitura sequencial (baixo nível) • JsonParser: permite ler um stream JSON e capturar eventos • JsonGenerator: métodos para criar uma estrutura JSON
gerar representações JSON e XML no response @XmlRootElement public class MyJaxbBean { public String name; public int age; ... } @GET @Produces("application/json") public MyJaxbBean getMyBean() { return new MyJaxbBean("Agamemnon", 32); } {"name":"Agamemnon", "age":"32"}
modernas e dinâmicas • A primeira versão open-source foi liberada em 2010 e desde então ele é mantido pela Google e pela comunidade • Aproximadamente 2 releases mensais, projeto altamente ativo • O que faz o AngularJS ser especial?
cliente e servidor em JSON via REST ou WebSocket • Java EE funciona muito bem como backend para clientes ricos em Javascript, especialmente JAX-RS, Java API para WebSockets, e JSON-P • Enjoy it ;)