<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	xmlns:media="http://search.yahoo.com/mrss/"
	>

<channel>
	<title>Bits y Bytes &#187; NetBeans</title>
	<atom:link href="http://ivanator.wordpress.com/tag/netbeans/feed/" rel="self" type="application/rss+xml" />
	<link>http://ivanator.wordpress.com</link>
	<description>Desarrollo de software, agilismo y Java</description>
	<lastBuildDate>Mon, 02 Nov 2009 23:07:26 +0000</lastBuildDate>
	<generator>http://wordpress.com/</generator>
	<language>es</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<cloud domain='ivanator.wordpress.com' port='80' path='/?rsscloud=notify' registerProcedure='' protocol='http-post' />
<image>
		<url>http://www.gravatar.com/blavatar/55516fa87678300b6cc434d14988dbba?s=96&#038;d=http://s.wordpress.com/i/buttonw-com.png</url>
		<title>Bits y Bytes &#187; NetBeans</title>
		<link>http://ivanator.wordpress.com</link>
	</image>
			<item>
		<title>Métricas de calidad con NetBeans y Hudson</title>
		<link>http://ivanator.wordpress.com/2009/03/05/metricas-de-calidad-con-netbeans-y-hudson/</link>
		<comments>http://ivanator.wordpress.com/2009/03/05/metricas-de-calidad-con-netbeans-y-hudson/#comments</comments>
		<pubDate>Thu, 05 Mar 2009 18:21:41 +0000</pubDate>
		<dc:creator>Iván Párraga García</dc:creator>
				<category><![CDATA[Uncategorized]]></category>
		<category><![CDATA[Agile]]></category>
		<category><![CDATA[Ant]]></category>
		<category><![CDATA[calidad]]></category>
		<category><![CDATA[Hudson]]></category>
		<category><![CDATA[Java]]></category>
		<category><![CDATA[NetBeans]]></category>

		<guid isPermaLink="false">http://ivanator.wordpress.com/2009/03/05/metricas-de-calidad-con-netbeans-y-hudson/</guid>
		<description><![CDATA[



Este artículo como PDF.
Introducción
En un post anterior expliqué  cómo montar un entorno de integración continua básico, que luego completamos para que también se convirtiera en un verdadero  entorno TDD con tests y métricas de cobertura. El siguiente paso para tener un entorno totalmente ágil consiste en añadir métricas de calidad de código para [...]<img alt="" border="0" src="http://stats.wordpress.com/b.gif?host=ivanator.wordpress.com&blog=6092394&post=297&subd=ivanator&ref=&feed=1" />]]></description>
			<content:encoded><![CDATA[<div class='snap_preview'><br /><div>
<div>
<p style="margin-bottom:.5cm;font-weight:normal;text-align:left;">
</div>
<p class="western">Este artículo como <a href="http://ivanator.files.wordpress.com/2009/03/metricas-de-calidad-con-netbeans-y-hudson.pdf">PDF</a>.</p>
<h1 class="western">Introducción</h1>
<p class="western" style="font-style:normal;font-weight:normal;text-align:justify;">En un post anterior expliqué  <a href="http://ivanator.wordpress.com/2009/01/12/montando-un-entorno-integracion-continua-hudson-ant-svn-netbeans/">cómo montar un entorno de integración continua básico</a>, que luego completamos para que también se convirtiera en un verdadero  <a href="http://ivanator.wordpress.com/2009/02/07/cobertura-con-netbeans-y-hudson/">entorno TDD</a> con tests y métricas de cobertura. El siguiente paso para tener un entorno totalmente ágil consiste en añadir métricas de calidad de código para favorecer la mejora continua del trabajo hecho por todo el equipo y para que los gestores puedan tener una perspectiva algo más “objetiva” del nivel de calidad. Continuaré en este tutorial sobre el mismo ejemplo, así que en numerosos puntos haré referencias a los artículos previos para no tener que repetirme más de lo necesario.</p>
<p class="western" style="font-style:normal;font-weight:normal;text-align:justify;">Nótese que he puesto la palabra  <em>objetiva</em> entre comillas. En este artículo vamos a ver una serie de herramientas que detectarán malas prácticas, bugs potenciales o puntos mejorables y generarán informes al respecto. El desarrollo de software es algo muy complejo y la medida de su calidad real no es automatizable (si así fuera, también lo sería su desarrollo y aún no hemos llegado a ese punto), ni hay un convenio universal de en qué consiste, con lo cual hay que saber leer e interpretar el resultado que producen estas herramientas. Si están avisando de que nuestro código tiene problemas, muy probablemente tengan razón y deberemos poner medidas correctoras. Lo contrario, sin embargo, no implica que nuestro código sea de alta calidad, sino que no se han encontrado aquellas cosas que se sabe que producen problemas. No se podrá automatizar, por ejemplo, cosas como la detección de uso inadecuado de patrones de diseño o la escalabilidad de la plataforma, que podríamos estar de acuerdo que son medidas de calidad. Simplificando mucho:  <em>se puede detectar que el código de un proyecto tiene poca calidad de manera automática, pero no se puede decir que para aquellos para los que no se ha detectado, el código sea de alta calidad</em>.</p>
<p class="western" style="font-style:normal;font-weight:normal;text-align:justify;">Hechas estas matizaciones, en este artículo vamos a explicar cómo integrar las siguientes herramientas en  <a href="http://www.netbeans.org/">NetBeans</a> y  <a href="https://hudson.dev.java.net/">Hudson</a> usando  <a href="http://ant.apache.org/">Ant</a>:</p>
<ul>
<li>
<p class="western" style="text-align:justify;"><a href="http://checkstyle.sourceforge.net/"><span style="font-style:normal;"><span style="font-weight:normal;">Checkstyle</span></span></a> <span style="font-style:normal;"><span style="font-weight:normal;">: genera informes sobre el grado de seguimiento del código a los estándares de codificación establecidos.</span></span></p>
</li>
<li>
<p class="western" style="text-align:justify;"><a href="http://www.redhillconsulting.com.au/products/simian/"><span style="font-style:normal;"><span style="font-weight:normal;">Simian</span></span></a> <span style="font-style:normal;"><span style="font-weight:normal;">: detector de copia de bloques de código.</span></span></p>
</li>
<li>
<p class="western" style="text-align:justify;"><a href="http://pmd.sourceforge.net/"><span style="font-style:normal;"><span style="font-weight:normal;">PMD</span></span></a> <span style="font-style:normal;"><span style="font-weight:normal;">: analizador de código estático en busca de posibles bugs y malas prácticas.</span></span></p>
</li>
<li>
<p class="western" style="text-align:justify;"><a href="http://findbugs.sourceforge.net/"><span style="font-style:normal;"><span style="font-weight:normal;">FindBugs</span></span></a> <span style="font-style:normal;"><span style="font-weight:normal;">: similar al anterior.</span></span></p>
</li>
</ul>
<p class="western" style="font-style:normal;font-weight:normal;text-align:justify;">El procedimiento de instalación de las herramientas va a ser análogo para cada una de ellas:</p>
<ol>
<li>
<p class="western" style="font-style:normal;font-weight:normal;text-align:justify;">Instalación del plugin correspondiente en el NetBeans (si lo hubiere) y configuración del mismo.</p>
</li>
<li>
<p class="western" style="font-style:normal;font-weight:normal;text-align:justify;">Instalación de las bibliotecas de la herramienta en la máquina de desarrollo.</p>
</li>
<li>
<p class="western" style="font-style:normal;font-weight:normal;text-align:justify;">Modificación del script del Ant para poder lanzar el análisis sin la ayuda del IDE.</p>
</li>
<li>
<p class="western" style="font-style:normal;font-weight:normal;text-align:justify;">Instalación del plugin correspondiente en el Hudson.</p>
</li>
<li>
<p class="western" style="font-style:normal;font-weight:normal;text-align:justify;">Modificación del  <em>job</em> para incluir la generación y publicación de informes correspondientes.</p>
</li>
</ol>
<h1 class="western">Checkstyle</h1>
<p class="western">Cuando hablo de estilo de codificación (o convenciones de código) me refiero a cosas como el nivel de indentación, utilizar espacios o tabuladores, añadir comentarios, cómo organizar las llaves de los bloques de código, el tamaño máximo de la línea, etcétera.</p>
<p class="western">Parece un hecho ampliamente aceptado que la uniformidad de estilo a la hora de codificar facilita el grado de cohesión del equipo de desarrollo, la mantenibilidad de la base de código y en definitiva la productividad. Si todo el equipo trabaja con un mismo estilo, puede entender la estructura del programa más fácilmente de un vistazo.</p>
<p class="western">En el mundo Java, Sun proporcionó unas guías de estilo que han sido ampliamente adoptadas por la comunidad. En  <a href="http://java.sun.com/docs/codeconv/">este documento</a> las podéis encontrar junto con los motivos que hay detrás de cada convención.</p>
<p class="western"><a href="http://checkstyle.sourceforge.net/">Checkstyle</a> es una herramienta que genera informes del nivel de seguimiento de estas convenciones. Seguir el estilo al pie de la letra puede resultar algo duro en ocasiones, así que también es posible rebajar el nivel de exigencia decidiendo para qué convenciones se quiere generar una alarma y cuáles se puede ignorar o incluso para definir nuestras propias convenciones.</p>
<p class="western">Checkstyle lo tenemos disponible en las tres vertientes: plugin para el NetBeans, como  <em>task</em> <span style="font-style:normal;">del Ant y como plugin del Hudson.</span></p>
<h2 class="western" style="font-style:normal;">Plugin para NetBeans del Checkstyle</h2>
<p class="western"><span style="font-style:normal;">El plugin no forma parte de los repositorios estándar, así que el primer paso para instalarlo será</span> <span style="font-style:normal;">añadir el repositorio correspondiente. Para ello iremos a:</span> <em>Tools -&gt; Plugins -&gt; Settings -&gt; Add</em> <span style="font-style:normal;">y añadiremos la siguiente</span> <em>url</em> <span style="font-style:normal;">:</span> <a href="http://www.sickboy.cz/checkstyle/autoupdate/autoupdate.xml"><span style="font-style:normal;">http://www.sickboy.cz/checkstyle/autoupdate/autoupdate.xml</span></a> <span style="font-style:normal;">tal y como se muestra en las siguientes imágenes.</span></p>
<p class="western" style="font-style:normal;"><img class="alignnone" style="border:0 none;text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/netbeans-repositorio-checkstyle.jpg?w=500&#038;h=179" border="0" alt="" width="500" height="179" /></p>
<p class="western" style="font-style:normal;"><img class="alignnone" style="border:0 none;text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/netbeans-repositorios.jpg?w=500&#038;h=331" border="0" alt="" width="500" height="331" /></p>
<p class="western"><span style="font-style:normal;">A continuación hacemos click en la pestaña de</span> <em>Available Plugins</em> <span style="font-style:normal;">y seleccionamos</span> <em>Checkstyle Beans Library</em> <span style="font-style:normal;">y</span> <em>Checkstyle Beans Plugin</em> <span style="font-style:normal;">y los instalamos (seguimos los pasos indicados por el asistente y cuando nos pregunte si queremos instalar plugins no firmados le decimos que sí)</span> <em>.</em> <span style="font-style:normal;">Ahora sólo resta reiniciar el NetBeans para que cargue el plugin.</span></p>
<p class="western" style="font-style:normal;">
<p class="western" style="font-style:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/netbeans-plugins-disponibles.jpg?w=500&#038;h=330" border="0" alt="" width="500" height="330" /></p>
<p class="western" style="font-style:normal;">Ahora cada vez que alguna línea de código no conforme con respecto a las convenciones de Sun, aparecerá una etiqueta a modo de aviso. Poniendo el ratón encima de la etiqueta, mostrará cuál es la convención que no se está siguiendo.</p>
<p class="western" style="font-style:normal;">
<p class="western" style="font-style:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/netbeans-editor-checkstyle.jpg?w=500&#038;h=294" border="0" alt="" width="500" height="294" /></p>
<p class="western"><span style="font-style:normal;">Como decía, se pueden utilizar plantillas con convenciones personalizadas. Para ello deberemos editar la configuración por defecto que se encuentra en:</span> <em>Tools -&gt; Options -&gt; Miscellaneous -&gt; Checkstyle.</em></p>
<p class="western">
<p class="western"><img class="alignnone" style="border:0 none;text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/netbeans-opciones-checkstyle.jpg?w=500&#038;h=370" border="0" alt="" width="500" height="370" /></p>
<h2 class="western">Incluir Checkstyle en el script de Ant del NetBeans</h2>
<p class="western">Como ya he explicado en otros artículos, NetBeans realiza todas sus tareas mediante scripts Ant que autogenera y el  <em>build.xml</em> <span style="font-style:normal;">que hay en la raíz del proyecto es extensible para que los desarrolladores podamos introducir nuestros propios</span> <em>targets</em> <span style="font-style:normal;">.</span></p>
<p class="western"><span style="font-style:normal;">Lo primero que tenemos que hacer es</span> <a href="http://sourceforge.net/project/showfiles.php?group_id=29721"><span style="font-style:normal;">descargarnos la distribución del Checkstyle</span></a> <span style="font-style:normal;">(en el momento de escribir este artículo la versión 5 sigue siendo</span> <em>beta</em> <span style="font-style:normal;">, así que yo me decanto por la versión 4.4) que, entre otras cosas, aporta los</span> <em>tasks</em> <span style="font-style:normal;">necesarios para invocarlo desde Ant. A continuación habremos de descomprimir el paquete en algún lugar de la máquina de desarrollo. Yo, siguiendo mi propio estándar, lo situaré en el directorio</span> <em>/srv</em> <span style="font-style:normal;">y luego lo enlazaré con un link simbólico para no tener que</span> <em>hardcodear</em> <span style="font-style:normal;">la ruta hacia una versión concreta en ningún fichero de configuración o script que haga uso del mismo. También crearé un enlace simbólico al</span> <em>jar</em> <span style="font-style:normal;">que contiene la implementación del</span> <em>task</em> <span style="font-style:normal;">por el mismo motivo.</span></p>
<pre class="western" style="font-style:normal;">root@hargon:/srv# tar -zxf checkstyle-4.4.tar.gz
<span lang="zxx">root@hargon:/srv# ln -s checkstyle-4.4 checkstyle</span>
<span lang="zxx">root@hargon:/srv# ls -alh checkstyle</span>
<span lang="zxx">lrwxrwxrwx 1 root root 14 2009-02-25 16:24 checkstyle -&gt; checkstyle-4.4</span>
<span lang="zxx">root@hargon:/srv# rm checkstyle-4.4.tar.gz</span>
<span lang="zxx">root@hargon:/srv# cd checkstyle</span>
<span lang="zxx">root@hargon:/srv/checkstyle# ln -s checkstyle-all-4.4.jar checkstyle-all .jar</span></pre>
<p class="western"><span style="font-style:normal;">El siguiente paso consiste en modificar el</span> <em>build.xml</em> <span style="font-style:normal;">. Lo primero que vamos a necesitar es importar el</span> <em>task</em> <span style="font-style:normal;">tal y como sigue:</span></p>
<pre class="western" style="font-style:normal;">&lt;taskdef
<span lang="zxx">    resource="checkstyletask.properties"</span>
<span lang="zxx">    </span><span lang="zxx">classpath="${checkstyle.dir}/checkstyle-all.jar"/&gt;</span></pre>
<p class="western"><span style="font-style:normal;">Puesto que el valor de</span> <em>checktyle.dir</em> <span style="font-style:normal;">puede ser diferente en la máquina de diferentes desarrolladores o en la máquina que contiene en el servidor de integración, esta propiedad la definimos en el fichero de propiedades privadas del proyecto (que recordemos no forma parte del repositorio de código):</span> <em>nbproject/private/private.properties.</em></p>
<pre class="western" style="font-style:normal;">checkstyle.dir=/srv/checkstyle</pre>
<p class="western" style="font-style:normal;">Como yo estoy partiendo del mismo ejemplo del  <a href="http://ivanator.wordpress.com/2009/02/07/cobertura-con-netbeans-y-hudson/">artículo previo</a> donde explicaba cómo integrar Cobertura, yo ya tengo definido un  <em>import</em> en el  <em>build.xml</em> del fichero de propiedades privadas, si no lo tuvierais acordaros de añadirlo.</p>
<pre class="western">&lt;property file="nbproject/private/private.properties"/&gt;</pre>
<p class="western">A continuación definimos el  <em>target</em> <span style="font-style:normal;">que construirá los informes en el</span> <em>build.xml</em> <span style="font-style:normal;">:</span></p>
<pre class="western" style="font-style:normal;">&lt;target name="checkstyle-report"&gt;

<span lang="zxx">    </span><span lang="zxx">&lt;property</span><span lang="zxx"> </span><span lang="zxx">file="nbproject/project.properties"/&gt;</span>

<span lang="zxx">    </span><span lang="zxx">&lt;mkdir dir="${checkstyle.report.dir}"/&gt;</span>

<span lang="zxx">    </span><span lang="zxx">&lt;checkstyle failOnViolation="false"</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">config="${checkstyle.dir}/sun_checks.xml"&gt;</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">&lt;fileset dir="${src.dir}" includes="**/*.java"/&gt;</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">&lt;formatter type="xml"</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">toFile="${checkstyle.report.dir}/checks.xml"/&gt;</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">&lt;formatter type="plain"</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">toFile="${checkstyle.report.dir}/checks.txt"/&gt;</span>
<span lang="zxx">    </span><span lang="zxx">&lt;/checkstyle&gt;</span>
<span lang="zxx">&lt;/target&gt;</span></pre>
<p class="western"><span style="font-style:normal;">El</span> <em>target</em> <span style="font-style:normal;">es bastante fácil de entender. En primer lugar construimos el directorio donde se generarán los diferentes informes. A continuación lanzamos el</span> <em>task</em> <span style="font-style:normal;">del Checkstyle indicándole que siga procesando el</span> <em>build.xml</em> <span style="font-style:normal;">aunque se encuentren fallos de formato (</span> <em>failOnViolation=”false”</em> <span style="font-style:normal;">) y que utilice las convenciones definidas en el fichero</span> <em>sun_checks.xml</em> <span style="font-style:normal;">que forma parte de la distribución estándar de Checkstyle. Este fichero es donde se describen, en un formato que escapa al ámbito de este artículo, cuáles son las convenciones que caso de no seguirse generan un aviso. Obviamente podemos editar o sustituir este fichero por otro adaptado a nuestras necesidades, pero si dejamos éste, forzamos las convenciones de Sun que, como ya había indicado previamente, se describen en</span> <a href="http://java.sun.com/docs/codeconv/"><span style="font-style:normal;">este documento</span></a> <span style="font-style:normal;">.</span></p>
<p class="western" style="font-style:normal;">Finalmente generamos los informes en dos formatos: en texto plano y en XML. El primero es más sencillo de leer por humanos pero el segundo es el adecuado para integrarlo con Hudson.</p>
<p class="western"><span style="font-style:normal;">En este</span> <em>target</em> <span style="font-style:normal;">presuponemos inicializadas las propiedades</span> <em>checkstyle.dir</em> <span style="font-style:normal;">,</span> <em>src.dir</em> <span style="font-style:normal;">y</span> <em>checstyle.report.dir</em> <span style="font-style:normal;">. La primera ya la hemos configurado en el punto anterior, la segunda forma parte de las propiedades creadas por NetBeans y la última la tenemos que definir.</span> <span style="font-style:normal;"><span style="font-weight:normal;">Las añadiremos como propiedades del proyecto en el</span></span> <em><span style="font-weight:normal;">nbproject/project.properties</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">:</span></span></p>
<pre class="western" style="font-style:normal;font-weight:normal;">reports.dir=${build.dir}/reports
<span lang="zxx">checkstyle.report.dir=${reports.dir}/checkstyle-report</span></pre>
<h2 class="western">Integrando Checkstyle con Hudson</h2>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">Para generar los informes en Hudson nos aprovecharemos del</span></span> <em><span style="font-weight:normal;">target</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">que acabamos de definir en el</span></span> <em><span style="font-weight:normal;">build.xml</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">del NetBeans, por tanto el proceso va a ser sencillo:</span></span></p>
<ol>
<li>
<p class="western" style="font-style:normal;font-weight:normal;">añadir el plugin del Checkstyle al Hudson,</p>
</li>
<li>
<p class="western" style="font-style:normal;font-weight:normal;">instalar las bibliotecas del Checkstyle en la máquina donde reside el Hudson,</p>
</li>
<li>
<p class="western" style="font-style:normal;font-weight:normal;">añadir las propiedades privadas a mano (recordad que este fichero no forma parte del Subversion),</p>
</li>
<li>
<p class="western" style="font-style:normal;font-weight:normal;">añadir el nuevo target en la construcción del proyecto,</p>
</li>
<li>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">configurar el</span></span> <em><span style="font-weight:normal;">job</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">para que utilice el</span></span> <em><span style="font-weight:normal;">target</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">.</span></span></p>
</li>
</ol>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">El proceso para instalar el plugin, ya lo hemos visto en el post donde explicaba</span></span> <a href="http://ivanator.wordpress.com/2009/02/07/cobertura-con-netbeans-y-hudson/"><span style="font-style:normal;"><span style="font-weight:normal;">cómo añadir el plugin de Cobertura</span></span></a> <span style="font-style:normal;"><span style="font-weight:normal;">. Básicamente, seleccionarlo de la lista de plugins disponibles, instalarlo con el asistente y reiniciar el Hudson.</span></span></p>
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/hudson-instalar-checkstyle-plugin.jpg?w=500&#038;h=383" border="0" alt="" width="500" height="383" /></p>
<p class="western" style="font-style:normal;font-weight:normal;">Utilizaré el mismo procedimiento y ruta para instalar la distribución del Checkstyle en el servidor donde se ejecuta el Hudson, así que repetiré exactamente los mismos pasos que hice en la máquina de desarrollo.</p>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">A continuación tenemos que modificar las propiedades privadas para añadir la ruta al Checkstyle. Nos situaremos en el directorio del</span></span> <em><span style="font-weight:normal;">job</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">y editaremos (o crearemos) el fichero</span></span> <em><span style="font-weight:normal;">nbproject/private/private.properties</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">para añadir la línea:</span></span></p>
<pre class="western" style="font-style:normal;font-weight:normal;">checkstyle.dir=/srv/checkstyle</pre>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">Finalmente añadimos</span></span> <em><span style="font-weight:normal;">checkstyle-report</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">a la lista de</span></span> <em><span style="font-weight:normal;">tagets</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">a construir y configuramos el job para que lea los informes generados en</span></span> <em><span style="font-weight:normal;">xml</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">y los publique. Adicionalmente podemos configurar opciones avanzadas como los límites a partir de los cuales el</span></span> <em><span style="font-weight:normal;">build</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">debe considerarse inestable o roto.</span></span></p>
<p class="western" style="font-style:normal;font-weight:normal;"><img style="border:0 none;text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/hudson-configurar-checkstyle-en-el-job.jpg?w=500&#038;h=337" border="0" alt="" width="500" height="337" /></p>
<p class="western" style="font-style:normal;font-weight:normal;">Ahora ya sólo resta ver los informes que genera el Hudson:</p>
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/hudson-informes-checkstyle.jpg?w=500&#038;h=437" border="0" alt="" width="500" height="437" /></p>
<h1 class="western">Simian</h1>
<p class="western">Alguna vez oí, aunque no recuerdo la fuente, que en informática la duplicación es el diablo y yo no puedo estar más de acuerdo. En ciertas ocasiones requerimientos no funcionales, como por ejemplo eficiencia, puede requerir duplicar información (denormalización de tablas en bases de datos, materialización de cálculos de agregación de información, etc), pero ello debe ser consecuencia de una decisión de diseño meditada y no de la dejadez. La replicación de código, en general, es menos justificable e introduce un gran coste en mantenibilidad y calidad: modificaciones en cualquiera de las copias debería llevar aparejadas actualizaciones en el resto, lo cual se hace muy tedioso y complicado de implementar.</p>
<p class="western">Por todo lo anterior, si tenemos repetido un bloque de código de un tamaño razonable, probablemente ello está descubriendo un mal diseño y ese código debería encapsularse de alguna manera para que exista en un único punto.</p>
<p class="western"><a href="http://www.redhillconsulting.com.au/products/simian/">Simian</a> es la herramienta que nos va a permitir encontrar replicación de bloques de código (o de texto en general) en una gran cantidad de lenguajes de programación y formatos de texto. Hasta donde yo sé, no existe como plugin para NetBeans (aunque sí para Eclipse e IntelliJ). Sí que existe el  <em>task</em> <span style="font-style:normal;">para Ant y como plugin para el Hudson y por tanto veremos estas configuraciones.</span></p>
<p class="western" style="font-style:normal;">Como el procedimiento para incluir nuevas herramientas de análisis y métricas va a ser siempre muy similar a lo que he explicado en el caso del Checkstyle, no explicaré el proceso de una manera tan detallada (entre otras cosas porque no aporta nada y sólo hace crecer el documento).</p>
<h2 class="western" style="font-style:normal;">Incluir Simian en el script de Ant del NetBeans</h2>
<p class="western"><span style="font-style:normal;">Lo primero es obtener y configurar la herramienta en el entorno de desarrollo, para ello lo</span> <a href="http://www.redhillconsulting.com.au/products/simian/"><span style="font-style:normal;">descargamos</span></a> <span style="font-style:normal;">, lo descomprimimos en</span> <em>/srv</em> <span style="font-style:normal;">/ (id con cuidado porque el paquete no contiene un directorio raíz) y creamos el enlace simbólico correspondiente.</span></p>
<pre class="western" style="font-style:normal;">root@hargon:/srv# mkdir simian-2.2.24
<span lang="zxx">root@hargon:/srv# cd simian-2.2.24</span>
<span lang="zxx">root@hargon:/srv/simian-2.2.24# mv ../simian-2.2.24.tar.gz .</span>
<span lang="zxx">root@hargon:/srv/simian-2.2.24# tar -zxf simian-2.2.24.tar.gz</span>
<span lang="zxx">root@hargon:/srv/simian-2.2.24# rm simian-2.2.24.tar.gz</span>
<span lang="zxx">root@hargon:/srv/simian# cd bin/</span>
<span lang="zxx">root@hargon:/srv/simian/bin# ln -s simian-2.2.24.jar simian.jar</span>
<span lang="zxx">root@hargon:/srv/simian-2.2.24# cd ../..</span>
<span lang="zxx">root@hargon:/srv# ln -s simian-2.2.24 simian</span></pre>
<p class="western"><span style="font-style:normal;">A continuación configuramos el</span> <em>build.xml</em> <span style="font-style:normal;">de manera análoga al Checkstyle:</span></p>
<pre class="western" style="font-style:normal;">&lt;taskdef
<span lang="zxx">    resource="simiantask.properties"</span>
<span lang="zxx">    </span><span lang="zxx">classpath="${simian.dir}/bin/simian.jar"/&gt;</span>

<span lang="zxx">&lt;target name="simian-report"&gt;</span>
<span lang="zxx">    </span><span lang="zxx">&lt;property</span><span lang="zxx"> </span><span lang="zxx">file="nbproject/project.properties"/&gt;</span>

<span lang="zxx">    </span><span lang="zxx">&lt;mkdir dir="${simian.report.dir}" /&gt;</span>

<span lang="zxx">    </span><span lang="zxx">&lt;simian&gt;</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx"><span style="font-style:normal;">&lt;fileset dir="${src.dir}" includes="**/*.java"/&gt;</span></span>

<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">&lt;formatter type="xml"</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">toFile="${simian.report.dir}/simian.xml"/&gt;</span>

<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">&lt;formatter type="plain"</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">toFile="${simian.report.dir}/simian.txt"/&gt;</span>
<span lang="zxx">    </span><span lang="zxx">&lt;/simian&gt;</span>
<span lang="zxx">&lt;/target&gt;</span></pre>
<p class="western"><span style="font-style:normal;">Las propiedades las definiremos allí donde toca: el</span> <em>simian.dir</em> <span style="font-style:normal;">(con valor</span> <em>/srv/simian</em> <span style="font-style:normal;">) en el</span> <em><span style="font-weight:normal;">nbproject/private/private.properties</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">y el</span></span> <em><span style="font-weight:normal;">simian.report.dir</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">(con valor</span></span> <em><span style="font-weight:normal;">${reports.dir}/simian-report</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">) en el</span></span> <em><span style="font-weight:normal;">nbproject/project.properties</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">.</span></span></p>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">El</span></span> <em><span style="font-weight:normal;">task</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">de Simian es muy sencillo. En su configuración simple tal y como tenemos, tan sólo hace falta especificar el directorio que contiene el código a analizar y el formato de los informes (en XML para integrarse con el Hudson y en texto plano para que sea más legible por humanos).</span></span></p>
<h2 class="western">Integrando Simian con Hudson</h2>
<p class="western">Como en el caso del Checkstyle, replicamos la instalación de Simian en la máquina donde se encuentra el Hudson y definimos la propiedad  <em>simian.dir</em> <span style="font-style:normal;">en el</span> <em><span style="font-weight:normal;">nbproject/private/private.properties</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">.</span></span></p>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">El siguiente paso debería consistir en instalar el plugin del Simian para el Hudson, pero cuando lo</span></span> <span style="font-style:normal;"><span style="font-weight:normal;">buscamos en la lista de plugins disponibles nos damos cuenta que no aparece. Afortunadamente existe un plugin llamado</span></span> <em><span style="font-weight:normal;">Violations</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">que integra diferentes herramientas de análisis estático de código (de hecho podríamos utilizar este mismo plugin para otras de las tratadas en este artículo) y que tiene soporte para Simian. Lo instalamos de la manera habitual, reiniciamos el Hudson para que lo cargue y el siguiente paso consiste en configurar el</span></span> <em><span style="font-weight:normal;">job</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">. Como siempre, añadiremos el</span></span> <em><span style="font-weight:normal;">simian-report</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">a la lista de</span></span> <em><span style="font-weight:normal;">targets</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">a construir por el Ant y a continuación configuramos el plugin tal y como se muestra en la ilustración (también podemos definir los límites a partir de los cuales se rompe el</span></span> <em><span style="font-weight:normal;">build</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">).</span></span></p>
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/hudson-configurar-simian-en-el-job.jpg?w=500&#038;h=390" border="0" alt="" width="500" height="390" /></p>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">Tras construir el</span></span> <em><span style="font-weight:normal;">job</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">podremos ver los informes de replicación de código de manera integrada en el Hudson.</span></span></p>
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/hudson-informes-simian-1.jpg?w=500&#038;h=436" border="0" alt="" width="500" height="436" /></p>
<p class="western" style="font-style:normal;font-weight:normal;">
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/hudson-informes-simian-2.jpg?w=500&#038;h=436" border="0" alt="" width="500" height="436" /></p>
<h1 class="western">PMD</h1>
<p class="western" style="font-style:normal;font-weight:normal;">Tanto  <a href="http://pmd.sourceforge.net/">PMD</a> como  <a href="http://findbugs.sourceforge.net/">FindBugs</a>, son herramientas de análisis estático de código (analizan el código y no el programa en ejecución) en busca de estructuras potencialmente peligrosas tales como:</p>
<ul>
<li>
<p class="western" style="font-style:normal;font-weight:normal;">posibles bugs,</p>
</li>
<li>
<p class="western" style="font-style:normal;font-weight:normal;">código “muerto” (variables no accedidas, bloques de ejecución inalcanzables, etc.),</p>
</li>
<li>
<p class="western" style="font-style:normal;font-weight:normal;">código subóptimo,</p>
</li>
<li>
<p class="western" style="font-style:normal;font-weight:normal;">bloques con una estructura poco legible o más complicada de lo necesario.</p>
</li>
</ul>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">Estas herramientas son especialmente útiles integradas en el IDE porque de esta manera el</span></span> <span style="font-style:normal;"><span style="font-weight:normal;">programador puede ir viendo mientras escribe el código las posibles alertas. Obviamente también tienen</span></span> <em><span style="font-weight:normal;">tasks</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">de Ant lo que permite integrarlas fácilmente en Hudson para obtener los informes correspondientes.</span></span></p>
<h2 class="western">Plugin para NetBeans del PMD</h2>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">En este caso el plugin del PMD no está accesible mediante un repositorio, por lo que para instalarlo procederemos a la</span></span> <a href="http://sourceforge.net/project/showfiles.php?group_id=56262&amp;package_id=63621"><span style="font-style:normal;"><span style="font-weight:normal;">descarga del mismo</span></span></a> <span style="font-style:normal;"><span style="font-weight:normal;">y a su instalación local.</span></span></p>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">Una vez tenemos el fichero, deberemos descomprimirlo y nos quedamos con el plugin empaquetado en el fichero</span></span> <em><span style="font-weight:normal;">pmd.nbm</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">que instalaremos a través la pestaña</span></span> <em><span style="font-weight:normal;">Downloaded</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">del gestor de plugins: primero lo añadimos a la lista mediante el botón</span></span> <em><span style="font-weight:normal;">Add Plugins</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">y después le damos a</span></span> <em><span style="font-weight:normal;">Install</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">.</span></span></p>
<p class="western" style="font-style:normal;font-weight:normal;">
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/netbeans-plugins-downloaded-pmd.jpg?w=500&#038;h=326" border="0" alt="" width="500" height="326" /></p>
<p class="western" style="font-style:normal;font-weight:normal;">Aceptamos la licencia, asumimos el riesgo de instalar un plugin “no firmado” y rebotamos el IDE para que se reflejen los cambios.</p>
<p class="western" style="font-style:normal;font-weight:normal;">
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/netbeans-plugin-pmd-warning1.jpg?w=375&#038;h=229" border="0" alt="" width="375" height="229" /></p>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">En este momento el plugin debería estar listo para ser utilizado. Mientras trabajamos PMD monitorizará el código que escribimos y nos pondrá una marca en el editor cuando algo no le guste. Si ponemos el puntero encima de la marca, se nos mostrará un mensaje informativo del problema en forma de</span></span> <em><span style="font-weight:normal;">tooltip</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">.</span></span></p>
<p class="western" style="font-style:normal;font-weight:normal;">
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/netbeans-plugin-pmd-en-accion.jpg" border="0" alt="" width="75%" /></p>
<p class="western" style="font-style:normal;font-weight:normal;">
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">Si queremos obtener un informe de todos los posibles problemas en una carpeta de código, paquete o clase, sobre la ventana</span></span> <em><span style="font-weight:normal;">Projects</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">hacemos</span></span> <em><span style="font-weight:normal;">click derecho en el elemento correspondiente -&gt; Tools -&gt; Run PMD.</span></em></p>
<p class="western" style="font-weight:normal;">
<p class="western" style="font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/netbeans-plugin-pmd-generar-informe.jpg" border="0" alt="" width="455" height="360" /></p>
<p class="western" style="font-style:normal;font-weight:normal;">Ello abrirá una nueva pestaña de informe con todas los avisos disponibles en forma de tabla (permitiendo ordenación por las diferentes columnas). Haciendo doble click sobre cualquiera de las advertencias, el editor se desplazará a la línea de código correspondiente.</p>
<p class="western" style="font-style:normal;font-weight:normal;">
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/netbeans-plugin-pmd-informe.jpg" border="0" alt="" width="500" height="170" /></p>
<p class="western" style="font-style:normal;"><span style="font-weight:normal;">PMD puede ser muy estricto y es posible que nos interese personalizar el tipo de alertas generadas y, quizá, desactivar algunas. Hay que encontrar un compromiso entre lo razonable y la calidad. Por ejemplo, algunas de las estructuras autogeneradas por el IDE (como el</span> <em><span style="font-weight:normal;">equals</span></em> <span style="font-weight:normal;">mostrado en el ejemplo unas líneas más arriba) producen advertencias que podríamos ignorar con tranquilidad ya que NetBeans</span> <em><span style="font-weight:normal;">sabe lo que hace</span></em> <span style="font-weight:normal;">. Para modificar la configuración del plugin y el tipo de reglas que aplicar, nos vamos al menú de opciones:</span> <em><span style="font-weight:normal;">Tools -&gt; Options -&gt; Miscellaneous.</span></em></p>
<p class="western" style="font-weight:normal;">
<p class="western" style="font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/netbeans-plugin-pmd-configuracion.jpg" border="0" alt="" width="500" height="257" /></p>
<p class="western" style="font-style:normal;"><span style="font-weight:normal;">Las reglas se gestionan mediante</span> <em><span style="font-weight:normal;">rulesets</span></em> <span style="font-weight:normal;">temáticos. Es decir un</span> <em><span style="font-weight:normal;">ruleset</span></em> <span style="font-weight:normal;">puede contener todas aquellas reglas (</span> <em><span style="font-weight:normal;">rules)</span></em> <span style="font-weight:normal;">que tengan que ver, por ejemplo, con código muerto. Mediante el menú de</span> <em><span style="font-weight:normal;">rulesets</span></em> <span style="font-weight:normal;">podemos incluir o no las incluidas con PMD y añadir otras propias generadas por nosotros o por terceros. En el menú</span> <em><span style="font-weight:normal;">Manage rules</span></em> <span style="font-weight:normal;">se nos permite la activación o desactivación de reglas individuales.</span></p>
<p class="western" style="font-style:normal;font-weight:normal;">
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/netbeans-plugin-pmd-rule-editor.jpg" border="0" alt="" width="75%" /></p>
<h2 class="western">Incluir PMD en el script del Ant de NetBeans</h2>
<p class="western"><span style="font-weight:normal;">De manera análoga a las herramientas anteriores,</span> <a href="http://downloads.sourceforge.net/pmd/pmd-bin-4.2.5.zip?use_mirror=dfn"><span style="font-weight:normal;">descargamos</span></a> <span style="font-weight:normal;">e instalamos la herramienta en</span> <em><span style="font-weight:normal;">/srv</span></em> <span style="font-weight:normal;">.</span></p>
<pre class="western" style="font-weight:normal;">root@hargon:/srv# unzip -q pmd-bin-4.2.5.zip
<span lang="zxx">root@hargon:/srv# ln -s pmd-4.2.5/ pmd</span>
<span lang="zxx">root@hargon:/srv# rm pmd-bin-4.2.5.zip</span>
<span lang="zxx">root@hargon:/srv# cd pmd/lib/</span>
<span lang="zxx">root@hargon:/srv/pmd/lib# ln -s pmd-4.2.5.jar pmd.jar</span></pre>
<p class="western">A continuación modificaros el  <em>build.xml</em> <span style="font-style:normal;">de manera adecuada.</span></p>
<pre class="western" style="font-style:normal;">&lt;taskdef name="pmd"
<span lang="zxx">    classname="net.sourceforge.pmd.ant.PMDTask"</span>
<span lang="zxx">    </span><span lang="zxx">classpath="${pmd.dir}/lib/pmd.jar"/&gt;</span>

<span lang="zxx">&lt;target name="pmd-report"&gt;</span>
<span lang="zxx">    </span><span lang="zxx">&lt;property
</span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">file="nbproject/project.properties"/&gt;</span>
<span lang="zxx"><span style="font-style:normal;">
</span></span><span lang="zxx">    </span><span lang="zxx"><span style="font-style:normal;">&lt;mkdir dir="${pmd.report.dir}" /&gt;</span></span>

<span lang="zxx">    </span><span lang="zxx">&lt;pmd shortFilenames="true"&gt;</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">&lt;ruleset&gt;unusedcode,basic,design,controversial&lt;/ruleset&gt;</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx"><span style="font-style:normal;">&lt;formatter type="xml"
</span></span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx"><span style="font-style:normal;">toFile="${pmd.report.dir}/pmd.xml"</span></span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">linkPrefix="http://pmd.sourceforge.net/xref/" /&gt;</span>

<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">&lt;formatter type="html"
</span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">toFile="${pmd.report.dir}/pmd.html"</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">linkPrefix="http://pmd.sourceforge.net/xref/" /&gt;</span>

<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">&lt;fileset dir="${src.dir}"&gt;</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">&lt;include name="**/*.java"/&gt;</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">&lt;/fileset&gt;</span>
<span lang="zxx">    </span><span lang="zxx">&lt;/pmd&gt;</span>
<span lang="zxx">&lt;/target&gt;</span></pre>
<p class="western"><span style="font-style:normal;">Las propiedades las añadimos al fichero correspondiente: el</span> <em>pmd.dir</em> <span style="font-style:normal;">(con valor</span> <em>/srv/pmd</em> <span style="font-style:normal;">) en el</span> <em><span style="font-weight:normal;">nbproject/private/private.properties</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">y el</span></span> <em><span style="font-weight:normal;">pmd.report.dir</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">(con valor</span></span> <em><span style="font-weight:normal;">${reports.dir}/pmd-report</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">) en el</span></span> <em><span style="font-weight:normal;">nbproject/project.properties</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">.</span></span></p>
<p class="western"><span style="font-style:normal;">El</span> <em>task</em> <span style="font-style:normal;">del PMD</span> <span style="font-style:normal;">es sencillo de utilizar. Primeramente indicamos qué</span> <em>rulesets</em> <span style="font-style:normal;">de los estándar queremos analizar (podríamos también especificar</span> <em>rulesets</em> <span style="font-style:normal;">personalizados) y a continuación indicamos en qué formatos queremos generar los informes. Como es habitual, tendremos los informes en XML que se integran bien con Hudson y en formato HTML para ser leídos por humanos.</span></p>
<p class="western" style="font-style:normal;font-weight:normal;">
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/pmd-html-report2.jpg" border="0" alt="" width="500" height="357" /></p>
<h2 class="western">Integrando PMD con Hudson</h2>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">Al igual que las dos herramientas anteriores, instalamos PMD en el servidor de integración y definimos la propiedad</span></span> <em><span style="font-weight:normal;">pmd.dir</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">en el</span></span> <em><span style="font-weight:normal;">nbproject/private/private.properties</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">.</span></span></p>
<p class="western"><a id="DDE_LINK7" name="DDE_LINK7"></a> <span style="font-style:normal;"><span style="font-weight:normal;">A continuación instalamos el plugin del PMD del Hudson de la manera habitual (no olvidéis rebotar lo) y ya podemos proceder a la configuración del</span></span> <em><span style="font-weight:normal;">job</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">tal y como se muestra en la siguiente captura. Obviamente podemos establecer los límites que por defecto, como casi siempre, están desactivados.</span></span></p>
<p class="western" style="font-style:normal;font-weight:normal;">
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/hudson-configurar-pmd-en-el-job.jpg" border="0" alt="" width="499" height="223" /></p>
<p class="western" style="font-style:normal;font-weight:normal;">Y estos serían los informes generados.</p>
<p class="western" style="font-style:normal;font-weight:normal;">
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/hudson-informes-pmd-1.jpg" border="0" alt="" width="500" height="404" /></p>
<p class="western" style="font-style:normal;font-weight:normal;">
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/hudson-informes-pmd-21.jpg" border="0" alt="" width="500" height="464" /></p>
<h1 class="western">FindBugs</h1>
<p class="western">Sin entrar en demasiados detalles,  <a href="http://findbugs.sourceforge.net/">FindBugs</a> es una herramienta similar a la anterior. En el momento de escribir este tutorial todavía no existe una versión compatible del plugin para NetBeans 6.5, pero los ingenieros de FindBugs  <a href="http://kenai.com/projects/sqe/forums/forum/topics/377-findbugs-plugin-for-netbeans-6-5">están trabajando en ello</a> y me imagino que estará disponible en breve (sí que existen versiones funcionales para otros IDEs).</p>
<h2 class="western">Incluir FindBugs en el plugin de Ant de NetBeans</h2>
<p class="western">Procedemos a  <a href="http://findbugs.sourceforge.net/downloads.html">descargar</a> el paquete para instalarlo de la manera habitual.</p>
<pre class="western">root@hargon:/srv# tar -zxf findbugs-1.3.7.tar.gz
<span lang="zxx">root@hargon:/srv# ln -s findbugs-1.3.7 findbugs</span>
<span lang="zxx">root@hargon:/srv# ls -alh findbugs</span>
<span lang="zxx">lrwxrwxrwx 1 root root 14 2009-03-05 15:55 findbugs -&gt; findbugs-1.3.7</span></pre>
<p class="western">A continuación modificamos el  <em>build.xml</em> <span style="font-style:normal;">para añadir un</span> <em>target</em> <span style="font-style:normal;">que use el</span> <em>task</em> <span style="font-style:normal;">del NetBeans.</span></p>
<pre class="western" style="font-style:normal;">&lt;taskdef name="findbugs"
<span lang="zxx">    </span><span lang="zxx"><span style="font-style:normal;">classname="edu.umd.cs.findbugs.anttask.FindBugsTask"</span></span>
<span lang="zxx">    </span><span lang="zxx">classpath="${findbugs.dir}/lib/findbugs.jar" /&gt;</span>

<span lang="zxx">&lt;target name="findbugs-report" depends="compile"&gt;</span>
<span lang="zxx">
</span><span lang="zxx">    </span><span lang="zxx">&lt;property file="nbproject/project.properties"/&gt;</span>

<span lang="zxx">    </span><span lang="zxx">&lt;mkdir dir="${findbugs.report.dir}"/&gt;</span>

<span lang="zxx">    </span><span lang="zxx">&lt;findbugs home="${findbugs.dir}"
</span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">output="xml"</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">outputFile="${findbugs.report.dir}/findbugs.xml"&gt;</span>
<span lang="zxx">    </span><span lang="zxx">
</span><span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">&lt;sourcePath path="${src.dir}" /&gt;</span>
<span lang="zxx">    </span><span lang="zxx">    </span><span lang="zxx">&lt;class location="${build.classes.dir}" /&gt;</span>

<span lang="zxx">    </span><span lang="zxx">&lt;/findbugs&gt;</span>
<span lang="zxx">&lt;/target&gt;</span></pre>
<p class="western"><span style="font-style:normal;">Como en los casos anteriores, añadimos las propiedades a los ficheros correspondientes: el</span> <em>findbugs.dir</em> <span style="font-style:normal;">(con valor</span> <em>/srv/findbugs</em> <span style="font-style:normal;">) en el</span> <em><span style="font-weight:normal;">nbproject/private/private.properties</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">y el</span></span> <em><span style="font-weight:normal;">findbugs.report.dir</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">(con valor</span></span> <em><span style="font-weight:normal;">${reports.dir}/findbugs-report</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">) en el</span></span> <em><span style="font-weight:normal;">nbproject/project.properties</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">.</span></span></p>
<p class="western"><span style="font-style:normal;">El</span> <em>task</em> <span style="font-style:normal;">del FindBugs requiere no sólo el código fuente sino también las clases compiladas y por ello el</span> <em>target</em> <span style="font-style:normal;">tiene una dependencia del</span> <em>target</em> <span style="font-style:normal;">estándar de NetBeans</span> <em>compile</em> <span style="font-style:normal;">. Las clases compiladas se referencian mediante una propiedad también estándar de NetBeans:</span> <em>build.classes.dir</em> <span style="font-style:normal;">.</span></p>
<p class="western" style="font-style:normal;">El resto de propiedades son las típicas que hemos ido viendo, el directorio de instalación de la herramienta y el directorio de salida de los informes.</p>
<p class="western" style="font-style:normal;">En este caso estamos generando los informes en XML que son los que entiende Hudson, pero también pueden generarse en otros formatos, como en texto plano o en HTML.</p>
<h2 class="western" style="font-style:normal;">Integrando FindBugs con Hudson</h2>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">Como siempre, replicamos la instalación del FindBugs en la máquina donde está instalado el Hudson y definimos la propiedad</span></span> <em><span style="font-weight:normal;">findbugs.dir</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">apuntando al directorio correspondiente en el</span></span> <em><span style="font-weight:normal;">nbproject/private/private.properties</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">.</span></span></p>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">FindBugs también tiene un plugin nativo para Hudson, así que lo instalamos y reiniciamos. A continuación, de manera totalmente análoga al PMD, configuramos el nuevo</span></span> <em><span style="font-weight:normal;">target</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">y las opciones de FindBugs.</span></span></p>
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/hudson-configurar-findbugs-en-el-job.jpg" border="0" alt="" width="75%" /></p>
<p class="western" style="font-style:normal;font-weight:normal;">Finalmente ya podemos ver los informes generados.</p>
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/hudson-informes-findbugs-1.jpg" border="0" alt="" width="500" height="414" /></p>
<p class="western" style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/03/hudson-informes-findbugs-2.jpg" border="0" alt="" width="500" height="458" /></p>
<h1 class="western">Conclusión</h1>
<p class="western" style="font-style:normal;font-weight:normal;">En este artículo hemos aprendido que existen herramientas que nos pueden ayudar a generar métricas sobre la calidad de nuestro código. También hemos enfatizado el hecho que obtener un 100% de puntuación no garantiza un software de calidad.</p>
<p class="western"><span style="font-style:normal;"><span style="font-weight:normal;">Hemos visto que las herramientas suelen venir en tres “sabores”: como plugin para el IDE que dan un feedback inmediato al programador, como</span></span> <em><span style="font-weight:normal;">task</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">del Ant que permiten ejecutarlas en modo consola e integrarlas con otros sistemas y como plugin para Hudson que permite publicar los informes correspondientes de una manera agradable para los humanos.</span></span></p>
<p class="western" style="font-style:normal;font-weight:normal;">Existen muchas más herramientas de calidad. En próximos artículos tocaré algunas mas.</p>
</div>
 Tagged: Agile, Ant, calidad, Hudson, Java, NetBeans <a rel="nofollow" href="http://feeds.wordpress.com/1.0/gocomments/ivanator.wordpress.com/297/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/comments/ivanator.wordpress.com/297/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/godelicious/ivanator.wordpress.com/297/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/delicious/ivanator.wordpress.com/297/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/gostumble/ivanator.wordpress.com/297/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/stumble/ivanator.wordpress.com/297/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/godigg/ivanator.wordpress.com/297/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/digg/ivanator.wordpress.com/297/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/goreddit/ivanator.wordpress.com/297/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/reddit/ivanator.wordpress.com/297/" /></a> <img alt="" border="0" src="http://stats.wordpress.com/b.gif?host=ivanator.wordpress.com&blog=6092394&post=297&subd=ivanator&ref=&feed=1" /></div>]]></content:encoded>
			<wfw:commentRss>http://ivanator.wordpress.com/2009/03/05/metricas-de-calidad-con-netbeans-y-hudson/feed/</wfw:commentRss>
		<slash:comments>5</slash:comments>
	
		<media:content url="" medium="image">
			<media:title type="html">ivanator</media:title>
		</media:content>

		<media:content url="http://ivanator.files.wordpress.com/2009/02/netbeans-repositorio-checkstyle.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/netbeans-repositorios.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/netbeans-plugins-disponibles.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/netbeans-editor-checkstyle.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/netbeans-opciones-checkstyle.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/hudson-instalar-checkstyle-plugin.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/hudson-configurar-checkstyle-en-el-job.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/hudson-informes-checkstyle.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/hudson-configurar-simian-en-el-job.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/hudson-informes-simian-1.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/hudson-informes-simian-2.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/netbeans-plugins-downloaded-pmd.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/netbeans-plugin-pmd-warning1.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/netbeans-plugin-pmd-en-accion.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/netbeans-plugin-pmd-generar-informe.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/netbeans-plugin-pmd-informe.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/netbeans-plugin-pmd-configuracion.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/netbeans-plugin-pmd-rule-editor.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/pmd-html-report2.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/hudson-configurar-pmd-en-el-job.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/hudson-informes-pmd-1.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/hudson-informes-pmd-21.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/hudson-configurar-findbugs-en-el-job.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/hudson-informes-findbugs-1.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/03/hudson-informes-findbugs-2.jpg" medium="image" />
	</item>
		<item>
		<title>Cobertura con NetBeans y Hudson</title>
		<link>http://ivanator.wordpress.com/2009/02/07/cobertura-con-netbeans-y-hudson/</link>
		<comments>http://ivanator.wordpress.com/2009/02/07/cobertura-con-netbeans-y-hudson/#comments</comments>
		<pubDate>Sat, 07 Feb 2009 19:00:41 +0000</pubDate>
		<dc:creator>Iván Párraga García</dc:creator>
				<category><![CDATA[Uncategorized]]></category>
		<category><![CDATA[Agile]]></category>
		<category><![CDATA[calidad]]></category>
		<category><![CDATA[Hudson]]></category>
		<category><![CDATA[Java]]></category>
		<category><![CDATA[NetBeans]]></category>

		<guid isPermaLink="false">http://ivanator.wordpress.com/?p=142</guid>
		<description><![CDATA[
Este artículo como PDF.
Introducción
Una vez tenemos montado un  entorno TDD, y si efectivamente estamos trabajando con esta orientación, nuestro proyecto empezará a acumular tests unitarios con  JUnit o cualquier otro framework lo que permitirá a nuestro servidor de integración (Hudson) utilizarlos como  tests de regresión.
Si estamos siendo estrictos en la aplicación de [...]<img alt="" border="0" src="http://stats.wordpress.com/b.gif?host=ivanator.wordpress.com&blog=6092394&post=142&subd=ivanator&ref=&feed=1" />]]></description>
			<content:encoded><![CDATA[<div class='snap_preview'><br /><div>
<p class="western" style="text-align:justify;">Este artículo como <a href="http://ivanator.files.wordpress.com/2009/02/cobertura-con-netbeans-y-hudson.pdf">PDF</a>.</p>
<h1 class="western" style="text-align:justify;">Introducción</h1>
<p class="western" style="text-align:justify;">Una vez tenemos montado un  <a href="http://ivanator.wordpress.com/2009/01/12/montando-un-entorno-integracion-continua-hudson-ant-svn-netbeans/">entorno TDD,</a> y si efectivamente estamos trabajando con esta orientación, nuestro proyecto empezará a acumular tests unitarios con  <a href="http://www.junit.org/">JUnit</a> o cualquier otro framework lo que permitirá a nuestro servidor de integración (Hudson) utilizarlos como  <a href="http://en.wikipedia.org/wiki/Regression_testing">tests de regresión</a>.</p>
<p class="western">Si estamos siendo estrictos en la aplicación de una  <a href="http://es.wikipedia.org/wiki/Desarrollo_guiado_por_pruebas">metodología TDD</a>, en principio el grado de cobertura del código (el porcentaje de líneas de código que son evaluadas por un test) debería estar cerca del 100% por aquello del  <a href="http://www.ibm.com/developerworks/java/library/j-xp042203/">“ escribe primero el test y luego el código que lo supera”</a>. Sin embargo bien porque seamos principiantes aplicando la metodología o bien porque escribamos los tests como parte del proceso de calidad en una metodología diferente, puede resultar muy complicado calcular el grado de cobertura mediante técnicas manuales. Incluso en un entorno TDD maduro, la propia complejidad del software puede complicar extraer esta información. Es por ello que se han desarrollado una serie de frameworks y herramientas que permiten automatizar este proceso.</p>
<p class="western">Como decía, no estamos solos ante el peligro y tenemos herramientas diversas y con todo tipo de licencias. Por citar algunas:  <a href="http://www.atlassian.com/software/clover/">Clover</a>,  <a href="http://emma.sourceforge.net/">EMMA</a>,  <a href="http://www.vectorcast.com/">VectorCAST</a> o  <a href="http://cobertura.sourceforge.net/">Cobertura</a>. Mi elección en este caso será la última por varios motivos: es un producto open source (por los que tengo debilidad, especialmente cuando uno es autónomo), ofrece  <em>tasks</em> <span style="font-style:normal;">para usarlo desde Ant y tiene plugins para Hudson. Es una lástima que no tenga un plugin nativo para NetBeans, aunque como el IDE nos permite ejecutar</span> <em>targets</em> <span style="font-style:normal;">de Ant fácilmente, se lo podemos perdonar.</span></p>
<p class="western"><span style="font-style:normal;">Por cierto, existe un</span> <a href="http://codecoverage.netbeans.org/"><span style="font-style:normal;">plugin</span></a> <span style="font-style:normal;">de análisis de cobertura de código para NetBeans. No he jugado mucho con él, pero parece que es un servicio que sólo puede activarse para proyectos de JSE y no está disponible, por ejemplo, para un proyecto web. Si alguien tiene experiencia con el mismo, se agradecerá el feedback.</span></p>
<p class="western"><span style="font-style:normal;">Gran parte de la información necesaria que he usado para escribir esta entrada la he obtenido de</span> <a href="http://weblogs.java.net/blog/fabriziogiudici/archive/2007/02/automating_test.html"><span style="font-style:normal;">este post</span></a> <span style="font-style:normal;">.</span></p>
<h1 class="western">Configurar Cobertura en el NetBeans</h1>
<p class="western">El proceso es sencillo:</p>
<ol>
<li>
<p class="western" style="margin-bottom:0;">descargar el software,</p>
</li>
<li>
<p class="western" style="margin-bottom:0;">instalarlo,</p>
</li>
<li>
<p class="western">modificar el  <em>build.xml</em> <span style="font-style:normal;">del NetBeans.</span></p>
</li>
</ol>
<p class="western">Así pues, lo primero que tenemos que hacer es  <a href="http://cobertura.sourceforge.net/download.html">descargarnos</a> el software desde SourceForge. Nosotros vamos a contentarnos con la versión binaria y dejaremos el hacking para otro momento. A continuación procederemos a la instalación que no es más que descomprimirlo en una carpeta de la máquina de desarrollo. Yo tengo la costumbre de instalar estas cosas en el directorio  <em>/srv</em> <span style="font-style:normal;">y además suelo crear un</span> <em><span style="font-style:normal;">enlace simbólico</span></em> <span style="font-style:normal;">(que será el que referencie desde los proyectos) que no incluya el número de versión para que cada vez que haga una actualización no tenga que recorrer todos los scripts para corregir la referencia.</span></p>
<pre class="western">root@hargon:/srv# tar -zxf cobertura-1.9.1-bin.tar.gz
<span lang="zxx">root@hargon:/srv# ln -s cobertura-1.9.1 cobertura</span>
<span lang="zxx">root@hargon:/srv# ls -alh </span><span lang="zxx">total 684K</span>
<span lang="zxx">drwxr-xr-x  7 root      root    4,0K 2009-02-07 13:42 .</span>
<span lang="zxx">drwxr-xr-x 21 root      root    4,0K 2009-02-07 11:23 ..</span>
<span lang="zxx">lrwxrwxrwx  1 root      root      15 2009-02-07 13:42 cobertura -&gt; cobertura-1.9.1</span>
<span lang="zxx">drwxr-xr-x  4 root      root    4,0K 2009-02-04 21:23 cobertura-1.9.1</span>
<span lang="zxx">-rw-r--r--  1 root      root    649K 2009-02-04 21:40 cobertura-1.9.1-bin.tar.gz</span>
<span lang="zxx">root@hargon:/srv# rm cobertura-1.9.1-bin.tar.gz</span></pre>
<p class="western"><span style="font-style:normal;">Como</span> <a href="http://ivanator.wordpress.com/2009/02/06/pasar-propiedades-de-sistema-a-los-tasks-junit-de-los-scripts-de-netbeans/"><span style="font-style:normal;">ya he explicado</span></a> <span style="font-style:normal;">en alguna ocasión, NetBeans usa scripts de Ant para realizar todas sus tareas y están modularizados de manera que podamos introducir nuestros propios</span> <em>targets</em> <span style="font-style:normal;">y personalizaciones. En particular el fichero que podemos “tocar” tranquilamente es el</span> <em>build.xml</em> <span style="font-style:normal;">que está en la raíz del proyecto así que, mientras no diga lo contrario, todos los fragmentos de código deberán picarse en este fichero.</span></p>
<p class="western"><span style="font-style:normal;">Lo primero que tendremos que hacer proporcionar a Ant los diferentes</span> <em>tasks</em> <span style="font-style:normal;">proporcionados por Cobertura.</span></p>
<pre class="western" style="font-style:normal;">&lt;path id="cobertura.classpath"&gt;
    <span lang="zxx">&lt;fileset dir="${cobertura.dir}"&gt;</span>
        <span lang="zxx">&lt;include name="cobertura.jar" /&gt;</span>
        <span lang="zxx">&lt;include name="lib/**/*.jar" /&gt;</span>
    <span lang="zxx">&lt;/fileset&gt;</span>
<span lang="zxx">&lt;/path&gt;</span>

<span lang="zxx">&lt;taskdef classpathref="cobertura.classpath"
</span>    <span lang="zxx">resource="tasks.properties" /&gt;</span></pre>
<p class="western"><span style="font-style:normal;">Esto es uso estándar de Ant. La ruta a la raíz del frameworks lo indicamos en la propiedad</span> <em>cobertura.dir</em> <span style="font-style:normal;">que a su vez cargamos del fichero de propiedades privadas. Usamos el fichero de propiedades privadas</span> <em>(nbproject/private/private.properties)</em> <span style="font-style:normal;">porque la ruta del framework tiene sentido que pueda ser diferente en las máquinas de desarrollo y en el servidor de integración y, recordemos, si tenemos bien montado el repositorio de código el directorio</span> <em>nbproject/private</em> <span style="font-style:normal;">no suele incluirse en el mismo. Para que todo esto funcione deberemos cargar desde el</span> <em>build.xml</em> <span style="font-style:normal;">directamente dicho fichero de propiedades:</span></p>
<pre class="western"><span style="font-style:normal;">&lt;property file="</span><span style="font-style:normal;">nbproject/private/private.properties</span><span style="font-style:normal;">"/&gt;</span></pre>
<p class="western"><span style="font-style:normal;">y efectivamente añadir la propiedad en</span> <em>nbproject/private/private.properties</em> <span style="font-style:normal;">:</span></p>
<pre class="western" style="font-style:normal;">cobertura.dir=/srv/cobertura</pre>
<p class="western" style="font-style:normal;">Los pasos para usar Cobertura son los siguientes:</p>
<ol>
<li>
<p class="western" style="font-style:normal;">instrumentalizar el código bajo testing,</p>
</li>
<li>
<p class="western" style="font-style:normal;">ejecutar los tests unitarios con JUnit que genera los datos de cobertura y</p>
</li>
<li>
<p class="western" style="font-style:normal;">parsear los datos de cobertura para generar informes.</p>
</li>
</ol>
<h2 class="western" style="font-style:normal;">Instrumentalizar el código</h2>
<p class="western"><span style="font-style:normal;">Cobertura (y la mayoría de de frameworks de este tipo) funciona de manera que al ejecutar los tests unitarios se analiza qué parte de las clases bajo testing están siendo accedidas por las pruebas y qué partes no y se vuelca esta información en un archivo. Para que el framework de testing (en nuestro caso JUnit) pueda obtener esta información, Cobertura habrá tenido que instrumentalizar previamente las clases bajo test. Es decir, antes de ejecutar los tests, habremos tenido que permitir a Cobertura que genere un</span> <em>bytecode</em> <span style="font-style:normal;">modificado (instrumentalizado) para cada una de las clases bajo testing. Es un proceso similar a como funcionan algunos</span> <em>profilers</em> <span style="font-style:normal;">. Para obtener las clases instrumentalizadas lo haremos de la siguiente manera:</span></p>
<pre class="western" style="font-style:normal;">&lt;target name="cobertura-instrument" depends="compile"&gt;
    <span lang="zxx">&lt;cobertura-instrument
</span>        <span lang="zxx">todir="${cobertura.classes.dir}"</span>
        <span lang="zxx">datafile="${cobertura.ser.file}" &gt;</span>

    <span lang="zxx">&lt;fileset dir="${build.classes.dir}"&gt;</span>
        <span lang="zxx">&lt;include name="**/*.class"/&gt;</span>
    <span lang="zxx">&lt;/fileset&gt;</span>
    <span lang="zxx">&lt;/cobertura-instrument&gt;</span>
<span lang="zxx">&lt;/target&gt;</span></pre>
<p class="western"><span style="font-style:normal;">Creo que el</span> <em>target</em> <span style="font-style:normal;">se entiende bastante fácilmente: básicamente estamos especificando donde guardaremos las las clases instrumentalizadas (</span> <em>cobertura.classes.dir</em> <span style="font-style:normal;">), en qué fichero se escribirán los datos de cobertura (</span> <em>cobertura.ser.file</em> <span style="font-style:normal;">) y dónde residen las clases a instrumentalizar (</span> <em>build.classes.dir</em> <span style="font-style:normal;">). Esta última propiedad es una de las propiedades estándar que residen en el</span> <em>nbproject/properties</em> <span style="font-style:normal;">; las otras dos deberemos introducirlas en dicho fichero.</span></p>
<pre class="western" style="font-style:normal;">cobertura.ser.file=${build.dir}/cobertura.ser
<span lang="zxx">cobertura.classes.dir=${build.dir}/cobertura/classes</span></pre>
<p class="western" style="margin-bottom:0;font-style:normal;">
<p class="western"><span style="font-style:normal;">Como nuestro</span> <em>target</em> <span style="font-style:normal;">requiere los bytecode de las clases a instrumentalizar tiene como dependencia el</span> <em>target</em> <span style="font-style:normal;">estándar</span> <em>compile</em> <span style="font-style:normal;">que es el que usa NetBeans para compilar el proyecto. Entre las dependencias del propio</span> <em>compile</em> <span style="font-style:normal;">se encuentra el</span> <em>target</em> <em>init</em> <span style="font-style:normal;">que, entre otras cosas, es el encargado de cargar el fichero de propiedades (y por eso nosotros no tenemos que hacer de manera explícita la carga del mismo).</span></p>
<h2 class="western" style="font-style:normal;">Ejecutar los tests</h2>
<p class="western"><span style="font-style:normal;">El siguiente paso consiste en ejecutar los tests sobre las clases instrumentalizadas para así extraer la información de cobertura. Aquí también podemos aprovecharnos de los scripts generados por NetBeans. Nuestro nuevo</span> <em>target</em> <span style="font-style:normal;">, que llamaremos</span> <em>test-cobertura</em> <span style="font-style:normal;">básicamente será una copia del</span> <em>target</em> <em>test</em> <span style="font-style:normal;">(que es el que normalmente usa NetBeans para ejecutar las diferentes tests unitarios) con pequeñas modificaciones. Esta es la pinta que debería tener:</span></p>
<pre class="western" style="font-style:normal;">&lt;target name="cobertura-test"
    <span lang="zxx">depends="set-cobertura-file, init,compile-test,</span>
    <span lang="zxx">-pre-test-run,cobertura-instrument,</span>
    <span lang="zxx">-do-test-run,test-report,-post-test-run,-test-browse"&gt;</span>
<span lang="zxx">&lt;/target&gt;</span></pre>
<p class="western"><span style="font-style:normal;">Con respecto al original, hemos añadido dos nuevas dependencias. Por un lado</span> <em>cobertura-instrument</em> <span style="font-style:normal;">que es una referencia al</span> <em>target</em> <span style="font-style:normal;">anterior para asegurarnos que existen las clases instrumentalizadas. Por otro lado hemos definido otra dependencia a otro</span> <em>target</em> <span style="font-style:normal;">que también debemos construir y que muestro a continuación:</span></p>
<pre class="western" style="font-style:normal;">&lt;target name="set-cobertura-file" depends="init"&gt;
    <span lang="zxx">&lt;property
</span>        <span lang="zxx">name="test-sys-prop.net.sourceforge.cobertura.datafile"</span>
        <span lang="zxx">value="${cobertura.ser.file}"/&gt;</span>
<span lang="zxx">&lt;/target&gt;</span></pre>
<p class="western"><span style="font-style:normal;">Este</span> <em>target</em> <span style="font-style:normal;">es necesario para establecer un parámetro de sistema que debemos pasar a todos los tasks de JUnit usados en los diferentes</span> <em>targets</em> <span style="font-style:normal;">de los ficheros de scripting incluidos desde el</span> <em>build.xml</em> <span style="font-style:normal;">(en particular el</span> <em>build-impl.xml</em> <span style="font-style:normal;">). Necesitamos pasar este parámetro para que los</span> <em>tasks</em> <span style="font-style:normal;">del JUnit sepan dónde escribir la información de cobertura mientras ejecuta los tests.</span></p>
<p class="western"><span style="font-style:normal;">El nombre de la propiedad de sistema a pasar es</span> <em>net.sourceforge.cobertura.datafile</em> <span style="font-style:normal;">el porqué se pasa el parámetro de sistema en una propiedad con un nombre algo diferente, lo tengo explicado en</span> <a href="http://ivanator.wordpress.com/2009/02/06/pasar-propiedades-de-sistema-a-los-tasks-junit-de-los-scripts-de-netbeans/"><span style="font-style:normal;">este otro post</span></a> <span style="font-style:normal;">. El valor de la propiedad, como ya hemos visto (pues es un propiedad que ya he necesitado en</span> <em>targets</em> <span style="font-style:normal;">anteriores), está dentro del fichero</span> <em>nbproject/project.properties</em> <span style="font-style:normal;">y por eso nuestro</span> <em>target</em> <span style="font-style:normal;">tiene una dependencia del</span> <em>target init</em> <span style="font-style:normal;">que es el que se encargaba de cargar las propiedades definidas en ese fichero.</span></p>
<p class="western"><span style="font-style:normal;">Falta un detalle adicional: especificar al JUnit que ejecute las clases instrumentalizadas y no las generadas directamente en la fase de compilación. Para hacer esto sólo tenemos que introducir unos pequeños cambios en el</span> <em>classpath</em> <span style="font-style:normal;">utilizado por los</span> <em>targets</em> <span style="font-style:normal;">de testing lo que implica cambiar la propiedad</span> <em>run.test.classpath</em> <span style="font-style:normal;">definida en el</span> <em>nbproject/project.properties</em> <span style="font-style:normal;">para que quede así:</span></p>
<pre class="western" style="font-style:normal;">run.test.classpath=\
    <span lang="zxx">${cobertura.dir}/cobertura.jar:\</span>
    <span lang="zxx">${cobertura.classes.dir}:\</span>
    <span lang="zxx">${javac.test.classpath}:\</span>
    <span lang="zxx">${build.test.classes.dir}</span></pre>
<p class="western"><span style="font-style:normal;">Si hemos hecho todo correctamente al ejecutar el</span> <em>target</em> <em>cobertura-test</em> <span style="font-style:normal;">deberían ejecutarse los tests unitarios como siempre pero además debería haberse construido el fichero</span> <em>build/cobertura.ser</em> <span style="font-style:normal;">que contiene la información de cobertura.</span></p>
<h2 class="western" style="font-style:normal;">Crear los informes de cobertura</h2>
<p class="western"><span style="font-style:normal;">El fichero</span> <em>build/cobertura.ser</em> <span style="font-style:normal;">, como decíamos, contiene la información de cobertura pero está en formato binario porque ha tenido que generase en tiempo de ejecución de los tests de manera eficiente. Para extraer información legible hay que parsear dicho fichero. Para ello Cobertura nos proporciona el</span> <em>task cobertura-report</em> <span style="font-style:normal;">.</span></p>
<p class="western"><span style="font-style:normal;">Este</span> <em>task</em> <span style="font-style:normal;">permite generar los informes en dos formatos: en</span> <em>XML</em> <span style="font-style:normal;">o en</span> <em>HTML</em> <span style="font-style:normal;">. Nosotros generaremos ambos. El primero lo necesitaremos para pasárselo al plugin correspondiente en el Hudson. El segundo nos proporcionará una vista agradable y legible. Veamos el</span> <em>target</em> <span style="font-style:normal;">que tenemos que definir:</span></p>
<pre class="western" style="font-style:normal;">&lt;target name="cobertura-report" depends="cobertura-test"&gt;
    <span lang="zxx">&lt;cobertura-report</span>
        <span lang="zxx">datafile="${cobertura.ser.file}"</span>
        <span lang="zxx">format="xml"</span>
        <span lang="zxx">destdir="${cobertura.report.dir}"</span>
        <span lang="zxx"><span style="font-style:normal;">srcdir="${src.dir}" /&gt;</span></span>

    <span lang="zxx">&lt;cobertura-report</span>
        <span lang="zxx">datafile="${cobertura.ser.file}"</span>
        <span lang="zxx">format="html"</span>
        <span lang="zxx">destdir="${cobertura.report.dir}"</span>
        <span lang="zxx">srcdir="${src.dir}" /&gt;</span>
<span lang="zxx">&lt;/target&gt;</span></pre>
<p class="western"><span style="font-style:normal;">Creo que aquí no hay mucho que explicar. Defino dónde está el fichero que contiene la información de cobertura, dónde quiero generar los informes y el formato de los mismos. Como siempre habremos de definir la propiedad</span> <em>cobertura.report.dir</em> <span style="font-style:normal;">en el</span> <em>nbproject/project.properties</em> <span style="font-style:normal;">:</span></p>
<pre class="western" style="font-style:normal;">reports.dir=${build.dir}/reports
<span lang="zxx">cobertura.report.dir=${reports.dir}/cobertura-report</span></pre>
<p class="western"><span style="font-style:normal;">Si ahora ejecutamos este</span> <em>target</em> <span style="font-style:normal;">deberían generarse los informes en el directorio especificado. Si le echamos un ojo a los informes de tipo</span> <em>HTML</em> <span style="font-style:normal;">vemos que básicamente ha generado unas páginas al estilo de la documentación de las API pero con la información de cobertura. Una manera muy útil e intuitiva de mostrar la información.</span></p>
<p class="western" style="font-style:normal;">
<p class="western" style="font-style:normal;"><img class="alignnone" style="border:0 none;text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/cobertura-report-main.jpg?w=500&#038;h=354" border="0" alt="" width="500" height="354" /></p>
<p class="western" style="text-align:center;">
<p class="western"><img class="alignnone" style="border:0 none;text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/cobertura-report-class-detail.jpg?w=500&#038;h=660" border="0" alt="" width="500" height="660" /></p>
<h1 class="western">Configurar el Hudson</h1>
<p class="western">Una vez hemos configurado todo lo anterior, ya tenemos hecho el trabajo duro. La configuración del Hudson, afortunadamente, es bastante sencilla. Estos son los pasos:</p>
<ol>
<li>
<p class="western">instalar Cobertura,</p>
</li>
<li>
<p class="western">instalar el plugin de Cobertura y</p>
</li>
<li>
<p class="western">configurar el  <em>job</em> <span style="font-style:normal;">:</span></p>
<ol>
<li>
<p class="western" style="font-style:normal;">configurar el workspace para indicar dónde está Cobertura,</p>
</li>
<li>
<p class="western" style="font-style:normal;">añadir el  <em>cobertura-report</em> en la lista de  <em>targets</em> que tiene que ejecutar el Ant y</p>
</li>
<li>
<p class="western" style="font-style:normal;">configurar el directorio donde están los informes en XML</p>
</li>
</ol>
</li>
</ol>
<h2 class="western" style="page-break-after:avoid;"><span style="font-family:Albany;"><span style="font-size:medium;"><strong>Instalar Cobertura</strong></span></span></h2>
<p class="western">Para que el Ant lanzado por Hudson al construir el proyecto sea capaz de instanciar los  <em>tasks</em> <span style="font-style:normal;">del</span> <span style="font-style:normal;">Cobertura, éste tendrá que ser instalado en la máquina donde reside Hudson. El proceso de instalación será exactamente análogo al que hemos hecho en el entorno de desarrollo: descargar el paquete, descomprimirlo</span> <span style="font-style:normal;">en el directorio</span> <em>/srv</em> <span style="font-style:normal;">y establecer un enlace simbólico.</span></p>
<h2 class="western"><span style="font-family:Albany;"><span style="font-size:medium;"><strong>Instalar el plugin</strong></span></span></h2>
<p class="western">Una vez tenemos Hudson instalado, añadir plugins es trivial. Desde la interfaz web vamos a  <em><strong>Hudson -&gt; Manage Hudson -&gt; Manage Plugins -&gt; Available</strong></em> <span style="font-style:normal;"><span style="font-weight:normal;">y entonces seleccionamos</span></span> <span style="font-style:normal;"><strong>Hudson Cobertura plugin</strong></span> <span style="font-style:normal;"><span style="font-weight:normal;">y le damos a</span></span> <em><strong>Install</strong></em> <span style="font-style:normal;"><span style="font-weight:normal;">. El asistente entonces instala el plugin tras los cual habrá que reiniciar Hudson.</span></span></p>
<h2 class="western"><strong><span style="font-style:normal;">Configurar el</span><em>job</em></strong></h2>
<p class="western">Aquí estamos presuponiendo que ya existe un  <em>job</em> <span style="font-style:normal;">previo correctamente configurado y al que simplemente vamos a añadir el soporte de cobertura. Si no fuera así, echadle un ojo a este</span> <a href="http://ivanator.wordpress.com/2009/01/12/montando-un-entorno-integracion-continua-hudson-ant-svn-netbeans/"><span style="font-style:normal;">otro post</span></a> <span style="font-style:normal;">que ya publiqué en su momento.</span></p>
<p class="western"><span style="font-style:normal;">Lo primero que haremos es configurar el</span> <em>workspace</em> <span style="font-style:normal;">para que sea capaz de encontrar Cobertura durante la fase de construcción del proyecto. Para ello, nos vamos al directorio</span> <em>nbproject/private/private.properties</em> <span style="font-style:normal;">del</span> <em>job</em> <span style="font-style:normal;">correspondiente y añadimos la propiedad que indica dicha ruta:</span></p>
<pre class="western">cobertura.dir=/srv/cobertura</pre>
<p class="western"><span style="font-style:normal;">A continuación modificaremos los</span> <em>targets</em> <span style="font-style:normal;">de Ant que usa Hudson para construir el</span> <em>job</em> <span style="font-style:normal;">. Si tuviéramos el</span> <em>target</em> <em>test</em> <span style="font-style:normal;">lo eliminaríamos y a continuación añadiremos el</span> <em>target cobertura-report</em> <span style="font-style:normal;">.</span></p>
<p class="western"><span style="font-style:normal;">Lo siguiente es configurar el plugin para que utilice el informe</span> <em>XML</em> <span style="font-style:normal;">generado:</span></p>
<ul>
<li>
<p class="western"><strong><span style="font-style:normal;">Publish Cobertura Coverage Report</span></strong> <span style="font-style:normal;"><span style="font-weight:normal;">: activamos la opción para que Hudson muestre la información de cobertura.</span></span></p>
<ul>
<li>
<p class="western" style="text-align:left;"><strong><span style="font-style:normal;">Cobertura xml report pattern</span></strong> <span style="font-style:normal;"><span style="font-weight:normal;">: build/reports/cobertura-report/coverage.xml<br />
Aquí ponemos la ruta al fichero donde hemos generado el informe en formato XML.</span></span></li>
<li>
<p class="western" style="text-align:left;"><strong><span style="font-style:normal;">Coverage Metric Targets</span></strong> <span style="font-style:normal;"><span style="font-weight:normal;">: este apartado nos permite definir las métricas a partir de las cuales la construcción del</span></span> <em><span style="font-weight:normal;">job</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">se considera estable o no.</span></span></p>
</li>
</ul>
</li>
</ul>
<p class="western" style="font-style:normal;font-weight:normal;"><span style="color:#000000;"><img class="alignnone" style="border:1px solid black;text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/hudson-cobertura-configuration.jpg" border="1" alt="" width="NaN" height="NaN" /><br />
</span></p>
<p class="western" style="text-align:left;"><span style="font-style:normal;"><span style="font-weight:normal;">En principio ya está todo. Ahora podríamos lanzar una construcción para comprobarlo. Si todo va bien, Hudson mostrará los informes de cobertura bajo el enlace</span></span> <em><span style="font-weight:normal;">Coverage Report</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">en la raíz del</span></span> <em><span style="font-weight:normal;">job</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">.</span></span></p>
<p class="western" style="font-style:normal;font-weight:normal;"><img class="alignnone" style="border:0 none;text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/hudson-cobertura-report.jpg" border="0" alt="" width="500" height="556" /></p>
<div>
<p style="margin-top:.5cm;margin-bottom:.4cm;font-weight:normal;text-align:center;">- 8 -</p>
</div>
</div>
 Tagged: Agile, calidad, Hudson, Java, NetBeans <a rel="nofollow" href="http://feeds.wordpress.com/1.0/gocomments/ivanator.wordpress.com/142/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/comments/ivanator.wordpress.com/142/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/godelicious/ivanator.wordpress.com/142/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/delicious/ivanator.wordpress.com/142/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/gostumble/ivanator.wordpress.com/142/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/stumble/ivanator.wordpress.com/142/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/godigg/ivanator.wordpress.com/142/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/digg/ivanator.wordpress.com/142/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/goreddit/ivanator.wordpress.com/142/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/reddit/ivanator.wordpress.com/142/" /></a> <img alt="" border="0" src="http://stats.wordpress.com/b.gif?host=ivanator.wordpress.com&blog=6092394&post=142&subd=ivanator&ref=&feed=1" /></div>]]></content:encoded>
			<wfw:commentRss>http://ivanator.wordpress.com/2009/02/07/cobertura-con-netbeans-y-hudson/feed/</wfw:commentRss>
		<slash:comments>2</slash:comments>
	
		<media:content url="" medium="image">
			<media:title type="html">ivanator</media:title>
		</media:content>

		<media:content url="http://ivanator.files.wordpress.com/2009/02/cobertura-report-main.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/cobertura-report-class-detail.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/hudson-cobertura-configuration.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/hudson-cobertura-report.jpg" medium="image" />
	</item>
		<item>
		<title>Pasar propiedades de sistema a los tasks JUnit de los scripts de NetBeans</title>
		<link>http://ivanator.wordpress.com/2009/02/06/pasar-propiedades-de-sistema-a-los-tasks-junit-de-los-scripts-de-netbeans/</link>
		<comments>http://ivanator.wordpress.com/2009/02/06/pasar-propiedades-de-sistema-a-los-tasks-junit-de-los-scripts-de-netbeans/#comments</comments>
		<pubDate>Fri, 06 Feb 2009 19:59:46 +0000</pubDate>
		<dc:creator>Iván Párraga García</dc:creator>
				<category><![CDATA[Uncategorized]]></category>
		<category><![CDATA[Ant]]></category>
		<category><![CDATA[JUnit]]></category>
		<category><![CDATA[NetBeans]]></category>

		<guid isPermaLink="false">http://ivanator.wordpress.com/?p=138</guid>
		<description><![CDATA[Como ya he explicado alguna vez, NetBeans utiliza scripts de Ant para realizar las diferentes tareas que lleva acabo el IDE, como compilar el proyecto, lanzar los tests, ejecutarlo, desplegarlo, etc.
NetBeans genera los scripts de manera modular de manera que el usuario pueda personalizarlo de forma adecuada. Genera el build.xml principal en la raíz del [...]<img alt="" border="0" src="http://stats.wordpress.com/b.gif?host=ivanator.wordpress.com&blog=6092394&post=138&subd=ivanator&ref=&feed=1" />]]></description>
			<content:encoded><![CDATA[<div class='snap_preview'><br /><p>Como ya he explicado alguna vez, NetBeans utiliza scripts de Ant para realizar las diferentes tareas que lleva acabo el IDE, como compilar el proyecto, lanzar los tests, ejecutarlo, desplegarlo, etc.</p>
<p>NetBeans genera los scripts de manera modular de manera que el usuario pueda personalizarlo de forma adecuada. Genera el <em>build.xml</em> principal en la raíz del proyecto que a su vez incluye otros tantos recursos del subdirectorio de <em>nbproject</em>. Estos recursos son ficheros de propiedades y de scripting adicionales con la implementación real de los diferentes <em>targets. </em></p>
<p>La idea de esta estructuración es que los ficheros con <em>targets</em> en el directorio <em>nbproject</em> (<em>build-impl.xml</em>, por ejemplo) nunca deberían tocarse ya que se pueden regenerar (y por tanto se perdería toda nuestra personalizacion) en función de las diferentes opciones que definamos para nuestro proyecto a través de la propia interfaz del IDE. Por contra, el <em>build.xml</em> principal no se regenerará y es por tanto en este fichero donde personalizaremos los scripts (añadir nuevos <em>targets</em>, definir propiedades, etc.).</p>
<p>El problema surge cuando queremos modificar el comportamiento de algunos de los <em>targets</em> definidos en los ficheros &#8220;intocables&#8221;. Por ejemplo, un requerimiento que puede surgir fácilmente (veremos algún ejemplo en próximos posts) es la necesidad de pasar parámetros de sistema a los <em>tasks</em> que ejecutan las pruebas unitarias con JUnit.</p>
<p>Afortunadamente el equipo de NetBeans ha previsto esta situación y la solución es bastante sencilla. Imaginemos que queremos pasar una propiedad de sistema que se llama <em>MiPropiedad </em>con valor <em>MiValor</em>, la solución pasa por definir en el <em>build.xml </em>una propiedad que se llame <em>test-sys-prop.MiPropiedad</em> y todos los <em>taks </em>Junit de los scripts generados automáticamente recibirán la propiedad <em>MiPropiedad</em> con el valor que le hayamos indicado. Si necesitáramos pasar más de una propiedad repetiríamos el mismo proceso con otro nombre de parámetro.</p>
<p>En código, habría que incluir el siguiente tag en el <em>build.xml</em> principal:</p>
<pre>&lt;property name="test-sys-prop.MiPropiedad" value="MiValor" /&gt;</pre>
<p>Si queréis entender el motivo exacto y hackear los scripts de NetBeans, echadle un ojo a <a href="http://weblogs.java.net/blog/fabriziogiudici/archive/2007/01/setting_propert.html" target="_blank">este post</a>.</p>
 Tagged: Ant, JUnit, NetBeans <a rel="nofollow" href="http://feeds.wordpress.com/1.0/gocomments/ivanator.wordpress.com/138/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/comments/ivanator.wordpress.com/138/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/godelicious/ivanator.wordpress.com/138/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/delicious/ivanator.wordpress.com/138/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/gostumble/ivanator.wordpress.com/138/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/stumble/ivanator.wordpress.com/138/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/godigg/ivanator.wordpress.com/138/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/digg/ivanator.wordpress.com/138/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/goreddit/ivanator.wordpress.com/138/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/reddit/ivanator.wordpress.com/138/" /></a> <img alt="" border="0" src="http://stats.wordpress.com/b.gif?host=ivanator.wordpress.com&blog=6092394&post=138&subd=ivanator&ref=&feed=1" /></div>]]></content:encoded>
			<wfw:commentRss>http://ivanator.wordpress.com/2009/02/06/pasar-propiedades-de-sistema-a-los-tasks-junit-de-los-scripts-de-netbeans/feed/</wfw:commentRss>
		<slash:comments>1</slash:comments>
	
		<media:content url="" medium="image">
			<media:title type="html">ivanator</media:title>
		</media:content>
	</item>
		<item>
		<title>Monitorizar Hudson desde NetBeans</title>
		<link>http://ivanator.wordpress.com/2009/02/06/monitorizar-hudson-desde-netbeans/</link>
		<comments>http://ivanator.wordpress.com/2009/02/06/monitorizar-hudson-desde-netbeans/#comments</comments>
		<pubDate>Fri, 06 Feb 2009 17:30:01 +0000</pubDate>
		<dc:creator>Iván Párraga García</dc:creator>
				<category><![CDATA[Uncategorized]]></category>
		<category><![CDATA[Hudson]]></category>
		<category><![CDATA[NetBeans]]></category>

		<guid isPermaLink="false">http://ivanator.wordpress.com/?p=131</guid>
		<description><![CDATA[
En un post anterior explicaba como montar un entorno básico para poder empezar a trabajar en un entorno TDD. El ide que elegí fue NetBeans y el servidor de integración continua Hudson. El siguiente paso natural es poder monitorizar Hudson desde el propio NetBeans sin tener que abrir el navegador web. Para poder llevar a [...]<img alt="" border="0" src="http://stats.wordpress.com/b.gif?host=ivanator.wordpress.com&blog=6092394&post=131&subd=ivanator&ref=&feed=1" />]]></description>
			<content:encoded><![CDATA[<div class='snap_preview'><br /><div>
<p>En un <a href="http://ivanator.wordpress.com/2009/01/12/montando-un-entorno-integracion-continua-hudson-ant-svn-netbeans/">post anterior</a> explicaba como montar un entorno básico para poder empezar a trabajar en un entorno TDD. El ide que elegí fue NetBeans y el servidor de integración continua Hudson. El siguiente paso natural es poder monitorizar Hudson desde el propio NetBeans sin tener que abrir el navegador web. Para poder llevar a cabo esta tarea, existe un plugin desarrollado por el propio equipo de Hudson. Aunque su uso es muy sencillo, en este post explicaré cómo instalarlo, configurarlo y utilizarlo. Los desarrolladores del plugin tienen un <a href="http://hudson.gotdns.com/wiki/display/HUDSON/NetBeansPlugin">entrada en su wiki</a>, así que mi post es un poco redundante <img src='http://s.wordpress.com/wp-includes/images/smilies/icon_razz.gif' alt=':-P' class='wp-smiley' /> </p>
<h1 class="western">Instalar el plugin</h1>
<p>El primer paso consiste en instalar el plugin. Para ello tenemos varias alternativas.</p>
<p>Podemos  <a href="http://plugins.netbeans.org/PluginPortal/faces/PluginDetailPage.jsp?pluginid=2178">descargarlo desde el NetBeans Plugin Portal</a> como un fichero  <em>nbm</em> <span style="font-style:normal;">que luego instalaremos a través del gestor de plugins del NetBeans:</span> <span style="font-style:normal;"><strong>Tools -&gt; Plugins -&gt; Downloaded -&gt; Add Plugins</strong></span> <span style="font-style:normal;"><span style="font-weight:normal;">y seleccionaremos el fichero descargado.</span></span></p>
<p>La otra alternativa, que prefiero, es hacerlo directamente desde la pestaña de plugins disponibles siempre que tengamos configurado el  <em>Update Center</em> <span style="font-style:normal;">del</span> <em>Plugin Portal</em> <span style="font-style:normal;">:</span> <span style="font-style:normal;"><strong>Tools -&gt; Plugins -&gt; Available Plugins -&gt; Install</strong></span> <span style="font-style:normal;"><span style="font-weight:normal;">.</span></span></p>
<p style="font-style:normal;font-weight:normal;">
<p style="font-style:normal;font-weight:normal;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/gestorpluginsnetbeans.jpg" border="0" alt="" width="75%" height="41%" /></p>
<h1 class="western">Configurar el plugin</h1>
<p style="font-style:normal;font-weight:normal;">Una vez instalado lo tenemos disponible en la pestaña  <em><strong>Services</strong></em> junto con los servidores, los SGBDs y demás, valga la redundancia, servicios.</p>
<p style="font-style:normal;font-weight:normal;">Haciendo click derecho sobre el icono del Hudson, nos aparece la opción de añadir una instancia (podríamos monitorizar simultáneamente diferentes servidores de integración). Simplemente tendremos que completar el asistente dándole un nombre a esa instancia, asignándole la  <em>url</em> donde reside el servidor y una frecuencia de refresco del estado del mismo.</p>
<h1 class="western">Usar el plugin</h1>
<p>A partir de ese momento podemos ver el estado de Hudson de un vistazo. Por un lado podemos ver el estado de todos los  <em>jobs</em> <span style="font-style:normal;">con códigos de colores (correcto, roto o inestable)</span> y el estado de ocupación de la cola de construcción. Por otro lado podemos programar nuevas construcciones de cualquier  <em>job</em> <span style="font-style:normal;">haciendo</span> <span style="font-style:normal;"><strong>click derecho sobre el</strong></span> <em><strong>job</strong></em> <span style="font-style:normal;"><strong>-&gt; Start job</strong></span> <span style="font-style:normal;"><span style="font-weight:normal;">o ver una información más extendida del</span></span> <em><span style="font-weight:normal;">job</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">haciendo doble click sobre el mismo (lo que abre una ventana en el panel del editor). Obviamente podemos saltar directamente a la página web correspondiente en el navegador haciendo</span></span> <span style="font-style:normal;"><strong>click derecho sobre el nodo correspondiente -&gt; Open in Browser</strong></span> <span style="font-style:normal;"><span style="font-weight:normal;">.</span></span></p>
<p style="text-align:center;"><a href="http://ivanator.files.wordpress.com/2009/02/netbeans_service_hudson.jpg"></a><a href="http://ivanator.files.wordpress.com/2009/02/netbeans_service_hudson1.jpg"><img class="aligncenter size-full wp-image-133" title="netbeans_service_hudson1" src="http://ivanator.files.wordpress.com/2009/02/netbeans_service_hudson1.jpg" alt="netbeans_service_hudson1" width="510" height="386" /></a></p>
<p><span style="font-style:normal;"><span style="font-weight:normal;">El “problema” de monitorizar de esta manera es que nos fuerza a acceder a la pantalla de servicios. Afortunadamente el plugin también proporciona una alerta a manera de iconito rojo parpadeante y estresante en el margen inferior izquierdo del NetBeans. El icono omnipresente persiste hasta que se arregle el</span></span> <em><span style="font-weight:normal;">job</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">roto. Clickando sobre el mismo nos da información del</span></span> <em><span style="font-weight:normal;">job</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">(o</span></span> <em><span style="font-weight:normal;">jobs</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">) que están causando la alerta.</span></span></p>
<p style="font-style:normal;font-weight:normal;">Disfrutadlo.</p>
<p style="font-style:normal;font-weight:normal;">
</div>
 Tagged: Hudson, NetBeans <a rel="nofollow" href="http://feeds.wordpress.com/1.0/gocomments/ivanator.wordpress.com/131/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/comments/ivanator.wordpress.com/131/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/godelicious/ivanator.wordpress.com/131/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/delicious/ivanator.wordpress.com/131/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/gostumble/ivanator.wordpress.com/131/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/stumble/ivanator.wordpress.com/131/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/godigg/ivanator.wordpress.com/131/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/digg/ivanator.wordpress.com/131/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/goreddit/ivanator.wordpress.com/131/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/reddit/ivanator.wordpress.com/131/" /></a> <img alt="" border="0" src="http://stats.wordpress.com/b.gif?host=ivanator.wordpress.com&blog=6092394&post=131&subd=ivanator&ref=&feed=1" /></div>]]></content:encoded>
			<wfw:commentRss>http://ivanator.wordpress.com/2009/02/06/monitorizar-hudson-desde-netbeans/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
	
		<media:content url="" medium="image">
			<media:title type="html">ivanator</media:title>
		</media:content>

		<media:content url="http://ivanator.files.wordpress.com/2009/02/gestorpluginsnetbeans.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/netbeans_service_hudson1.jpg" medium="image">
			<media:title type="html">netbeans_service_hudson1</media:title>
		</media:content>
	</item>
		<item>
		<title>Montando un entorno de integración continua (Hudson + Ant + SVN + NetBeans)</title>
		<link>http://ivanator.wordpress.com/2009/01/12/montando-un-entorno-integracion-continua-hudson-ant-svn-netbeans/</link>
		<comments>http://ivanator.wordpress.com/2009/01/12/montando-un-entorno-integracion-continua-hudson-ant-svn-netbeans/#comments</comments>
		<pubDate>Mon, 12 Jan 2009 18:24:45 +0000</pubDate>
		<dc:creator>Iván Párraga García</dc:creator>
				<category><![CDATA[Agile]]></category>
		<category><![CDATA[Ant]]></category>
		<category><![CDATA[Hudson]]></category>
		<category><![CDATA[Integración Continua]]></category>
		<category><![CDATA[Java]]></category>
		<category><![CDATA[JUnit]]></category>
		<category><![CDATA[NetBeans]]></category>
		<category><![CDATA[SVN]]></category>
		<category><![CDATA[TDD]]></category>

		<guid isPermaLink="false">http://ivanator.wordpress.com/?p=3</guid>
		<description><![CDATA[
Este post como artículo en PDF
Introducción
Cuando uno lee posts, libros y escucha experiencias de la aproximación ágil al desarrollo del software y empieza a profundizar y a aprender conceptos tales como el TDD (test driven development) y la integración continua, se le despiertan una ganas irrefrenables de empezar a trabajar de este modo. En este [...]<img alt="" border="0" src="http://stats.wordpress.com/b.gif?host=ivanator.wordpress.com&blog=6092394&post=3&subd=ivanator&ref=&feed=1" />]]></description>
			<content:encoded><![CDATA[<div class='snap_preview'><br /><div>
<p>Este post como artículo en <a href="http://ivanator.files.wordpress.com/2009/02/montando-un-entorno-de-integracion-continua-doc-v12.pdf">PDF</a></p>
<h1>Introducción</h1>
<p>Cuando uno lee posts, libros y escucha experiencias de la aproximación ágil al desarrollo del software y empieza a profundizar y a aprender conceptos tales como el TDD (test driven development) y la integración continua, se le despiertan una ganas irrefrenables de empezar a trabajar de este modo. En este post lo que intentaré explicar es cómo montar un entorno sencillo que nos permita tener los cimientos de un sistema ágil que cada uno podrá extender para satisfacer sus propias necesidades. Como se menciona en  <a href="http://architects.dzone.com/articles/the-three-pillars-continuous-i" target="_blank">este post</a>, tres de los pilares de la integración continua son: un repositorio de datos, un servidor de integración continua y una herramienta que permita automatizar la construcción de la aplicación. Yo además añadiría un cuarto: un IDE que se integre bien con el resto del entorno. No es el propósito de este post discutir la utilidad de estos componentes (el post mencionado anteriormente puede ser un buen punto de partida para los recién llegados), sino elegir unas herramientas concretas y describir su configuración e integración. En mi caso he hecho la siguiente elección (que no tiene por qué ser la única ni la mejor):</p>
<ul>
<li>repositorio de código: SVN</li>
<li>servidor de integración continua: Hudson</li>
<li>herramienta de construcción: ANT</li>
<li>IDE: NetBeans 6.5</li>
</ul>
<p>Por simplicidad he distribuido el entorno entre una máquina Linux (Ubuntu 8.10) con IP 192.168.1.10 (dato que es relevante para tareas de configuración que iremos viendo) que actúa simultáneamente como repositorio de datos y de integración continua y un número arbitrario de máquinas de desarrollo (también Ubuntu 8.10) que contienen los IDE&#8217;s y el entorno con el que trabajan los desarrolladores. En cualquier caso, otras distribuciones tienen exactamente la misma complicación y los pasos requeridos son directamente extrapolables y fácilmente adaptables. Para explicar los diferentes pasos utilizaré un mismo ejemplo ya que considero que hablar con algo concreto suele ser más sencillo, tanto para el que explica como para el que intenta entender, que hacerlo en general o en abstracto. Presupongo la inteligencia de los potenciales lectores para hacer las adaptaciones requeridas en cada punto a sus propias necesidades. No es muy relevante, pero utilizaré un proyecto personal que estoy empezando estos días que sirve para hacer un seguimiento de las valoraciones de mis cursos y que se titula, en un alarde de originalidad, &#8220;Encuestas&#8221;.</p>
<p>En el siguiente diagrama muestro el entorno que acabaremos montando. Las rutas que aparecen en el servidor no tenemos por qué entenderlas ahora pero servirán como referencia a medida que vayamos construyendo la solución.</p>
<p style="text-align:center;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/sbres-1233750052-0.jpg" border="0" alt="" width="75%" height="25%" /></p>
<h1>Configuración Subversion</h1>
<p>He elegido Subversion porque es un repositorio más o menos potente que supera algunas de las limitaciones de nuestro querido y viejo CVS y aún a día de hoy es una de las herramientas que, bajo mi humilde opinión, están más extendidas en la industria. No voy a entrar en los detalles de instalación y de configuración del propio Subversion. Para ello ya hay mucha documentación online (por ejemplo  <a href="http://svnbook.red-bean.com/" target="_blank">aquí</a>). Me centraré en la creación de un repositorio para nuestro proyecto. Es cierto que se puede usar el mismo repositorio para más de un proyecto, pero yo prefiero tenerlo separado (alguno de los motivos aparecerán más adelante, aunque también se puede justificar razonadamente la otra postura).</p>
<h2>Crear el repositorio del SVN que almacenará el proyecto</h2>
<p>Como decía, presupongo que el servidor que actuará como repositorio de datos tiene instalado el Subversion y sus herramientas administrativas (si no es así, nuestro querido  <em>apt-get</em> o  <em>Synaptic</em> vendrán a nuestro rescate). Lo primero que hay que hacer es determinar en qué ruta almacenaré el repositorio. Yo tiendo a almacenar los diferentes repositorios de cada uno de los proyectos bajo una misma raíz y así mantengo el servidor más o menos organizado. La ruta también debería ser vistosa porque luego formará parte de la configuración que utilicen los clientes (como los IDE&#8217;s) para conectarse con dicho repositorio. Trabajando con Ubuntu, y si hemos instalado el Subversion desde paquetes, se crea un usuario  <em>svn</em>. Para que todo funcione de manera correcta, deberíamos asegurarnos que el propietario del repositorio es este usuario. Una manera de hacerlo es ejecutar los comandos de administración del Subversion con este usuario y otro es cambiar después el propietario del directorio (y su contenido) con el comando  <em>chown</em> (chown -fR svn .)/Encuestas-2). En definitiva, para crear el repositorio haremos lo siguiente:</p>
<pre style="margin-bottom:.5cm;">svnadmin create /srv/svn/Encuestas-2</pre>
<p>El directorio  <em>/srv/svn</em> es la raíz de todos mis repositorios y  <em>Encuestas-2</em> es el que usaré para este proyecto en particular. Si ahora nos situamos en en el ese directorio y hacemos un listado deberíamos ver (más o menos) lo siguiente:</p>
<pre>ivan@hargon:/srv/svn/Encuestas-2$ ls -alh
total 36K
drwxr-xr-x 7 svn root 4,0K 2009-01-11 10:43 .
drwxr-xr-x 6 svn root 4,0K 2009-01-11 10:43 ..
drwxr-xr-x 2 svn root 4,0K 2009-01-11 10:43 conf
drwxr-xr-x 2 svn root 4,0K 2009-01-11 10:43 dav
drwxr-sr-x 5 svn root 4,0K 2009-01-11 10:43 db
-r--r--r-- 1 svn root    2 2009-01-11 10:43 format
drwxr-xr-x 2 svn root 4,0K 2009-01-11 10:43 hooks
drwxr-xr-x 2 svn root 4,0K 2009-01-11 10:43 locks
-rw-r--r-- 1 svn root  229 2009-01-11 10:43 README.txt</pre>
<h2>Establecer la seguridad del repositorio</h2>
<p>Cuando se crea un repositorio en el Subverson es, obviamente, para que los desarrolladores puedan utilizarlo. Por tanto un paso importante es la parte de securización y autorización del mismo: establecer quién podrá acceder y sus credenciales. Subversion tiene diferentes mecanismos de autenticación y de transporte seguro de la información (incluyendo  <em>ssh</em>). No voy a entrar en los detalles y explicaré cómo hacerlo de la manera más sencilla. Para más información os remito al  <a href="http://svnbook.red-bean.com/" target="_blank">anterior enlace</a>.</p>
<p>En mi caso, todas las máquinas residen en el interior de mi red de área local, que puede considerarse &#8220;zona segura&#8221;. Un punto importante es que los mecanismos de seguridad (y otros más) se establecen por repositorio (en contraposición a una configuración global para todos ellos) lo que nos permite que convivan repositorios con diferentes configuraciones (este es uno de los motivos por los que prefiero tener un repositorio por proyecto). Lo primero es determinar el modo de autenticación. Para ello habremos de editar el fichero</p>
<pre style="margin-bottom:.5cm;">/srv/svn/Encuestas/conf/svnserve.conf</pre>
<p>para que quede más o menos así:</p>
<pre>[general]
anon-access = none
auth-access = write
password-db = passwd
realm = My Realm</pre>
<p>De hecho estas opciones seguramente preexisten en el fichero generado en el repositorio y lo único que habrá que hacer es descomentarlas. También es posible que en el fichero haya otras tantas opciones comentadas que no vamos a necesitar para nuestra configuración básica, así como diferentes comentarios descriptivos. Yo los he obviado por claridad. Con esta configuración lo que estamos haciendo es decirle al Subversion que busque los usuarios y las credenciales en un fichero de texto plano. El siguiente paso, precisamente, consiste en configurar qué usuarios y con qué credenciales podrán acceder al repositorio editando dicho fichero que se encuentra en:</p>
<pre style="margin-bottom:.5cm;">/srv/svn/Encuestas/conf/passwd</pre>
<p>que editaremos para que quede algo parecido a lo siguiente:</p>
<pre>[users]
ivan = super-secret-passwd
ana = top-secret-passwd</pre>
<p>Estos serán los usuarios y passwords que habrá que configurar en los clientes (en nuestro caso usaremos NetBeans).</p>
<h1>Configuración NetBeans</h1>
<p>Supongamos que nuestra aplicación de ejemplo  <em>Encuestas</em> es una aplicación web típica que va a correr encima de un Tomcat 6. Así pues lo primero que haremos es crear un nuevo proyecto web mediante los asistentes del IDE. Asumo que estos pasos los sabréis llevar a cabo pero, en cualquier caso, no son el tema del presente post y lo pasaré muy rápido.</p>
<p style="text-align:center;"><img style="text-align:bottom;" src="http://ivanator.files.wordpress.com/2009/01/netbeans-crear-web-project1.jpg" border="0" alt="Asistente del NetBeans para crear un proyecto web" width="510" height="168" /></p>
<h2>Conexión con el Subversion</h2>
<p>Una vez tenemos creado el proyecto web, lo siguiente es enlazarlo con el Subversion. Para ello haremos  <strong>click derecho</strong> en el nombre del  <strong>proyecto</strong>, y en el menú  <strong>versioning</strong> seleccionaremos la opción  <strong>Import</strong> <strong>into Subversion Repository</strong> que nos abrirá un asistente para configurar la conexión. En mi versión de NetBeans (la 6.5) el aspecto que tiene es el que podéis ver en la siguiente captura de pantalla.</p>
<p style="text-align:center;"><img style="text-align:bottom;" src="http://ivanator.files.wordpress.com/2009/01/asistente-svn-paso-11.jpg" border="0" alt="Asistente del NetBeans para configurar el SVN - Paso 1" width="510" height="194" /></p>
<p>Los siguientes pasos nos permiten seleccionar qué ficheros importar al repositorio. En este punto se podría abrir una discusión sobre si sólo subir los ficheros de código y recursos necesarios para el proyecto desde un punto de vista de independencia del IDE utilizado o si bien subir también los ficheros y recursos asociados al IDE. Puede ser una discusión interesante, pero la dejaremos para otro momento. En esta ocasión, para simplificar (y en mi caso es siempre la opción preferida), subiremos todos los recursos tal y como muestra el tercer paso del asistente.</p>
<p style="text-align:center;"><img style="text-align:bottom;" src="http://ivanator.files.wordpress.com/2009/01/asistente-svn-paso-3.jpg" border="0" alt="bla bla" width="574" height="372" /></p>
<p>En este momento ya tenemos conectado el IDE con el repositorio. Los pasos e ideas fundamentales son los mismos para otros entornos como Eclipse y CVS. Quizá en otro post puedo explicar cómo configurar alguna de estas otras opciones. Para utilizar el NetBeans como cliente de Subversion tan sólo hay que hacer  <strong>click derecho</strong> sobre el recurso correspondiente (un directorio o un fichero) y elegir la opción  <strong>Subversion</strong> del menú contextual. La siguiente captura de pantalla lo ilustra.</p>
<p style="text-align:center;"><img style="text-align:bottom;" src="http://ivanator.files.wordpress.com/2009/01/netbeans-uso-svn.jpg" border="0" alt="Usando NetBeans como cliente de SVN" width="50%" height="50%" /></p>
<h2>Tests unitarios con JUnit en NetBeans</h2>
<p>En posts siguientes explicaré cómo integrar en el entorno ágil que estamos montando diferentes herramientas que aumenten nuestra productividad y la calidad de nuestro código ( <a href="http://pmd.sourceforge.net/" target="_blank">PMD</a>,  <a href="http://checkstyle.sourceforge.net/">CheckStyle</a>,  <a href="http://findbugs.sourceforge.net/" target="_blank">FindBugs</a>, herramientas de cobertura&#8230;), pero por el momento me centraré en aquellas herramientas mínimas que requiere un entorno ágil que quiera trabajar con TDD: test unitarios. La idea es la siguiente: imaginemos que queremos trabajar con el framework de tests unitarios  <a href="http://www.junit.org/">JUnit</a>. Instalaríamos la herramienta en tres &#8220;lugares&#8221;:</p>
<ol>
<li>Como plug-in del IDE, lo que permite un uso integrado en el entorno de desarrollo del programador, con todas la ventajas de uso que ello conlleva.</li>
<li>Como  <em>target</em> en el  <em>build.xml</em> del Ant. Este  <em>target</em> nos permitirá ejecutar las mismas tareas desde la línea de comandos sin ayuda del IDE pero además, y más importante, permitirá al servidor de integración continua lanzar la tarea y generar los diferentes informes (en el caso de JUnit serán ficheros XML) especificando qué tests han sido superados, cuáles han fallado, etcétera.</li>
<li>Como plug-in (que a veces puede venir preinstalado) del servidor de integración continua (en nuestro caso Hudson), de manera que pueda utilizar los informes generados en el  <em>target</em> del Ant para mostrar información útil y gráfica y determinar si la construcción ha fallado o no entre otras cosas.</li>
</ol>
<p>En este apartado veremos el primer punto y el resto los iré desarrollando a lo largo del post. También os aconsejo que leáis  <a href="http://www.netbeans.org/kb/docs/java/junit-intro.html">este magnífico post</a> sobre cómo usar JUnit en NetBeans del propio blog oficial del IDE. El plug-in de JUnit ya está instalado por defecto en la mayoría (si no en todas) las distribuciones de NetBeans como IDE Java. Incluye además las dos versiones que se utilizan hoy día (la familia 3.X y la familia 4.X). Yo personalmente me decanto por la versión 4 porque me gusta poder utilizar anotaciones. Al crear cualquier proyecto Java con los asistentes de NetBeans, éste ya está preconfigurado para utilizar el framework (aunque nosotros podamos ignorarlo). Lo podéis comprobar en dos sitios: por un lado en la ventana  <strong>Projects</strong> podemos ver que se ha creado un directorio  <strong>Test Packages</strong> y por otro lado si hacemos  <strong>click derecho</strong> sobre el nombre del  <strong>proyecto</strong>, y abrimos el diálogo  <strong>Properties</strong> en la sección  <strong>Libraries &#8211; Compile Test</strong> veremos que están están agregadas las bibliotecas correspondientes.</p>
<p style="text-align:center;"><img style="text-align:bottom;" src="http://ivanator.files.wordpress.com/2009/01/netbeans-properties-junit-libraries.jpg" border="0" alt="Configuración por defecto de librerias en un proyecto web de NetBeans" width="509" height="396" /></p>
<p>La manera de construir un test unitario con NetBeans es muy simple. Simplemente tenemos que  <strong>seleccionar</strong> la clase que queremos  <strong>testear</strong>, hacer  <strong>click derecho</strong>, seleccionar  <strong>Tools</strong> y finalmente  <strong>Create JUnit Tests</strong> que lanzará un menú modal en el que podremos seleccionar qué código autogenerar en el unit test que se va a crear (en el primero unit test que creemos, el entorno nos preguntará qué versión de JUnit querremos usar y ya la mantendrá para el resto del proyecto).</p>
<p style="text-align:center;"><img style="text-align:bottom;" src="http://ivanator.files.wordpress.com/2009/01/netbeans-crear-unit-test.jpg" border="0" alt="Crear un unit test a partir de una clase" width="510" height="478" /></p>
<p>Una vez codifiquemos el test unitario, lo siguiente es poder ejecutarlo; nada más sencillo:  <strong>Run -&gt; Test Project</strong> (o la combinación  <strong>ALT + F6</strong>) y el resultado indicando si se pasan o no los tests se mostrará por la consola del IDE.</p>
<h1>Configuración del Ant</h1>
<p>Supongo que en este punto no tengo que explicar qué es  <a href="http://ant.apache.org/" target="_blank">Ant</a> y asumo que conocéis su funcionamiento. Si bien trabajando en local, el IDE puede hacernos transparentes todos aquellos aspectos engorrosos como configurar el CLASSPATH, compilar, generar documentación, correr herramientas de análisis, desplegar en el servidor de aplicaciones, etcétera, cuando tenemos que hacerlo de manera remota, o hacer el despliegue en producción, o automatizar cualquiera de estas tareas dentro de un script, o, lo que más nos interesa en este post, permitir a un servidor de integración continua que construya nuestro proyecto, necesitamos una herramienta independiente del IDE. Ant es una opción que nos permite llevar esto a cabo (otra opción, por ejemplo, sería  <a href="http://maven.apache.org/" target="_blank">Maven</a>).</p>
<p>Trabajando con Eclipse, hasta donde llega mi conocimiento, no tenemos más remedio que construir los  <em>build.xml</em> a mano. De todos modos una vez que se ha hecho uno, si se organiza y parametriza de forma adecuada, el resto de proyectos pueden ir reutilizando una misma “plantilla” con cambios mínimos.</p>
<p>Trabajando con NetBeans podemos tener parte del trabajo resuelto. Ello se debe a que todas las tareas que realiza el IDE (compilar, testear, configurar los diferentes CLASSPATH, desplegar, etc.), lo hace con Ant mediante ficheros  <em>build.xml</em> (y otros ficheros auxiliares) que construye dinámicamente en función de lo que nosotros configuremos de manera gráfica en la interfaz de NetBeans o a través de cualquiera de sus asistentes. Nosotros podemos aprovecharnos de estos ficheros generados para utilizarlos desde fuera del IDE para nuestros propios propósitos. La propia estructura de los scripts preveen diferentes  <em>targets</em> &#8220;hook&#8221; donde añadir nuestro propio scripting si el generado no es suficiente.</p>
<p>A primera vista esto podría parecer genial y bueno, en gran parte lo es, pero hay que tener en cuenta una serie de consideraciones. La principal es que los scripts generados no son totalmente independientes del IDE ni de la máquina en la que se han generado porque utilizan recursos (bibliotecas y  <em>tasks</em>) que sólo existen en los directorios de instalación del IDE y algunas de las rutas generadas son absolutas y por tanto dependientes de la máquina en donde se generaron. Por tanto si estos ficheros quieren utilizarse en otro contexto habrá que hacer algunos ajustes a mano. Dependiendo del tipo de proyecto en el que estemos trabajando (no será lo mismo una aplicación JSE que una aplicación J2EE con dependencias de un servidor de aplicaciones) la cantidad de esfuerzo requerida para hacer los scripts portables será mayor o menor. En  <a href="http://weblogs.java.net/blog/fabriziogiudici/archive/2006/11/setting_up_netb.html" target="_blank">este post</a> se muestran algunos de los perfiles de aplicaciones y sus posibles problemas y soluciones (se corresponde con una versión de NetBeans algo vieja, pero la mayor parte de lo que se explica todavía aplica a las versiones actuales).</p>
<p>NetBeans genera el  <em>build.xml</em> en la raíz del proyecto. Si lo abrís podéis ver que básicamente es un script vacío que lo que hace es importar otros tantos ficheros (más o menos dependiendo del tipo de proyecto) que se encuentran en la subcarpeta  <strong>nbproject</strong>. NetBeans utiliza esta carpeta como su directorio de trabajo y para contener los diferentes metadatos del proyecto. En principio no deberíamos tocar ninguno de los ficheros contenidos bajo esta jerarquía de directorios a no ser que sepamos exactamente lo que estamos haciendo. Dentro de nbproject tenemos otra carpeta especial:  <strong>private</strong>. Deberemos ir con cuidado de no incluir esta carpeta en el repositorio (por defecto el asistente ya lo hace bien) ya que contiene ficheros de parámetros con paths absolutos y dependientes de la máquina de cada desarrollador (o servidor de integración) usados por, entre otras cosas, por los scripts de Ant. Esto será parte de lo que tendremos que replicar a mano en el servidor de integración adaptándolo adecuadamente. (pero ya lo veremos más adelante).</p>
<p style="text-align:center;"><img style="text-align:bottom;" src="http://ivanator.files.wordpress.com/2009/01/netbeans-ant-build.jpg" border="0" alt="Detalle del build.xml generado por el NetBeans" width="510" height="546" /></p>
<p>Al abrir el  <em>build.xml</em> desde el propio NetBeans, la ventana Navigator nos muestra las tareas disponibles. Si queréis detalles, investigad un poco, pero bueno tenemos todos aquellos  <em>targets</em> que esperaríamos encontrar y que podemos necesitar (y muchos más), por ejemplo:</p>
<ul>
<li>clean</li>
<li>compile</li>
<li>debug</li>
<li>dist</li>
<li>javadoc</li>
<li>test</li>
</ul>
<p>Los incrédulos podéis abrir una consola y comprobar como podéis invocar las tareas desde la línea de comandos. Para ello sólo os tenéis que situar en el directorio raíz del proyecto y ejecutar el ant como lo haríais con cualquier otro build.xml.</p>
<h1>Configuración Hudson</h1>
<p>Sólo resta la última pieza del puzle: automatizar la construcción del proyecto desde nuestro servidor de integración continua (Hudson) conectándolo al repositorio. También explicaré como automatizar el hecho que se programe una nueva construcción automática cada vez que hay un  <em>commit</em> en el repositorio.  <a href="https://hudson.dev.java.net/" target="_blank">La instalación y configuración inicial de Hudson</a> es trivial pero está fuera del ámbito de este tutorial (baste decir que básicamente consiste en desplegar un  <em>war</em> en un servidor de aplicaciones o un contenedor web como Tomcat).</p>
<h2>Automatización de la construcción</h2>
<p>Hudson automatiza la construcción de proyectos en lo que denomina  <em><strong>j</strong></em> <strong><em>obs</em></strong>, así pues un mismo proyecto software puede tener diferentes  <em>jobs</em>. Por ejemplo, tener configurado un  <em>job</em> que se ejecute cada vez que se haga un  <em>commit</em> en el repositorio y que lo único que compruebe es que el proyectos es compilable y que se pasan los test unitarios y tener otro  <em>job</em> programado para ejecutarse tres veces al día para correr tests de aceptación o de integración más pesados. Nosotros empezaremos por algo muy sencillo, configurando un  <em>job</em> para que haga lo que explicaba en el primer ejemplo (compilación más tests unitarios).</p>
<p>Hudson tiene una interfaz muy sencilla de utilizar. A diferencia de otros servidores de integración, tanto la creación como la configuración de los  <em>jobs</em> se hace íntegramente desde la interfaz web.</p>
<p style="text-align:center;"><img style="text-align:bottom;" src="http://ivanator.files.wordpress.com/2009/01/hudson-dashboard.jpg" border="0" alt="Página principal del Hudson" width="510" height="347" /></p>
<p>Para crear nuestro  <em>job</em> haremos click en  <span style="font-style:normal;"><strong>New Job</strong></span> y se iniciará el asistente de creación. Lo primero que tenemos que hacer es dar un nombre a nuestro  <em>job</em> que será utilizado tanto en la interfaz e internamente en el servidor de integración. Se puede utilizar cualquier símbolo en el nombre e incluso espacios, pero yo prefiero (manía personal) utilizar nombres con una sola palabra porque cuando luego se quiere navegar &#8220;a mano&#8221; (mediante la consola) por los  <em>jobs</em> siempre es más cómodo no tener espacios puesto que el nombre que usemos se corresponderá con el nombre del directorio donde residirá toda la información asociada a dicho  <em>job</em>. En nuestro ejemplo:  <strong>Encuestas-2</strong>. Lo siguiente es elegir el tipo de  <em>job</em> que estamos construyendo; nosotros construiremos un  <em>job</em> estándar que es lo que Hudson llama un  <strong><span style="font-style:normal;">Build a free-style software project</span></strong>. Lo seleccionamos y le damos a  <span style="font-style:normal;"><strong>OK</strong></span> para ir a la siguiente pantalla del asistente. En esta nueva pantalla tenemos todos los elementos de configuración del  <em>job</em> que rellenaremos (aquéllos que haga falta) como sigue:</p>
<ul>
<li><strong>Source Code Managment:</strong> Subversion
<ul>
<li>
<p style="text-align:left;"><strong>Repository URL:</strong> svn://192.168.1.10/srv/svn/Encuestas-2/Encuestas<br />
Aquí hay que tener en cuenta una cosa. La raíz del repositorio es  <em>Encuestas-2</em> <span style="font-style:normal;">pero configuramos la url hasta</span> <em>Encuestas</em> <span style="font-style:normal;">. Si no fuera así, el</span> <em>workspace</em> <span style="font-style:normal;">generado para construir</span> <span style="font-style:normal;">este job tendría un subdirectorio</span> <em>Encuestas</em> <span style="font-style:normal;">(lo que complicaría la configuración del resto de elementos del</span> <em>job</em> <span style="font-style:normal;">)</span></li>
</ul>
</li>
<li><strong>Build Triggers</strong>: permite definir en qué momento Hudson tiene que construir el proyecto. Hay diferentes políticas con las que podéis experimentar. Nuestro propósito para este  <em>job</em> es que se construya cada vez que hay modificaciones en el repositorio. Para implementar esta política hay dos posibles maneras: o bien Hudson va preguntando al repositorio &#8220;¿ha habido cambios en el código?&#8221; de forma periódica (lo que se conoce como mecanismo por encuesta y que se corresponde con la opción &#8220;poll&#8221;) o bien que el propio Subversion avise a Hudson cada vez que ha habido un cambio. Esta última aproximación es más eficiente (y a mí parecer más elegante), pero se trata más bien de una configuración del Subversion (que veremos más adelante) y no del Hudson. Así pues no activaremos ningún tipo de  <em>trigger</em> (lo dejamos tal y como está).</li>
<li><strong>Build</strong>: nos permite configurar cómo Hudson debe construir el proyecto. Existen diferentes mecanismos pero nosotros en este post vemos cómo se hace mediante Ant, así que será esta la opción que elegiremos.
<ul>
<li><strong>Ant version</strong>: dejaremos la &#8220;Default&#8221; con lo que Hudson utilizará la versión que encuentre en el path (esto está presuponiendo que existe un Ant configurado correctamente en la máquina donde reside Hudson).</li>
<li><strong>Targets</strong> <strong><span style="font-weight:normal;">:</span></strong> compile test javadoc<br />
Aquí tendremos que especificar los  <em>targets</em> definidos en el  <em>build.xml</em> que queremos que Hudson ejecute. Como estamos utilizando el  <em>build.xml</em> automáticamente generado por NetBeans tendremos que tener en cuenta algunas consideraciones que explico más adelante (como había comentado el  <em>build.xml</em> generado no es del todo independiente del NetBeans y habrá que poner algún parche para que funcione desde Hudson).</li>
</ul>
</li>
<li><strong>Publish Javadoc</strong> <span style="font-weight:normal;">:</span> <span style="font-weight:normal;">marcamos esta opción para publicar los documentos generados por</span> <em><span style="font-weight:normal;">javadoc</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">asociados a nuestro proyecto. Los</span></span> <em><span style="font-weight:normal;">javadoc</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">se generarán porque hemos definido</span></span> <span style="font-style:normal;"><span style="font-weight:normal;">el</span></span> <em><span style="font-weight:normal;">target</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">correspondiente en la sección previa.</span></span>
<ul>
<li><strong><span style="font-style:normal;">Javadoc directory</span></strong> <span style="font-style:normal;"><span style="font-weight:normal;">:Encuestas/dist/javadoc</span></span></li>
</ul>
</li>
<li><strong><span style="font-style:normal;">Publish JUnit test result report</span></strong> <span style="font-style:normal;"><span style="font-weight:normal;">: marcamos esta opción para publicar los informes (y gráficas) de los resultados producidos por los diferentes test unitarios creados en el proyecto.</span></span>
<ul>
<li><strong><span style="font-style:normal;">Test report XMLs</span></strong> <span style="font-style:normal;"><span style="font-weight:normal;">: Encuestas/build/test/results/*.xml</span></span></li>
</ul>
</li>
</ul>
<p style="font-style:normal;font-weight:normal;">
<p><span style="color:#000000;"><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/sbres-1233750052-9.jpg" border="1" alt="" width="75%" height="65%" /><br />
</span></p>
<h2>Configurar el workspace del Hudson y añadir dependencias externas al Subversion</h2>
<p>Hudson almacena todos los ficheros que necesita para su funcionamiento así como los diferentes artefactos para cada uno de los  <em>job</em> en un directorio raíz conocido como el HUDSON_HOME. No es el propósito de este artículo explicar cómo configurarlo, pero puesto que necesitaremos modificar algunos aspectos de nuestro  <em>job</em> sí que necesitamos saber cuál es su valor. Para verlo, desde la página principal de Hudson accedemos al menú de configuración y es el primer parámetro que nos encontramos (Hudson Home -&gt; Manage Hudson -&gt; System Configuration -&gt; Home directory). En mi caso es el siguiente</p>
<pre style="margin-bottom:.5cm;">HUDSON_HOME = /srv/hudson_home</pre>
<p>Dentro de este directorio existe el subdirectorio  <em>jobs</em> que contiene una entrada para cada  <em>job</em> configurado. Así pues aquí tendremos un subdirectorio  <em>Encuestas-2</em> (que, como decía, se corresponde con el nombre que le asignamos al  <em>job</em> cuando lo configuramos previamente). Dentro del directorio correspondiente a cada job, uno de los subdirectorios es <em>workspace</em> <em><span style="font-style:normal;">(si no existiera, le daríamos a construir el job desde la</span></em> <em><span style="font-style:normal;">interfaz del Hudson)</span></em>. Este directorio contiene los ficheros descargados desde el repositorio de código cada vez que se lanza una construcción del job (el código fuente, los ficheros a usar por Ant, las bibliotecas incluidas en el proyecto, etc) y los artefactos generados en la construcción del proyecto. Lo remarco de nuevo:  <strong>contiene los ficheros descargados desde el repositorio</strong>. Lo que quiere decir que si el proyecto para ser construido necesita artefactos que existen en los entornos de desarrollo pero que no están incluidos en el repositorio (o sus localizaciones son diferentes a las especificadas en los diferentes scripts de construcción), la construcción inevitablemente fallará. Con las herramientas con las que estamos trabajando habrá que hacer lo siguiente:</p>
<ul>
<li>opcionalmente, si es un proyecto J2EE o requiere bibliotecas externas al contenido del repositorios de código, hacerlas accesibles al Hudson,</li>
<li>añadir algunas otras dependencias de NetBeans que tiene el  <em>build.xml</em> generado,</li>
<li>construir el directorio  <em>/nbproject/private</em> y su contenido para solucionar las dependencias.</li>
</ul>
<p>Nuestro proyecto es un proyecto web que hemos configurado de manera que haga uso de la implementación de Servlets de Tomcat. Ello significa que en nuestro entorno de desarrollo tenemos un Tomcat y por tanto tendremos que replicar su instalación en la máquina donde reside Hudson para que pueda usar dichas bibliotecas. Lo más sencillo es empaquetar el Tomcat instalado en uno de los entornos de desarrollo y desempaquetarlo tal cual en el servidor de integración. En mi caso lo situaré en  <em>/srv/apache-tomcat-6.0.18</em>.</p>
<p>La manera más sencilla de solucionar las dependencias que tienen los scripts del Ant del NetBeans es seguir una aproximación similar a la del caso del servidor de aplicaciones: copiar el propio NetBeans en la máquina donde reside el servidor de integración para que éste tenga acceso directo a todos los recursos. Obviamente es posible usar una aproximación menos “drástica” y copiar sólo aquellos recursos necesarios, pero sin duda copiar todo el entorno es la manera más rápida que hacerlo y la única penalización que obtenemos es ocupar unos cuantos MB más en el disco duro. Personalmente prefiero “malgastar” esos pocos a MB a tener que estar hackeando los scripts para ver qué necesito exactamente o pelearme con posibles y sutiles errores difíciles de trazar. Recapitulando: empaquetaremos el NetBeans de una de las máquinas de desarrollo y en mi caso lo situaré en  <em>/srv/netbeans-6.5</em>. Adicionalmente NetBeans también guarda parte de la configuración en un directorio oculto en el  <em>home</em> del usuario en un directorio oculto llamado  <em>.netbeans</em> <span style="font-style:normal;">(esto es útil para que diferentes usuarios en una misma máquina física puedan tener configuraciones diferentes). En este caso no será necesario copiarlo absolutamente todo (un poco más adelante daré los detalles).</span></p>
<p>A continuación hemos de construir el directorio  <em>private</em> <span style="font-style:normal;">que contiene los metadatos donde, entre otras cosas, se está configurando parte de la información local a utilizar por Ant y que por ello no se ha incluido en el Subversion. El tipo de proyecto que estamos usando (proyecto web usando Tomcat) sólo genera un fichero en esta carpeta,</span> <em>build.properties</em> <span style="font-style:normal;">. Construiremos la carpeta</span> <em>private</em> <span style="font-style:normal;">como un subdirectorio de la carpeta</span> <em>nbproject</em> <span style="font-style:normal;">y copiaremos el fichero</span> <em>properties</em> <span style="font-style:normal;">de uno de los entornos de desarrollo.</span></p>
<p>Los cambios que aplicaremos (todos relacionados con el cambio de rutas absolutas diferentes en desarrollo y en integración) son los siguientes:</p>
<ul>
<li><a id="DDE_LINK" name="DDE_LINK"></a>cambiar la ruta al Tomcat por la que corresponde (sustituir la cadena  <em>/home/ivan/apache-tomcat-6.0.18</em> <span style="font-style:normal;">por la cadena</span> <em>/srv/apache-tomcat-6.0.18</em>),</li>
<li>cambiar la ruta del Netbeans (sustituir la cadena  <em>/home/ivan/.netbeans-6.5</em> <span style="font-style:normal;">por la cadena</span> <em>/srv/netbeans-6.5</em>),</li>
<li>cambiar las referencias al directorio de configuración del  <em>home</em> (sustituir la cadena  <em>/home/ivan/.netbeans</em> <span style="font-style:normal;">por</span> <em>/srv/.netbeans</em>).</li>
</ul>
<p><span style="font-style:normal;">La primera línea del</span> <em><span style="font-weight:normal;">nbproject/private/</span></em> <em>properties</em> <span style="font-style:normal;">pasará a referenciar al archivo de configuración del Tomcat a la nueva ubicación en el servidor de integración en la ruta</span> <em>/srv/.netbeans/6.5/tomcat60.properties</em> <span style="font-style:normal;">. Editaremos el fichero para adaptarlo al nuevo entorno; de hecho realmente la única línea que habremos de modificar es la que define el directorio donde reside el Tomcat porque el resto sólo tienen sentido si queremos arrancar el propio Tomcat (circunstancia que se da en las máquinas de desarrollo pero no, en principio, en el servidor de integración). Tras modificar el fichero tendríamos algo así:</span></p>
<pre>tomcat.home=/srv/apache-tomcat-6.0.18
tomcat.password=super-password-de-administracion-de-Tomcat
tomcat.url=http://localhost:8084
tomcat.username=ide</pre>
<p>Otro fichero referenciado en el  <em><span style="font-weight:normal;">nbproject/private/</span></em> <em>properties</em> <span style="font-style:normal;">que también tendemos que modificar será el</span> <em>/srv/.netbeans/6.5/build.properties</em> <span style="font-style:normal;">. Este fichero gestiona una gran cantidad de propiedades. Afortunadamente con sólo sustituir todas las ocurrencias de la cadena que indica la ruta de la raíz de la instalación del NetBeans por la nueva ruta en el servidor de integración será suficiente (en mi caso, sustituir</span> <em>/home/ivan/netbeans-6.5</em> <span style="font-style:normal;">por</span> <em>/srv/netbeans-6.5</em> <span style="font-style:normal;">).</span></p>
<h2 style="font-style:normal;">Comprobando que el job está bien configurado</h2>
<p style="font-style:normal;">¡Felicidades! El camino ha sido más o menos largo y tedioso (no tanto, ¿verdad?), pero ya hemos casi terminado.</p>
<p><span style="font-style:normal;">Para asegurarnos que todo se ha configurado correctamente intentaremos construir el proyecto desde la consola del servidor de integración. Para ello nos situaremos en el directorio</span> <em>Encuestas</em> <span style="font-style:normal;">dentro del workspace del job que estamos configurando</span> <em>(/srv/hudson_home/jobs/Encuestas-2/workspace/Encuestas)</em> <span style="font-style:normal;">que define la raíz del proyecto tal y como está en el Subversion. En ese directorio tenemos el</span> <em>build.xml</em> <span style="font-style:normal;">así que podemos invocar al Ant tranquilamente. Una buena idea es probar todos los</span> <em>targets</em> <span style="font-style:normal;">que querremos ejecutar desde el Hudson (</span> <em>compile, test</em> <span style="font-style:normal;">y</span> <em>javadoc</em> <span style="font-style:normal;">). Si se ejecutan sin problemas tenemos la garantía que Hudson también podrá hacerlo.</span></p>
<p><span style="font-style:normal;">La prueba de fuego será construir el job desde la interfaz del</span> <span style="font-style:normal;"><span style="font-weight:normal;">Hudson:</span></span> <span style="font-style:normal;"><strong>Hudson Home -&gt; Encuestas-2 -&gt; Build Now</strong></span> <span style="font-style:normal;">lo que programa una ejecución. Al terminar la misma, tendremos una nueva entrada en la</span> <span style="font-style:normal;"><strong>Build History</strong></span> <span style="font-style:normal;"><span style="font-weight:normal;">que tendrá una bullet azul o roja en función de si el proyecto se ha podido construir de forma correcta o de si ha habido algún error.</span></span></p>
<p><span style="font-style:normal;"><span style="font-weight:normal;">Los links</span></span> <span style="font-style:normal;"><strong>Javadoc</strong></span> <span style="font-style:normal;"><span style="font-weight:normal;">y</span></span> <span style="font-style:normal;"><strong>Latest Test Result</strong></span> <span style="font-style:normal;"><span style="font-weight:normal;">nos permiten acceder respectivamente a la documentación del proyecto y a los informes de testing generados por los diferentes tests unitoarios (Junit).</span></span></p>
<p><img style="text-align:left;" src="http://ivanator.files.wordpress.com/2009/02/sbres-1233750052-10.jpg" border="0" alt="" width="680" height="442" /> <span style="font-style:normal;"><span style="font-weight:normal;">En este punto no estaría de más hacer una copia de seguridad del directorio</span></span> <em><span style="font-weight:normal;">nbproject/private</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">y todo su contenido. El resto del workspace lo podéis regenerar al momento desde el repositorio, pero los cambios que habéis aplicado a los ficheros contenidos en el</span></span> <em><span style="font-weight:normal;">private</span></em> <span style="font-style:normal;"><span style="font-weight:normal;">no los tenéis en ningún otro sitio. Puesto que a priori, como decía, podemos recuperar (casi) todo el workspace desde el Subversion, en según qué circunstancias podemos decidir borrarlo sin acordarnos de todas las modificaciones que hemos tenido que hacer manualmente.</span></span></p>
<h2>Conexión con el SVN</h2>
<p>Podríamos dejarlo aquí, pero uno de los requisitos que queríamos configurar es que el  <em>job</em> se construyera automáticamente cada vez que hiciéramos un  <em>commit</em> en el Subversion. Para conseguir este comportamiento tenemos que configurar el Subversion para que avise al Hudson cada vez que detecte esta situación.</p>
<p>Una de las características que tiene Hudson es que permite programar la construcción de un  <em>job</em> <span style="font-style:normal;">haciendo una petición</span> <em>GET</em> <span style="font-style:normal;">sobre una</span> <em>url</em> <span style="font-style:normal;">asociada al proyecto en cuestión. Siguiendo con nuestro ejemplo dicha</span> <em>url</em> <span style="font-style:normal;">sería</span> <em>http://192.168.1.10:8180/hudson/job/Encuestas-2/build.</em></p>
<p style="font-style:normal;">Por otro lado, el Subversion prevee una serie de eventos que pueden ser detectados y asociarles un script. Tenemos una serie de plantillas de los scripts y eventos que se pueden detectar en la carpeta  <em>hooks</em> del repositorio que estamos utilizando. Puesto que nosotros queremos activar la construcción del proyecto cada vez que hacemos un commit, el script que necesitamos codificar es uno que se llame  <em>post-commit</em>:</p>
<pre>svn@hargon:/srv/svn/Encuestas-2/hooks$ ls
post-commit.tmpl          post-unlock.tmpl  pre-revprop-change.tmpl

post-lock.tmpl            pre-commit.tmpl   pre-unlock.tmpl
post-revprop-change.tmpl  pre-lock.tmpl     start-commit.tmpl
svn@hargon:/srv/svn/Encuestas-2/hooks$ cp post-commit.tmpl
post-commit
svn@hargon:/srv/svn/Encuestas-2/hooks$ chmod ug+x post-commit</pre>
<p>El fichero que hemos copiado es una simple plantilla. Ahora la editaremos para que efectivamente haga una petición  <em>GET</em> <span style="font-style:normal;">. Esto lo podemos hacer de múltiples maneras, aunque una muy fácil es utilizar Perl y algunas de sus bibliotecas (el gestor de paquetes</span> <em>apt-get</em> <span style="font-style:normal;">o</span> <em>Synaptic</em> <span style="font-style:normal;">nos permitirá</span> <span style="font-style:normal;"><span style="font-weight:normal;">instalar todas las bibliotecas requeridas de una manera sencilla</span></span> <span style="font-style:normal;">). Una posible implementación de este script es la siguiente:</span></p>
<pre style="font-style:normal;">#!/usr/bin/perl
use LWP::Simple;
use HTML::LinkExtor;
my $content =
get("http://192.168.1.10:8180/hudson/job/Encuestas-2/build");</pre>
<p>Comprobemos que el script funciona ejecutándolo desde la consola. Si todo va bien, al abrir el Hudson veremos que se ha lanzado una construcción del job.</p>
<p>El último paso es probar que la integración con el Subversion funciona correctamente. Simplemente tenemos que hacer un  <em>commit</em> <span style="font-style:normal;">que modifique cualquier fichero y el Hudson debería construir el job.</span></p>
<p><span style="font-style:normal;">ACTUALIZACIÓN: <a href="http://ivanator.wordpress.com/2009/02/07/cobertura-con-netbeans-y-hudson/">2ª Parte &#8211; Añadir métricas de cobertura de código de los tests unitarios</a></span></p>
<p><span style="font-style:normal;">ACTUALIZACIÓN: <a href="http://ivanator.wordpress.com/2009/03/05/metricas-de-calidad-con-netbeans-y-hudson/">3ª Parte &#8211; Métricas de calidad con NetBeans y Hudson</a><br />
</span></div>
 Tagged: Agile, Ant, Hudson, Integración Continua, Java, JUnit, NetBeans, SVN, TDD <a rel="nofollow" href="http://feeds.wordpress.com/1.0/gocomments/ivanator.wordpress.com/3/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/comments/ivanator.wordpress.com/3/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/godelicious/ivanator.wordpress.com/3/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/delicious/ivanator.wordpress.com/3/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/gostumble/ivanator.wordpress.com/3/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/stumble/ivanator.wordpress.com/3/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/godigg/ivanator.wordpress.com/3/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/digg/ivanator.wordpress.com/3/" /></a> <a rel="nofollow" href="http://feeds.wordpress.com/1.0/goreddit/ivanator.wordpress.com/3/"><img alt="" border="0" src="http://feeds.wordpress.com/1.0/reddit/ivanator.wordpress.com/3/" /></a> <img alt="" border="0" src="http://stats.wordpress.com/b.gif?host=ivanator.wordpress.com&blog=6092394&post=3&subd=ivanator&ref=&feed=1" /></div>]]></content:encoded>
			<wfw:commentRss>http://ivanator.wordpress.com/2009/01/12/montando-un-entorno-integracion-continua-hudson-ant-svn-netbeans/feed/</wfw:commentRss>
		<slash:comments>8</slash:comments>
	
		<media:content url="" medium="image">
			<media:title type="html">ivanator</media:title>
		</media:content>

		<media:content url="http://ivanator.files.wordpress.com/2009/02/sbres-1233750052-0.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/01/netbeans-crear-web-project1.jpg" medium="image">
			<media:title type="html">Asistente del NetBeans para crear un proyecto web</media:title>
		</media:content>

		<media:content url="http://ivanator.files.wordpress.com/2009/01/asistente-svn-paso-11.jpg" medium="image">
			<media:title type="html">Asistente del NetBeans para configurar el SVN - Paso 1</media:title>
		</media:content>

		<media:content url="http://ivanator.files.wordpress.com/2009/01/asistente-svn-paso-3.jpg" medium="image">
			<media:title type="html">bla bla</media:title>
		</media:content>

		<media:content url="http://ivanator.files.wordpress.com/2009/01/netbeans-uso-svn.jpg" medium="image">
			<media:title type="html">Usando NetBeans como cliente de SVN</media:title>
		</media:content>

		<media:content url="http://ivanator.files.wordpress.com/2009/01/netbeans-properties-junit-libraries.jpg" medium="image">
			<media:title type="html">Configuración por defecto de librerias en un proyecto web de NetBeans</media:title>
		</media:content>

		<media:content url="http://ivanator.files.wordpress.com/2009/01/netbeans-crear-unit-test.jpg" medium="image">
			<media:title type="html">Crear un unit test a partir de una clase</media:title>
		</media:content>

		<media:content url="http://ivanator.files.wordpress.com/2009/01/netbeans-ant-build.jpg" medium="image">
			<media:title type="html">Detalle del build.xml generado por el NetBeans</media:title>
		</media:content>

		<media:content url="http://ivanator.files.wordpress.com/2009/01/hudson-dashboard.jpg" medium="image">
			<media:title type="html">Página principal del Hudson</media:title>
		</media:content>

		<media:content url="http://ivanator.files.wordpress.com/2009/02/sbres-1233750052-9.jpg" medium="image" />

		<media:content url="http://ivanator.files.wordpress.com/2009/02/sbres-1233750052-10.jpg" medium="image" />
	</item>
	</channel>
</rss>