viernes, diciembre 07, 2012

Grails, elasticSearch, classcircularity y datepicker

Quiero una aplicación en grails que use datepickers molones y tenga un buscador incrustado, sencillo verdad??

En "efesto" pasen, pasen.... Lo cierto es que la cosa empieza bien, creas la aplicación, haces tus chorra dominios y lo primero que te encuentras es que el picker por defecto de grails es .....un poco mierdoso??

 

 Entonces te das cuenta que sin un framework de css y un conjunto de utilidades hechas por gente mucho mejor que tu no eres nadie y no quieres perder el tiempo con esas pijadas por lo que decides que es hora de meter bootstrap en tu vida (y por tanto jquery).

Así que después de explorar un poco consigues que tu app tenga un aspecto muy chulo (ver ejemplos) e incluso encuentras un datepicker que es compatible con tu nuevo look (http://www.eyecon.ro/bootstrap-datepicker/).

La cosa va bien, eres feliz, aunque ingeniuo porque si haces la prueba tal cual veras que tu controlador cuando hace la instrucción:

miDominio.properties=params

no se entera del precioso campo "fechaDeMiCreacion" y pasa de ti, en otras palabras no sabe como convertir de String a Date por ese formato tan chulo que estas usando debido al datepicker.

Así que volvemos a tirar de San Google y leemos que tenemos que hacer un CustomPropertyEditor para decirle como se convierte el agua en vino, quiero decir, de String a Date, fácil, creamos nuestro PropertyEditorRegistrar que dice lo que tiene que hacer, supongamos (src/groovy/mi.paquete/CustomDatePropertyEditorRegistrar.groovy):  

package my.paquete

import java.text.SimpleDateFormat;
import java.util.Date;

import org.springframework.beans.PropertyEditorRegistrar;
import org.springframework.beans.PropertyEditorRegistry;
import org.springframework.beans.propertyeditors.CustomDateEditor;
import org.springframework.context.i18n.LocaleContextHolder;

class CustomDatePropertyEditorRegistrar implements PropertyEditorRegistrar {

 def messageSource;

 public void registerCustomEditors(PropertyEditorRegistry registry) {
   registry.registerCustomEditor(Date.class, new CustomDateEditor(new SimpleDateFormat(messageSource.getMessage("editor.date.format",null,'dd/MM/yyyy',LocaleContextHolder.locale )),true));
 }

}
Con esto ya sabemos como hacer el paso, fíjate que uso las messages.properties para poder cambiar el formato de la fecha y hacerlo fácilmente i18n messages.properties:

editor.date.format=MM/dd/yyyy

messages_es.properties:

editor.date.format=dd/MM/yyyy

Ahora para que toda la aplicación sepa de tu logro (ejem...), debes registrar el bean que da el cambiazo, ya sabes, resources.groovy:

beans = {

 customDatePropertyEditor(my.paquete.CustomDatePropertyEditorRegistrar) {
      messageSource = ref('messageSource')
 }
................

Ahora si, el binding de las fechas funciona ok, :D. Estamos???NOOOOOOOOO

Deciamos al principio que queriamos la app con buscador, así que muy ufános hacemos la búsqueda y aunque dudamos entre elasticSearch y searchable por el primero, parece más guay y nosotros queremos ser guays, no?

grails install-plugin elasticSearch

Revisamos nuestros dominios (como Saurón) y añadimos :

static searchable=true

Arrancamos la moto y ..........ostión!

Lo más seguro que el mensaje que diga algo obvio como ClassCircularityError y te quedes pensando que tienes alguna milonga de referencias circulares o sabe Dios el qué.

La solución??

ElasticSearch como cualquier bicho que indexa necesita persistir los objetos que maneja, es decir serializar y deserializar, es decir escribir en fichero y leer del fichero, en su caso lo hace usando el formato JSON.

Te vas oliendo la tostada?

El caso es que acabamos de cambiar la forma en que se hace el binding que pasa de String a Date y eso aunque creamos que solo nos afecta al convertir la request en chicha en los controllers es exactamente lo que hace el buscador cuando escribe y sobre todo lee los datos indexados para devolverte resultados.

Así que ahora resulta que la conversión de agua en vino es más bien la conversión de agua en vino o mosto según se pida por lo que el CustomDateEditor que antes usamos se nos queda un poco flojo y tenemos que darle cera por ahi, al tema:  

package my.paquete
import java.beans.PropertyEditorSupport
import java.text.ParseException
import java.text.SimpleDateFormat

import org.apache.commons.lang.time.DateUtils;
import org.springframework.beans.PropertyEditorRegistrar
import org.springframework.beans.PropertyEditorRegistry
import org.springframework.beans.propertyeditors.CustomDateEditor
import org.springframework.context.i18n.LocaleContextHolder

/**
 * Registro de customs
 * @author ivan.arrizabalaga
 */
class CustomDatePropertyEditorRegistrar implements PropertyEditorRegistrar {
 def messageSource;

 /**
  * Registra el customDateEditor
  */
 public void registerCustomEditors(PropertyEditorRegistry registry) {
  def inputFormat=messageSource.getMessage("editor.date.format",null,'dd/MM/yyyy',LocaleContextHolder.locale)
   registry.registerCustomEditor(Date.class,
     new SearchableDateEditor(inputFormat,true)
    );
  }
}

Hasta aquí más  o menos lo mismo de antes pero metiendo mi propia clase en vez del CustomDateEditor por defecto, sigo:

/**
 * Custom property editor que admite 2 formatos para el parseo de fechas uno para el parseo de fechas desde los formularios y otro para el unmarshall del buscador
 * @author ivan.arrizabalaga
 *
 */
class SearchableDateEditor extends PropertyEditorSupport{
 boolean allowEmpty
 def searchableFormat="yyyy-MM-dd'T'HH:mm:ss'Z'"
 def inputFormat="dd/MM/yyyy"
 String[] formats=[]

 SearchableDateEditor(String inputFormat,boolean allowEmpty){
  this.allowEmpty=true
  this.inputFormat=inputFormat
  this.formats=[this.inputFormat,this.searchableFormat]
 }

 /**
  * Crea un 'Date' a partir de un 'String' se usa en:
  *  -binding de los params (Ej: en los controllers: miCosa.properties=params)
  *  -unmarshall de los indices del buscador
  */
 void setAsText(String text) throws IllegalArgumentException {
  //0.Fechas vacias
  if(this.allowEmpty && !text){
   setValue(null)
  }else{
  //1.Intentamos parsear
   try{
    setValue(DateUtils.parseDate(text, this.formats))
   }catch(ParseException ex){
    throw new IllegalArgumentException("Could not parse date: " + ex.getMessage(), ex)
   }
  }
 }

 /**
  * Devuelve el 'String' a partir del 'Date', se usa:
  *  -marshall del buscador
  */
 String getAsText() {
  def val=getValue()
  val?.respondsTo('format')? val.format(this.searchableFormat):''
 }

}

Ahora si, esta es la buena, ahora cuando parsea String a Date lo intenta con ambos formatos el de el buscador y el de el datepicker y cuando pasa de Date a String usa el formato propio del JSON del buscador para hacer la persistencia.

Mola o qué?

Grails N a M: como diablos va esto??

Las relaciones N a M en grails oficialmente se resuelven simplemente con un doble hasMany una parte que manda a través del belongsTo, sin embargo esto sólo es válido para aquellas relaciones N a M sin atributos adicionales. Lo cierto es que esto casi nunca es así y siempre que hay una N a M en realidad hay una entidad nueva pendiente de ser "descubierta". Un ejemplo:
1 Autor -> N Libros
1 Libro -> N Autores
En realidad hay una entidad intermedia que podría indicar si el autor es principal, los capitulos que ha escrito del libro, un contrato de derechos de autor especifico para cada colaboración.......aja! Vaya! Pues si que habia una entidad en medio, supongamos que la llamamos Autoría y que por simplificar indica si el autor es el principal o no (como Ana Rosa....ejem). Con esto vemos que en realidad tendriamos lo siguiente:
1 Autor -> N Autorias
1 Libro -> N Autorias
1 Autoría -> 1 Libro, 1 Autor
Ahora en virtud de las bondades de GORM (no confundir con la Isla de Gorm) hacer ciertas operaciones como añadir autorias, borrar autores o libros puede resultar un infierno de problemas de hibernate, flushes y demás historias. Así que visto lo visto y basándonos en este articulo(http://grails.org/Many-to-Many+Mapping+without+Hibernate+XML) quedaría el siguiente código fuente: Autor:
package com.nortia.simple
class Autor {

 String nombre

 static hasMany=[autorias:Autoria]
 static constraints = {}
 def libros(){
   return autorias.collect{it.libro}
 }

 def addToLibros(Libro libro, def params=null){
   Autoria.link(this, libro,params)
   return libros()
 }

 def removeFromLibros(Libro libro){
   Autoria.unlink(this, libro)
   return libros()
 }

 def clearLibros(){
   libros().each{
     removeFromLibros(it)
   }
 }

 def safeDelete(){
   this.clearLibros()
   this.delete()
 }
}
Libro:
package com.nortia.simple
class Libro {

 String titulo

 static hasMany=[autorias:Autoria]
static constraints = {
 }

 def autores(){
   return autorias.collect{it.autor}
 }

 def addToAutores(Autor autor, def params=null){
   Autoria.link(autor, this,params)
   return autores()
 }

 def removeFromAutores(Autor autor){
   Autoria.unlink(autor, this)
   return autores()
 }

 def clearAutores(){
   autores().each{
     removeFromAutores(it)
   }
 }

 def safeDelete(){
   this.clearAutores()
   this.delete()
 }
}
Autoría:
package com.nortia.simple
class Autoria {
 Autor autor
 Libro libro
 boolean autorPrincipal=true

 static belongsTo=[autor:Autor, libro:Libro]

 static Autoria link(def autor, def libro,def params=null){
   def autoria=Autoria.findByAutorAndLibro(autor,libro)
   if(!autoria){
     autoria=new Autoria(params)
     autor?.addToAutorias(autoria)
     libro?.addToAutorias(autoria)
     autoria.save()
   }
   return autoria
 }

 static void unlink(def autor, def libro){
   def autoria=Autoria.findByAutorAndLibro(autor,libro)
   if(autoria){
     autor?.removeFromAutorias(autoria)
     libro?.removeFromAutorias(autoria)
     autoria.delete()
   }
 }

}
Test (lanzando la consola como grails console):
import com.nortia.simple.*
println "0.Autorias: ${Autoria.list()}"
println "0.Autores: ${Autor.list()}"
println "0.Libros: ${Libro.list()}"
Autoria.executeUpdate("delete from Autoria")
Autor.executeUpdate("delete from Autor")
Libro.executeUpdate("delete from Libro")
def a=new Autor(nombre:"Carlos Ruiz Zafon")
a.save()
def l=new Libro(titulo:"La sombra del viento")
l.save()
def l2=new Libro(titulo:"La sombra del viento 2")
l2.save()
a.libros()
Autoria.link(a,l,[autorPrincipal:false]) //Opcion con params1
// Opción completa a.addToLibros(l2,[autorPrincipal:false]
a.addToLibros(l2) //Opción básica
a.libros()*.titulo
println "1.Autorias: ${Autoria.list()}"
def autorias=Autoria.list()
autorias.each{
  println "Autor: ${it.autor.nombre} Libro:${it.libro.titulo} \\\
          Es principal: ${it.autorPrincipal}"
}
a.safeDelete()
println "2.Autorias: ${Autoria.list()}"
Fijate que no hay ningún flush por ningún lado, esto es buen síntoma, quiere decir que tienes operaciones atómicas que no requieren del volcado de la sesión. No obstante si ejecutas 2 veces el mismo test se ve que los libros existen al principio y efectivamente

miércoles, febrero 01, 2012

Debugging Griffon in Eclipse/STS

Developing big java clients could sound a little bit strange these days, Where is my Jquery? Where is my Android stuff? :D

All right, I must admit that these days are awful for those in charge of developing Swing apps but there is a great framework out there that could help you a lot.

It's Griffon (http://griffon.codehaus.org/), it's based on Groovy and Grails and it's definitively cool but.... What should I do in order to watch my variables in Eclipse?

Let's get started:
  1. Create an Eclipse project (File > New > Groovy Project; "Create from existing source", i.e. your [projectname] folder, as created by the command line 'griffon create-app')

  2. Preparing everything:

    • You'll need to set GRIFFON_HOME and USER_HOME variables

      • Menu: Window > Preferences

      • Drill into Java > Build Path > 'Classpath Variables', and add a GRIFFON_HOME

      • GRIFFON_HOME as usual

      • Watch out! USER_HOME in Windows could be tricky (ex: C:\Users\ivan.arrizabalaga)


    • Build project (everything should work no error messages)
      • In some cases there is still one error, there is no "services" folder just create it (a workaround, :D).


  3. Start app in 'debug' mode (could be also done from console with griffon run-app -debug):

    • Outside Eclipse create the Ant file for the project (build.xml): griffon integrate-with -ant

    • I'm running the debug-app task from the eclipse 'Ant' view.

      • If you cant see an Ant view, choose menu: Window > Show View > Ant

      • Drag [projectname]/build.xml from the Package Explorer view into the Ant view

      • Expand the project item and double-click debug-app.

      • Then, I'm noting the port number from the console output: [exec] Listening for transport dt_socket at address: 18290


  4. Start a Remote Debugger:
    • Create and save a 'Debug Configuration':

      • Menu: Run > Debug Configurations ...

      • Right-click "Remote Java Application" and select 'New'.

      • In Connection Properties, put Host: localhost, Port: 18290, then click 'Debug'.

    • Change perspective to 'Debug'


Now you can see threads (really important for swing apps), variables (want to see an Alien? inspect an Expando..) but above all Expressions, it will sweeten the experience, :D.

PD:
This post is possible due to:
None of them worked for me but if you mix it properly, bumm!

sábado, enero 28, 2012

Que podemos hacer?

Durante años los vientos favorables nos hicieron pensar que, como decia Andres Montes, "la vida puede ser maravillosa".
Hoy los periodicos, la televisión, incluso la mirada de mucha gente te dice lo contrario, de forma machacona y monótona ... vamos de culo.

¿Es esto así? ¿Que podemos hacer?

Lo primero es un ejercicio de salud, mandarlos a todos a la mierda. Estudios científicos demuestran que esta técnica jamás falla.

Lo segundo hacer un ejercicio de introspección para repriorizar las personas, costumbres y vivencias que nos llenan, notesé como NO se hace referencia a las "cosas".

Lo último pensar que nos quedan 2 días antes de que o bien nos caiga una maceta en la cabeza o bien nos invadan los extraterrestres que para el caso tanto nos daría.

Ya está, ya lo tenemos. Ahora sólo queda actuar en consecuencia, apaga la tele y sal a liarla.