Páginas

jueves, 9 de abril de 2009

Google App Engine and Struts

Recientemente hemos podido leer como ahora Google App Engine soporta Java. Ni corto ni perezoso me he lanzado a probarlo. Pero ya puestos no nos íbamos a conformar con probar un simple Servlet ¿no?

En este enlace podemos ver una prueba de Google App Engine funcionando con Struts 1.3.

Para desarrollar la aplicación y desplegarla he utilizado el plugin de Google para Eclipse.

Por ahora me he quedado aquí, pero habrá que seguir estudiando todas las posibilidades que Google App Engine nos ofrece a los desarrolladores Java.

viernes, 13 de febrero de 2009

Maven corrompe las imágenes

"Maven está corrompiendo las imágenes" es una frase que que he oido a tres compañeros de mi proyecto actual. Tranquilos, no os preocupeis por eso - les dije yo.

¿Por qué Maven corrompe las imágenes cuando por ejemplo, construimos un war con nuestra aplicación Web? Este problema surge cuando "filtramos" nuestros archivos de recursos (src/main/resources) para sustituir las variables indicadas con los delimitadores ${...}. Qué gran utilidad sin duda para poder construir nuestra aplicación Web para diferentes entornos de forma simple y automática.

¡¡Pero es que corrompe las imáganes!! La solución es bastante sencilla, sólo debéis buscar en Google las siguientes palabras: maven filtering resource images y en la segunda entrada, además de explicaros qué es eso del filtrado, os da la solución para que éste no corrampa las imágenes u otros archivos.

Aquí os pongo el fagmento en particular:

<project> ... <build> ... <resources> <resource> <directory>src/main/resources</directory> <filtering>true</filtering> <excludes> <exclude>**/*.jpg</exclude> </excludes> </resource> <resource> <directory>src/main/resources</directory> <filtering>false</filtering> <includes> <include>**/*.jpg</include> </includes> </resource> ... </resources> ... </build> ...</project>


Aquí os dejo el enlace con más detalles: http://maven.apache.org/plugins/maven-resources-plugin/examples/filter.html

lunes, 29 de diciembre de 2008

Supervivencia a la hora de escribir código

"Always code as if the person who will maintain your code is a maniac serial killer that knows where you live"

Y es que, nunca se sabe si la persona que va a mantener tu código es un programador o un asesino en serie. Pero claro, qué fue antes, ¿el huevo o la gallina?, porque ¿a quienes no os ha entrado ganas de matar a alguien tras ver algunas líneas de código? A mí sí, es algo que ya sabéis.

Por otro lado, si queréis mantener vuestro puesto de trabajo para siempre, corriendo el riesgo de ser asesinado por un asesino en serie claro está, siempre podéis usar las técnicas de escritura de código inmantenible. Si echáis un vistazo a estas prácticas seguro que os daréis cuenta que sois habituales en algunas de ella aunque de forma menos exagerada (o quizás no). Tal vez sea nuestro instinto de supervivencia lo que nos hace escribir código inmantenible pero paradójicamente esto nos pueda llevar a ser asesinados  por un asesino en serie.

viernes, 5 de diciembre de 2008

Spring - JPA - Hibernate

Estos días he tenido que montar el esqueleto para un nuevo proyecto. Para mi sorpresa la combinación de Frameworks que se requerían no estaba entre las disponibles de Appfuse Light así que tuve que montarla casi desde cero.

Son muchos los ejemplos y la documentación que se puede encontrar que hablan acerca de cómo montar JPA con Spring, pero como suele suceder casi todos proceden de un ejemplo original por lo que al final todos quedan reducidos a muy pocos que para colmo, están incompletos. Así pues, voy a hacer mi pequeña aportación y describiré el archivo de Spring al completo. No describiré cómo se realizan Test o cómo se crean claves primarias compuestas porque son temas que ya tratan bien en otros blogs.

En primer lugar, empezaré definiendo una entidad, sobre la que girará el ejemplo: User.

