Nas partes 1 e 2 conseguimos fazer com que isso funcionasse manualmente:
${linkTo[ProdutoController].adiciona} => /produtos
Mas se a gente precisasse colocar todas as URIs do sistema no mapa linkTo
, não ia adiantar muita coisa. O legal é conseguir gerar esse mapa com todas as lógicas que existem no sistema de uma vez só. Para isso precisamos do componente do VRaptor que é responsável pelas rotas (URIs) das lógicas: o Router.
Com ele você consegue a URI de uma lógica a partir da classe e do método:
String uri = router.uriFor(Controller.class, metodo, argumentos);
com esse método conseguiríamos gerar o nosso mapa linkTo
:
Map<String, Map<String,String>> linkTo = Maps.newHashMap();
List<Class<?>> controllers = //todos os controllers do sistema
for (Class<?> controller : controllers) {
//*
List<Method> metodos = new Mirror().on(controller).reflectAll().methods();
Map<String, String> mapaDoMetodo = Maps.newHashMap();
for(Method metodo : metodos) {
String nome = metodo.getName();
//aridade é o número de parâmetros de um método. Precisamos passar um array
//com o tamanho da aridade do método para o uriFor funcionar
String uri = router.uriFor(controller, metodo, new Object[aridade(metodo)]);
mapaDoMetodo.put(nome, uri);
}
linkTo.put(controller.getSimpleName(), mapaDoMetodo);
}
request.setAttribute("linkTo", linkTo);
//...
private int aridade(Method method) {
return method.getParameterTypes().length;
}
* ajuda do Mirror, uma biblioteca bem legal feita pelo Jonas Abreu para fazer reflection. Já vem como dependência do VRaptor.
Temos dois problemas com esse código. O primeiro deles é: Como conseguir uma lista de todos os controllers do sistema? O VRaptor não tem um método específico pra isso na sua API mas, como sempre, dá pra improvisar se a gente conhecer um pouco da sua API interna.
Um dos jeitos é pegar todas as rotas do sistema (router.allRoutes()
) e, a partir delas construir a lista de controllers. O problema é que as rotas (interface Route) não têm um método direto pra isso (o mais perto seria: rota.resourceMethod(requestMockada, "qqer uri").getResource().getType()
, não mto legal, ou reflection acessando os fields da rota).
Outro jeito seria interceptar o registro dos controllers e guardá-los em uma lista. Com o VRaptor é possível interceptar qualquer uma das classes anotadas com uma das anotações do VRaptor (ou até com alguma anotação sua anotada com @Stereotype). Esse interceptador é uma classe que implementa StereotypeHandler. Então podemos fazer:
@Component
@ApplicationScoped //nesse escopo pois isso vai rodar na inicialização do VRaptor
public class LinkToHandler implements StereotypeHandler {
public Class<? extends Annotation> stereotype() {
return Resource.class; // intercepta todos os controllers
}
private List<Class<?>> controllers = Lists.newArrayList(); //guava ajudando aqui também =)
public void handle(Class<?> controller) {
controllers.add(controller);
}
public List<Class<?>> getControllers() {
return this.controllers;
}
}
Cada vez que o VRaptor acha um controller, ele vai chamar esse método handle, e adicioná-lo na lista. Assim conseguimos rodar o código que popula o mapa linkTo que eu postei acima. O código funciona muito bem, se nenhuma das rotas tem parâmetro (ou seja, se tivermos @Path("/produtos/{produto.id}"
) como vamos passar o id do produto?). O que nos leva ao segundo problema do código: como tratar URIs que têm parâmetros?
Se uma URI tem parâmetro, não podemos usar a nossa estratégia de gerar o mapa inteiro já com as URIs, pois ela vai ser diferente dependendo de qual parâmetro passamos para ela. Mas como passar parâmetros usando EL padrão?
${linkTo[ProdutosController].visualiza(produto)} => seria o ideal, mas não funciona =/
relaxando um pouco:
${linkTo[ProdutosController].visualiza[produto]} => isso sim, funciona! mas como?
Para poder fazer isso, o "método" visualiza precisa ser um mapa, que contém uma chave que é o produto que a gente passou. Poxa, será que precisamos criar um mapa com todos os produtos possíveis? Seria só loucura. Precisamos então de uma gambiarra solução mais criativa.
Para que o visualiza[produto]
funcione, precisamos que visualiza seja um mapa, mas ninguém falou que precisa ser um HashMap
ou um TreeMap
. A única coisa necessária é implementar Map
! Então vamos criar um mapa hackeado que retorne a uri dependo do objeto passado. O único método que a gente precisa implementar de verdade é o get, que é o método chamado quando a gente faz o vizualiza[produto]
.
O chato é que a interface Map tem vários métodos, e precisamos implementar todos. Mas o nosso amigo guava tem uma classe chamada ForwardingMap que é bem útil nesse caso (existem várias classes Forwarding que podem ser bem úteis também):
class Linker extends ForwardingMap<Object, String> {
public Linker(Class<?> controller, Method method) {...} //guarda em fields
@Override
protected Map<Object, Linker> delegate() {
return Maps.newHashMap(); //precisamos delegar para um mapa qualquer
}
@Override
public String get(Object key) {
return router.urlFor(controller, method, new Object[] { key });
}
}
//código da solução anterior
for(Class<?> controller : controllers) {
Map<String, Linker> mapaDosMetodos = Maps.newHashMap();
for(Method metodo : metodos) {
mapaDosMetodos.put(metodo.getName(), new Linker(controller, metodo));
}
}
Agora se chamarmos no jsp:
${linkTo[ProdutoController].visualiza[produtoComId3]} ==> /produtos/3, legal =)
mas
${linkTo[ProdutoController].adiciona} ==> br.com.caelum...Linker@1def231 =S
Droga, criamos outro problema. Tudo bem, só sobrescrever o toString
do Linker
para também retornar a uri. Antes de mostrar o código completo, como fazer para passar mais de um parâmetro?
public class ProdutoController {
@Get("/produtos/vendas/{dia}/{mes}/{ano}")
public void vendasDe(int dia, int mes, int ano) {...}
}
${linkTo[ProdutoController].vendasDe(6,6,2011)} ==> err.. não
${linkTo[ProdutoController].vendasDe[6,6,2011]} ==> hum.. seria bom, mas também não
${linkTo[ProdutoController].vendasDe[6][6][2011]} ==> aí sim =)
ou seja, cada vez que recebermos um parâmetro precisamos retornar um outro mapa. Tudo bem, só retornar sempre um linker, só que a cada vez com um parametro a mais preenchido:
class Linker extends ForwardingMap<Object, Linker> {
public Linker(Class<?> type, Method method) { // para não mudar o código anterior
this(type, method, new Object[aridade(method)], 0);
}
private Linker(Class<?> type, Method method, Object[] args, int index) {//guarda em fields}
@Override
protected Map<Object, Linker> delegate() {
return Maps.newHashMap();
}
@Override
public Linker get(Object key) {
Object[] newArgs = args.clone(); //para evitar memorização de argumentos
newArgs[index] = key;
return new Linker(type, method, newArgs, index + 1); //aumentando o índice para preencher o próximo parâmetro
}
@Override
public String toString() {
return router.urlFor(type, method, args); //só mostra a uri no último momento,
//qdo o mapa for mostrado na jsp
}
}
E pronto, dessa forma conseguimos suportar a geração dos links de qualquer lógica de um controller. Lembrando que se um parâmetro não é usado na URI, não precisa ser passado.
O legal dessa solução é que ela passa por várias coisas úteis (um pouco roubadas às vezes ;)), que são legais saber quando um problema cabeludo aparece na sua frente. Espero que tenham gostado =).
Código da solução final: https://gist.github.com/1064176. Em breve estará integrado ao código do VRaptor, com os devidos testes e documentação.
Grande, Lucas! Curti a solução! Só faltou concatenar o contextPath antes de retornar o router.urlFor(...) que fica melhor ainda! (Assim nao precisaríamos nem do c:url! ;D) Eu andei pensando e chamar ${linkTo[ProdutoController].vendasDe(6,6,2011)} seria EXCELENTE! Mas sem criação de classes extras em tempo de compilação ou runtime realmente fica impossível né? Mesmo com a EL 2.2 nos permitindo invocar métodos... Valeu e já estou utilizando em um projeto! (Com o contextPath embutido! ;) [ ]'s
Muito bom, Lucas. Espero ter essa funcionalidade como default no VRaptor. ;-)
Oi Lucas, parabéns pelo Post. Quando estará integrado ao vraptor?! Coloquei uma dúvida no GUJ, se puder dar uma olhada, agradeço!