package es.whatabout.model; @Entity @Table(name = "USER") public class User{ @Id @Column(name = "ID") private Integer idUser; @Column(name = "FIRST_NAME") private String firstName; @Column(name = "LAST_NAME") private String lastName; ...get/set methods...


Como se puede ver, esta clase lleva anotaciones JPA que identifican a esta clase como un entidad, su clave primaria y el mapeo de sus atributos a la tabla y columnas correspondientes en la base de datos. Esta será la única clase que contenga anotaciones ya que el resto de la configuración, como la transaccionalidad, la configuraremos desde Spring.

Antes de definir la clase DAO crearemos un interfaz genérico y su implementación que proporcionará a nuestro dao particular y a todos los que generemos (y su implementaciones) de todos los métodos CRUD gracias a los tipos genéricos de Java.

import java.io.Serializable; import java.util.List; public abstract interface GenericDao <T, PK extends Serializable> { public abstract List<T> getAll(); public abstract T findByPK(PK id); public abstract void update(T object); public abstract void remove(T object); public abstract void insert(T object); }

Y su implementación.

package es.whatabout.dao.impl; import java.io.Serializable; import java.lang.reflect.ParameterizedType; import java.util.ArrayList; import java.util.List; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.springframework.orm.jpa.support.JpaDaoSupport; import es.whatabout.dao.GenericDao; public class GenericDaoImpl extends JpaDaoSupport implements GenericDao { private static Log logger = LogFactory.getFactory().getInstance(GenericDaoImpl.class); private Class persistentClass; public GenericDaoImpl() { this.persistentClass = (Class) ((ParameterizedType) getClass() .getGenericSuperclass()).getActualTypeArguments()[0]; } public void insert(T t) { getJpaTemplate().persist(t); } public T findByPK(PK id) { return getJpaTemplate().find(persistentClass, id); } public List getAll() { List entities = new ArrayList(); try { String s = "select c from " + persistentClass.getSimpleName() + " c"; entities = getJpaTemplate().find(s); } catch (Exception e) { logger.error(e.getStackTrace()); } return entities; } public void remove(T t) { getJpaTemplate().remove(getJpaTemplate().merge(t)); } public void update(T object) { getJpaTemplate().merge(object); } }


T será la entidad que maneje nuestra clase dao mientras que PK será la clase de la clave primaria de ésta, que puede ser de tipo simple o compuesta (composed primary key)

A continuación crearemos una interfaz para un dao particular y su implementación utilizando el interfaz y la implementación de éste antes definido.

package es.whatabout.dao; import es.whatabout.model.User; public interface UserDao extends GenericDao<User, Integer> { }

Vemos que nuestro interfáz es muy simple, ya que dispone de los métodos CRUD del interfaz GenericDao. Del mismo modo, su implementación también será muy simple ya que ésta extiende de la implementación de GenericDao, GenericDaoImpl

package es.whatabout.dao.impl; import es.whatabout.dao.UserDao; import es.whatabout.model.User; public class UserDaoImpl extends GenericDaoImpl<User, Integer> implements UserDao{ }

Sólo queda definir el archivo de configuración de Spring:

<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd http://www.springframework.org/schema/aop http:// www.springframework.org/schema/aop/spring-aop-2.5.xsd"> <!-- archivo de configuracion para los datos de conexion --> <bean id="propertyConfigurer" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"> <property name="location" value="classpath:jdbc.properties" /> </bean> <!-- Transaccionalidad para todas las clases del paquete indicado --> <aop:config> <aop:advisor id="managerTx" advice-ref="txAdvice" pointcut="execution(* es.whatabout.dao.impl.*.*(..))" /> </aop:config> <!-- restringimos a solo lectura las transacciones metodos que comienzan por get --> <tx:advice id="txAdvice"> <tx:attributes> <tx:method name="get*" read-only="true" /> <tx:method name="*" /> </tx:attributes> </tx:advice> <!-- configuramos el transaction manager--> <bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager"> <property name="dataSource" ref="dataSource" /> <property name="entityManagerFactory" ref="entityManagerFactory" /> </bean> <!-- Conexion a bd con las propiedades del arhivo jdbc.properties--> <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource"> <property name="driverClassName" value="${jdbc.driverClassName}" /> <property name="url" value="${jdbc.url}" /> <property name="username" value="${jdbc.username}" /> <property name="password" value="${jdbc.password}" /> </bean> <!-- configuramos la Factory Manager, que trabajará con hibernate--> <bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean"> <property name="dataSource" ref="dataSource" /> <property name="jpaVendorAdapter"> <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter"> <property name="showSql" value="true" /> <property name="databasePlatform" value="${hibernate.dialect}" /> </bean> </property> <property name="jpaProperties"> <value>hibernate.show_sql=true</value> </property> </bean> <!-- configuramos LOG4J --> <bean id="log4jInitialization" class="org.springframework.beans.factory.config.MethodInvokingFactoryBean"> <property name="targetClass" value="org.springframework.util.Log4jConfigurer" /> <property name="targetMethod" value="initLogging" /> <property name="arguments"> <list> <value>classpath:log4j.xml</value> </list> </property> </bean> <!-- inyeccion del entityManagerFActory a nuestro Dao particular--> <bean id="userDao" class="es.whatabout.dao.impl.UserDaoImpl"> <property name="entityManagerFactory" ref="entityManagerFactory" /> </bean> </beans>


Con este archivo de configuración ya podemos disponer de nuestro dao capaz de realizar todas las operaciones CRUD sin ningún problema.

Referencia: The Spring Framework - Reference Documentation, chapter 9, Introduction to Spring 2 and JPA
Código: descargar

jueves, 2 de octubre de 2008

Y llegó el final

Después de diez meses en el proyecto para el Grupo Antena3, que empezó cuatro meses antes de mi llegada, éste ha llegado su fin, al menos para la mayoría, yo entre ellos.

Se han cometido muchos errores, a todos los niveles y de toda clases, algunos incluso fuera de mi alcance y comprensión pero sin duda también ha habido muchas cosas positivas que merecen ser destacadas.

Entre ellas, la evolución de todos los miembros del proyecto y del equipo en conjunto que se puede ver reflejada en los cuatro portales desarrollados hasta ahora (y un quinto en camino) ya que cada portal que hemos desarrollado refleja las ganas y el esfuerzo del equipo por hacer las cosas mejor.

Sé que el propio equipo, desde los programadores hasta los gestores, e incluso el mismo cliente son conscientes de esta evolución resultado del esfuerzo personal a través de muchas horas de trabajo y las ganas de aprender y hacer las cosas bien por parte de todos los componentes.

Todos los miembros del equipo podemos y debemos, a pesar de todo, estar orgullosos de nuestro trabajo, de lo que hemos conseguido bajos unas condiciones muy duras (falta de documentación, tiempos de entrega imposibles, desplazamiento en cliente...).

Recuerdo cómo en mi primera etapa en Indra cuando al equipo nos caía un marrón decíamos entre risas: ¿y lo que vamos a aprender?. Ahora, tras tanto esfuerzo, yo le digo al equipo: ¿y lo que hemos aprendido?. Sin duda mucho. Por eso es tiempo de mirar atrás para contemplar todo aquello que hemos aprendido y reflexionar sobre qué aspectos debemos seguir mejorando. Sí sí, yo tengo que mejorar mi diplomacia y mi mano izquierda, lo sé lo sé.

Por otra parte, debido al tamaño del proyecto, son viarios los miembros del equipo que han tenido la oportunidad de dar un paso al frente y hacerse cargo de responsabilidades huérfanas cuando, como se dice comúnmente, no están pagadas. Algunos de ellos lo hicieron, dieron un paso al frente, tomaron responsabilidades y demostraron su capacidad. Qué gusto da ver a una persona aprovechar una oportunidad para demostrar su valía y salir triunfante. Otros, no lo hicieron y ha sido una lástima ya que estas oportunidades no se presentan todos los días.

Ahora, sólo queda esperar nuevo proyecto.

martes, 12 de agosto de 2008

La caducidad de lo evidente

A pesar de los años que han pasado, todavía recuerdo con exactitud aquel día de mi primer año en la facultad, hace ya 9 años, en el que estando cogiendo apuntes como un loco en la clase de Física mi fijé en cómo tomaba apuntes la compañera que tenía a mi izquierda. La chica tomaba apuntes de todo lo que decía o escribía en la pizarra el profesor como si de una mecanógrafa se tratase, cosa que me parecía absurdo e inútil, mientras que yo sólo tomaba nota de aquello que consideraba importante, no molestándome en anotar aquellas cosas que me parecían evidentes y que por tanto no se me olvidarían.

Con la llegada de los primeros exámenes, momentos en lo que tiré por primera vez de mis apuntes de forma seria, descubrí que mis apuntes eran un desastre puesto que había grandes lagunas sobre esas cosas tan evidentes que no me molesté en anotar. Pero aprendí. Aprendí que todas las cosas que en un momento dado en caliente parecen evidentes pueden dejar de serlo cuatro meses después.

Mis apuntes del segundo cuatrimestres fueron muy distintos. Eran muchos más gruesos ya que al igual que hacía la chica (que se ve que aprendió esta lección antes que yo) yo anotaba toda y cada una de las cosas que decía el profesor. Si éste decía algo evidente o que yo no consideraba importante en ese momento, lo anotaba. Que dos minutos después lo repetía, pues yo lo volvía a anotar como si fuera tonto.

Sin duda, esta decisión de considerarme tonto a la hora de coger apuntes me vino de gran ayuda el resto de años en la facultad.

Hoy en día, procuro extender esta práctica a mi profesión, tomando nota de cualquier chorrada que se diga en las reuniones o comentando los métodos que programo sabiendo con certeza que debido a las líneas que tiro a lo largo de una semana que en un mes no recordaré ni haber hecho tales métodos o clases. Y si yo no recordaré si tan si quiera haber programado esos métodos... ¿cómo voy a recordar qué hacen? Y peor aún, ¿cómo sabrá alguien que ni si quiera ha escrito esos métodos qué hace?

La respuesta es bien fácil: documentando el método. ¡Pero si es un método muy intuitivo que no necesita documentación! Se comenta. ¡Pero si está claro lo que hace! Se comenta. ¡Pero si hasta un tonto sabría lo que hace con sólo leer el nombre del método! Se comenta, que para tonto yo.

Así que para concluir: señores y señoritas, si no se consideran suficiente tontos como yo como para comentar lo evidente por favor, piensen en mí que sí lo soy y comenten todo.

domingo, 22 de junio de 2008

No reinventemos la rueda

A lo largo de mi corta carrera profesional, y sobre todo en los primeros años, me las he visto con frameworks caseros, he leído ficheros XML utilizando Dom, he subido ficheros al servidor extrayéndolos directamente de la Request y hecho muchas otras barbaridades. En algunos casos la necesidad de utilizar soluciones "caseras" era una imposición de la empresa, reacia a utilizar librerías ajenas a ella, aunque en la mayoría de los casos fue por desconocimiento.

Hoy en día es sabido por todos que Internet podemos llegar a encontrar cientos de frameworks y librerías de código abierto que pueden llegar a facilitarnos la vida de una forma asombrosa. Aun así, todavía podemos encontrar desarrolladores reacios a utilizar todas estas librerías testeadas y utilizadas en algunos casos cientos de miles de desarrolladores. De Struts, por ejemplo, he oído decir cosas como que "es un framework que está de moda y la gente lo utiliza para ser guay", "limita mucho" o "a mí me gusta hacer las cosas a mi modo, si todo te lo hace un framework no es divertido". No hace falta aclarar en estos casos las frases salieron de personas que nunca han trabajado con este framework o similar, que sólo saben desarrollar código espagueti o lo divertido que resulta trabajar con un framework casero.

¿Qué le lleva a una persona a tratar de implementar la solución a un problema que ya está resuelto? Si es desconocimiento, creo que queda excusado pero si has oído hablar de alguna herramienta que soluciona tu problema, ¿por qué no utilizarla? Puede haber algunos casos en los que alguien opte por su solución casera frente a una profesional considerando que tiene capacidad y tiempo para implementar dicha solución (insensato) pero creo que en la mayoría de los casos es debido al miedo. Este miedo al desconocimiento, que admito haber sentido alguna vez, hace buscar una y mil excusas para no utilizar estas librerías que tanto nos pueden facilitar la existencia: es un framework que nos limita mucho, sólo se utiliza porque está de moda, no se ajusta a mi problema, no tengo tiempo para documentarme, no me fio del código de otra persona, me limita mi "liberta creativa" y muchas más. ¡Ay! Cuándo daño hace tanta libertad creativa suelta.

Optar por una solución casera frente a una profesional si duda nos llevará a un error que nos hará perder tiempo y nos causará problemas hasta el fin de la existencia de nuestro código y a todo el que lo herede. Y el primer error que se comete se encuentra en la primera palabra de la frase anterior: optar.

Y es que, escoger una solución profesional frente a una casera no debe ser una opción sino un imperativo. ¿O es que alguien pretende realizar una aplicación Web 2.0 de forma casera sin las herramientas, librerías y framework de los que disponemos?. Sin duda imposible.

Cuando se nos plantea un problema no debemos preguntarnos cómo lo vamos a resolver, sino cuales de las soluciones ya implementas nos gusta más o se ajusta más a nuestra necesidad.