<?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/"
	>

<channel>
	<title>Blog de Justo Aguilar &#187; java</title>
	<atom:link href="http://blog.justoaguilar.com/tag/java/feed/" rel="self" type="application/rss+xml" />
	<link>http://blog.justoaguilar.com</link>
	<description>Libros, Java y lo que surja.</description>
	<lastBuildDate>Sun, 22 Jan 2012 14:26:48 +0000</lastBuildDate>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>http://wordpress.org/?v=3.3.1</generator>
		<item>
		<title>Refactoring, Martin Fowler</title>
		<link>http://blog.justoaguilar.com/2010/10/refactoring-martin-fowler/</link>
		<comments>http://blog.justoaguilar.com/2010/10/refactoring-martin-fowler/#comments</comments>
		<pubDate>Tue, 05 Oct 2010 23:25:59 +0000</pubDate>
		<dc:creator>justo</dc:creator>
				<category><![CDATA[clásico]]></category>
		<category><![CDATA[java]]></category>
		<category><![CDATA[lectura obligatoria]]></category>
		<category><![CDATA[libros]]></category>
		<category><![CDATA[productividad]]></category>
		<category><![CDATA[profesional]]></category>
		<category><![CDATA[reference]]></category>
		<category><![CDATA[best practices]]></category>
		<category><![CDATA[kent beck]]></category>
		<category><![CDATA[martin fowler]]></category>
		<category><![CDATA[patterns]]></category>
		<category><![CDATA[refactoring]]></category>

		<guid isPermaLink="false">http://blog.justoaguilar.com/?p=747</guid>
		<description><![CDATA[Sinopsis Editorial: As the application of object technology&#8211;particularly the Java programming language&#8211;has become commonplace, a new problem has emerged to confront the software development community. Significant numbers of poorly designed programs have been created by less-experienced developers, resulting in applications that are inefficient and hard to maintain and extend. Increasingly, software system professionals are discovering [...]]]></description>
			<content:encoded><![CDATA[
<!-- google_ad_section_start -->
<!--INFOLINKS_ON-->
<p><div style="float:right"><a href="http://www.amazon.com/exec/obidos/ASIN/0201485672/ref=nosim/blodejusagu-20" target="_blank"><img src="http://ecx.images-amazon.com/images/I/51T4YZ3HieL._BO2,204,203,TopRight,35,-76_AA160_SH20_OU01_.jpg" border="0" /></a></div>
<p>Sinopsis Editorial:</p>
<blockquote><p>
As the application of object technology&#8211;particularly the Java programming language&#8211;has become commonplace, a new problem has emerged to confront the software development community. Significant numbers of poorly designed programs have been created by less-experienced developers, resulting in applications that are inefficient and hard to maintain and extend. Increasingly, software system professionals are discovering just how difficult it is to work with these inherited, &#8220;non-optimal&#8221; applications. For several years, expert-level object programmers have employed a growing collection of techniques to improve the structural integrity and performance of such existing software programs. Referred to as &#8220;refactoring,&#8221; these practices have remained in the domain of experts because no attempt has been made to transcribe the lore into a form that all developers could use. . .until now. In Refactoring: Improving the Design of Existing Code, renowned object technology mentor Martin Fowler  breaks new ground, demystifying these master practices and demonstrating how software practitioners can realize the significant benefits of this new process.</p>
<p>&nbsp;</p>
<p>With proper training a skilled system designer can take a bad design and rework it into well-designed, robust code. In this book, Martin Fowler shows you where opportunities for refactoring typically can be found, and how to go about reworking a bad design into a good one. Each refactoring step is simple&#8211;seemingly too simple to be worth doing. Refactoring may involve moving a field from one class to another, or pulling some code out of a method to turn it into its own method, or even pushing some code up or down a hierarchy. While these individual steps may seem elementary, the cumulative effect of such small changes can radically improve the design. Refactoring is a proven way to prevent software decay.</p>
<p>&nbsp;</p>
<p>In addition to discussing the various techniques of refactoring, the author provides a detailed catalog of more than seventy proven refactorings with helpful pointers that teach you when to apply them; step-by-step instructions for applying each refactoring; and an example illustrating how the refactoring works. The illustrative examples are written in Java, but the ideas are applicable to any object-oriented programming language.
</p></blockquote>
<p>&nbsp;</p>
<p>El mero hecho de querer leer este libro o de haberlo leído ya te identifica como un profesional con unas inquietudes superiores a la media. Este libro es para mejorar tu código, para aprender a reescribir tu código para que sea más mantenible, más comprensible, más coherente y cohesionado. Habla de refactorizar, es decir, reescribir tu código para conseguir lo anterior sin añadir, cambiar o eliminar ninguna funcionalidad existente.</p>
<p>&nbsp;</p>
<p>Martin Fowler es una de las personalidades actuales en el campo de la ingeniería del software, un profesional que sienta cátedra con cada uno de sus escritos. Su página web es digna de leerse de principio a fin. Quizás este libro no destaque especialmente por su complejidad o profundidad, es un libro de buenas prácticas, de consejos para reescribir el código, sin embargo el nivel de profundidad al que llega es digno de agradecer. </p>
<p>&nbsp;</p>
<p>Valoración: <strong>IMPRESCINDIBLE</strong></p>
<p>&nbsp;</p>
<div id="referencias">
<b>Más información :</b></p>
<ul>
<li>Reseña &#8211; <a href="http://www.amazon.com/Refactoring-Improving-Design-Existing-Code/product-reviews/0201485672/ref=sr_1_1_cm_cr_acr_txt?ie=UTF8&#038;showViewpoints=1">Amazon</a></li>
<li>Refactoring &#8211; <a href="http://www.refactoring.com/">Página oficial</a></li>
<li>Martin Fowler &#8211; <a href="http://martinfowler.com/">Página oficial</a></li>
</ul>
</div>

<!--INFOLINKS_OFF-->
<!-- google_ad_section_end -->
]]></content:encoded>
			<wfw:commentRss>http://blog.justoaguilar.com/2010/10/refactoring-martin-fowler/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Clean Code, Robert C. Martin</title>
		<link>http://blog.justoaguilar.com/2010/08/clean-code-robert-c-martin/</link>
		<comments>http://blog.justoaguilar.com/2010/08/clean-code-robert-c-martin/#comments</comments>
		<pubDate>Sat, 28 Aug 2010 12:25:54 +0000</pubDate>
		<dc:creator>justo</dc:creator>
				<category><![CDATA[java]]></category>
		<category><![CDATA[lectura obligatoria]]></category>
		<category><![CDATA[libros]]></category>
		<category><![CDATA[profesional]]></category>
		<category><![CDATA[reference]]></category>
		<category><![CDATA[bad smells]]></category>
		<category><![CDATA[best practices]]></category>
		<category><![CDATA[patterns]]></category>
		<category><![CDATA[programming]]></category>
		<category><![CDATA[refactoring]]></category>
		<category><![CDATA[robert c. martin]]></category>
		<category><![CDATA[tdd]]></category>
		<category><![CDATA[testing]]></category>

		<guid isPermaLink="false">http://blog.justoaguilar.com/?p=678</guid>
		<description><![CDATA[Una de las habilidades más importantes con la que debe contar todo desarrollador de software es saber escribir código que otros puedan entender, incluso que uno mismo sea capaz de entender meses después de escribirlo. Después de unos años trabajando en esto, he leído mucho código y podría decir que el nivel medio es mediocre, [...]]]></description>
			<content:encoded><![CDATA[
<!-- google_ad_section_start -->
<!--INFOLINKS_ON-->
<p><div style="padding-left:20px;float:right"><a href="http://www.amazon.com/Clean-Code-Handbook-Software-Craftsmanship/dp/0132350882/ref=sr_1_1?s=books&#038;ie=UTF8&#038;qid=1282992360&#038;sr=1-1" target="_blank"><img src="http://ecx.images-amazon.com/images/I/41znMZniZ1L._BO2,35,-76_AA160_SH20_OU01_.jpg" border="0" /></a></div>
<p>Una de las habilidades más importantes con la que debe contar todo desarrollador de software es saber escribir código que otros puedan entender, incluso que uno mismo sea capaz de entender meses después de escribirlo. Después de unos años trabajando en esto, he leído mucho código y podría decir que el nivel medio es mediocre, incluso bajo me atrevería a decir; por suerte, hay excepciones. Probablemente, yo no sea el mejor ejemplo, pero sí que me preocupo por mejorar esta carencia y me alegra reconocer que hoy escribo mejor código que ayer.</p>
<p>&nbsp;</p>
<p><strong>Clean Code: A Handbook of Agile Software Craftsmanship</strong> debe ser uno de esos libros que debes comprar cuando acabes la carrera, o cuando empieces a trabajar. Intenta introducirte lo antes posible en el desempeño de buenas prácticas, incorpóralas a tus hábitos de trabajo e intenta convencer al resto del equipo para que las adopte. <em>Intenta comprender</em> porqué es mejor hacer algo de una forma que de otra, así dispondrás de argumentos que defiendan objetivamente un cambio y darás pie a un coloquio del que todos saldréis ganando, todos.</p>
<p>&nbsp;</p>
<p>Este libro se divide en 17 capítulos y 3 apéndices. Cada capítulo habla de un aspecto muy concreto, exponiendo los principales errores y su solución más adecuada. Aunque algunos capítulos puedan dar la sensación, a priori, de poco interesantes por su trivialidad, nada más lejos de la realidad. No juzgues nada antes de conocerlo. Aspectos, tan poco tenidos en cuenta, como los <em>comentarios</em>, los <em>nombres</em> o el <em>formato</em> son examinados al detalle, así como otros más habituales: <em>funciones</em>, <em>clases</em>, <em>excepciones</em> o <em>tests</em>. Si eres un desarrollador experimentado reconocerás, en mayor o menor medida, las casuísticas que describe el libro, ello no quita que sirva como refresco de conocimientos.</p>
<p>&nbsp;</p>
<p>Valoración: <strong>LECTURA OBLIGATORIA</strong></p>
<p>&nbsp;</p>
<div id="referencias">
<b>Vídeos</b></p>
<p><iframe src="http://player.vimeo.com/video/13439458?portrait=0" width="640" height="240" frameborder="0"></iframe></p>
<p>&nbsp;</p>
<p><iframe src="http://player.vimeo.com/video/13702091?portrait=0" width="640" height="240" frameborder="0"></iframe>
</div>
<p>&nbsp;</p>
<div id="referencias">
<b>Más información :</b></p>
<ul>
<li>Reseña &#8211; <a href="http://www.amazon.com/Clean-Code-Handbook-Software-Craftsmanship/product-reviews/0132350882/ref=dp_top_cm_cr_acr_txt?ie=UTF8&#038;showViewpoints=1">Amazon</a></li>
<li>Robert C. Martin (<em>Uncle Bob</em>) &#8211; <a href="http://www.objectmentor.com/omTeam/martin_r.html">Object Mentor CEO</a></li>
<li>Robert C. Martin &#8211; <a href="http://en.wikipedia.org/wiki/Robert_Cecil_Martin">Wikipedia</a></li>
</ul>
</div>

<!--INFOLINKS_OFF-->
<!-- google_ad_section_end -->
]]></content:encoded>
			<wfw:commentRss>http://blog.justoaguilar.com/2010/08/clean-code-robert-c-martin/feed/</wfw:commentRss>
		<slash:comments>3</slash:comments>
		</item>
		<item>
		<title>Bundle Manifest Headers, Cabeceras OSGi</title>
		<link>http://blog.justoaguilar.com/2009/08/bundle-manifest-headers-cabeceras-osgi/</link>
		<comments>http://blog.justoaguilar.com/2009/08/bundle-manifest-headers-cabeceras-osgi/#comments</comments>
		<pubDate>Sat, 01 Aug 2009 15:07:44 +0000</pubDate>
		<dc:creator>justo</dc:creator>
				<category><![CDATA[java]]></category>
		<category><![CDATA[profesional]]></category>
		<category><![CDATA[reference]]></category>
		<category><![CDATA[manifest]]></category>
		<category><![CDATA[osgi]]></category>

		<guid isPermaLink="false">http://blog.justoaguilar.com/?p=437</guid>
		<description><![CDATA[Una parte fundamental dentro de la especificación OSGi es el conjunto de metadatos que deben ser situados en el fichero MANIFEST.MF y que permitirán a la implementanción OSGi instalar el bundle dentro del entorno de ejecución, resolver sus dependencias y publicar todo aquello que queramos sea visto por otros bundles. Me voy a basar en [...]]]></description>
			<content:encoded><![CDATA[
<!-- google_ad_section_start -->
<!--INFOLINKS_ON-->
<p><p>Una parte fundamental dentro de la especificación <em>OSGi</em> es el conjunto de metadatos que deben ser situados en el fichero <em>MANIFEST.MF</em> y que permitirán a la implementanción <em>OSGi</em> instalar el <em>bundle</em> dentro del entorno de ejecución, resolver sus dependencias y publicar todo aquello que queramos sea visto por otros <em>bundles</em>.</p>
<p>Me voy a basar en el documento que define la especificación <em>OSGi 4.1</em> para listar todas las cabeceras disponibles actualmente, su finalidad y sus características.</p>
<p>La lista de cabeceras, por orden alfabético, es la siguiente:</p>
<ul>
<li><strong><a href="#Bundle-ActivationPolicy">Bundle-ActivationPolicy</a></strong></li>
<li><strong><a href="#Bundle-Activator">Bundle-Activator</a></strong></li>
<li><strong><a href="#Bundle-Category">Bundle-Category</a></strong></li>
<li><strong><a href="#Bundle-Classpath">Bundle-Classpath</a></strong></li>
<li><strong><a href="#Bundle-ContactAddress">Bundle-ContactAddress</a></strong></li>
<li><strong><a href="#Bundle-Copyright">Bundle-Copyright</a></strong></li>
<li><strong><a href="#Bundle-Description">Bundle-Description</a></strong></li>
<li><strong><a href="#Bundle-DocURL">Bundle-DocURL</a></strong></li>
<li><strong><a href="#Bundle-Localization">Bundle-Localization</a></strong></li>
<li><strong><a href="#Bundle-ManifestVersion">Bundle-ManifestVersion</a></strong></li>
<li><strong><a href="#Bundle-Name">Bundle-Name</a></strong></li>
<li><strong><a href="#Bundle-NativeCode">Bundle-NativeCode</a></strong></li>
<li><strong><a href="#Bundle-RequiredExecutionEnvironment">Bundle-RequiredExecutionEnvironment</a></strong></li>
<li><strong><a href="#Bundle-SymbolicName">Bundle-SymbolicName</a></strong></li>
<li><strong><a href="#Bundle-UpdateLocation">Bundle-UpdateLocation</a></strong></li>
<li><strong><a href="#Bundle-Vendor">Bundle-Vendor</a></strong></li>
<li><strong><a href="#Bundle-Version">Bundle-Version</a></strong></li>
<li><strong><a href="#DynamicImport-Package">DynamicImport-Package</a></strong></li>
<li><strong><a href="#Export-Package">Export-Package</a></strong></li>
<li><strong><a href="#Export-Service">Export-Service</a></strong></li>
<li><strong><a href="#Fragment-Host">Fragment-Host</a></strong></li>
<li><strong><a href="#Import-Package">Import-Package</a></strong></li>
<li><strong><a href="#Import-Service">Import-Service</a></strong></li>
<li><strong><a href="#Require-Bundle">Require-Bundle</a></strong></li>
</ul>
<div id="titulo">
Descripción de cabeceras
</div>
<p><a name="Bundle-ActivationPolicy"></a><br />
<strong>Bundle-ActivationPolicy</strong><br />
Especifica como el <em>framework</em> debe activar el <em>bundle</em>. Si debe hacerlo de forma normal cuando arranca el <em>framework</em> o de forma perezosa, no se activará hasta que no sea cargada una clase.<br />
<em>Valores permitidos</em> : lazy<br />
<em>Valor por defecto</em> : ninguno. Si no se incluye esta cabecera se activará el <em>bundle</em> al arranque del framework.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-ActivationPolicy : lazy
</pre>
<p><a name="Bundle-Activator"></a><br />
<strong>Bundle-Activator</strong><br />
Nombre de la clase usada para arrancar y parar el <em>bundle</em>. La clase debe implementar la interfaz <em>BundleActivator</em>.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-Activator : es.jmac.osgi.infrastructure.Activator
</pre>
<p><a name="Bundle-Category"></a><br />
<strong>Bundle-Category</strong><br />
Lista de nombres de categoría, separadas por comas.<br />
<em>Valores permitidos</em> : libre, valores separados por comas. <em>OSGi</em> recomienda utilizar los valores de una <a href="http://www.osgi.org/Specifications/Reference#categories">lista predefinida</a>.<br />
<em>Valor por defecto</em> : ninguno. Si no se incluye esta cabecera se activará el <em>bundle</em>al arranque del <em>framework</em>.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-Category : example, device, testing
</pre>
<p><a name="Bundle-Classpath"></a><br />
<strong>Bundle-Classpath</strong><br />
Define un lista, separada por comas, de rutas de ficheros <em>JAR</em> y directorios (dentro del <em>bundle</em>) con las clases y recursos del <em>bundle</em>. El &#8220;.&#8221; especifica el directorio raiz del <em>JAR</em> del bundle.<br />
<em>Valor por defecto</em> : .<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-Classpath : lib/commons-logging-1.1.1.jar, . , resources/html
</pre>
<p><a name="Bundle-ContactAddress"></a><br />
<strong>Bundle-ContactAddress</strong><br />
Dirección de contacto del proveedor.</p>
<p><a name="Bundle-Copyright"></a><br />
<strong>Bundle-Copyright</strong><br />
Especificación de <em>copyright</em> y licencia para el <em>bundle</em>.</p>
<p><a name="Bundle-Description"></a><br />
<strong>Bundle-Description</strong><br />
Descripción corta del <em>bundle</em>.</p>
<p><a name="Bundle-DocURL"></a><br />
<strong>Bundle-DocURL</strong><br />
URL con la documentación del <em>bundle</em>.</p>
<p><a name="Bundle-Localization"></a><br />
<strong>Bundle-Localization</strong><br />
Esta cabecera debe contener el fichero base dentro del <em>bundle</em>que defines las información multi-idioma.<br />
<em>Valor por defecto</em> : OSGI-INF/l10n/bundle<br />
Esta ruta la resuelve el framework como OSGI-INF/l10n/bundle.properties, por tanto cada fichero de idioma debe tener el siguiente nombre : bundle_es.properties, bundle_de.propertie<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-Localization : resources/labels
</pre>
<p><a name="Bundle-ManifestVersion"></a><br />
<strong>Bundle-ManifestVersion</strong><br />
Define que versión de la especificación <em>OSGi</em> debe seguir el <em>bundle</em>.<br />
La versión 1 del Manifest es para la release 3.<br />
La versión 2 es para la release 4, la actual.<br />
<em>Valor por defecto</em> : 1<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-ManifestVersion : 2
</pre>
<p><a name="Bundle-Name"></a><br />
<strong>Bundle-Name</strong><br />
Nombre corto para el bundle. Puede contener espacios.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-Name : Pruebas OSGi
</pre>
<p><a name="Bundle-NativeCode"></a><br />
<strong>Bundle-NativeCode</strong><br />
Esta cabecera contiene una especificación de las librerías de código nativo contenidas en el <em>bundle</em>.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-NativeCode : /lib/http.DLL; osname = QNX; osversion = 3.1
</pre>
<p><a name="Bundle-RequiredExecutionEnvironment"></a><br />
<strong>Bundle-RequiredExecutionEnvironment</strong><br />
Contiene una lista, separada por comas, de los entornos de ejecución que deben estar presentes en la Plataforma de Servicio.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-RequiredExecutionEnvironment : CDC-1.0/Foundation-1.0
</pre>
<p><a name="Bundle-SymbolicName"></a><br />
<strong>Bundle-SymbolicName</strong><br />
Especifica un nombre único, dentro de la <em>VM</em>, para el <em>bundle</em>. El nombre debería seguir la convención de dominio inverso. <strong>Esta cabecera es obligatoria</strong>.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-SymbolicName : es.jmac.osgi.infrastructure
</pre>
<p><a name="Bundle-UpdateLocation"></a><br />
<strong>Bundle-UpdateLocation</strong><br />
Especifica la URL a partir de la cual se recuperarán las actualizaciones para el <em>bundle</em>. Si el <em>bundle</em> es actualizado, esta ruta debería ser usada si está presente, para recuperar el fichero <em>JAR</em> actualizado.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-UpdateLocation : https://justoaguilar.com/repo/osgi
</pre>
<p><a name="Bundle-Vendor"></a><br />
<strong>Bundle-Vendor</strong><br />
Descripción del proveedor del <em>bundle</em>.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-Vendor : Justo Aguilar
</pre>
<p><a name="Bundle-Version"></a><br />
<strong>Bundle-Version</strong><br />
Especifica la versión del <em>bundle</em>. La versión debe seguir obligatoriamente el formato :<br />
<code><br />
version ::=<br />
major( '.' minor ( '.' micro ( '.' qualifier )? )? )?<br />
major ::= number                   // See 1.3.2<br />
minor ::= number<br />
micro ::= number<br />
qualifier ::= ( alphanum | ’_’ | '-' )+<br />
</code><br />
<em>Valor por defecto</em> : 0.0.0<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Bundle-Version : 1.0.1.Built20090801-143221
</pre>
<p><a name="DynamicImport-Package"></a><br />
<strong>DynamicImport-Package</strong><br />
Esta cabecera contiene una lista, separada por comas, de nombres de paquetes que deberían ser dinámicamente importados cuando se necesiten.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
DynamicImport-Package : es.jmac.osgi.*
</pre>
<p><a name="Export-Package"></a><br />
<strong>Export-Package</strong><br />
Contiene la declaración de los paquetes a exportar.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Export-Package : !es.jmac.osgi.infrastructure, es.jmac.osgi
</pre>
<p><a name="Export-Service"></a><br />
<strong>Export-Service</strong><br />
Obsoleto. (Deprecated)</p>
<p><a name="Fragment-Host"></a><br />
<strong>Fragment-Host</strong><br />
Define el <em>bundle</em>  maestro para este fragmento.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Fragment-Host : es.jmac.osgi.infrastructure;version=&quot;[1.0.0,2.0.0)&quot;
</pre>
<p><a name="Import-Package"></a><br />
<strong>Import-Package</strong><br />
Especifica los paquetes a importar por este <em>bundle</em>.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Export-Package : !es.jmac.osgi.infrastructure, es.jmac.osgi
</pre>
<p><a name="Import-Service"></a><br />
<strong>Import-Service</strong><br />
Obsoleto. (Deprecated)</p>
<p><a name="Require-Bundle"></a><br />
<strong>Require-Bundle</strong><br />
Especifica la lista de <em>bundles</em> (paquetes) obligatorios a utilizar en este <em>bundle</em>.<br />
<em>Ejemplo</em></p>
<pre class="brush: java; title: ; notranslate">
Require-Bundle : org.apache.commons.logging;version=&quot;1.1.1&quot;,org.springframework.context;version=&quot;(2.0.0,3.0.0)&quot;
</pre>
<div id="referencias">
<strong>Referencias:</strong></p>
<ul>
<li><a href="http://www.osgi.org/Release4/HomePage">OSGi Specification</a></li>
<li><a href="http://blog.justoaguilar.com/2009/08/que-es-osgi-y-para-que-sirve/">¿Qué es OSGi?</a></li>
</ul>
</div>

<!--INFOLINKS_OFF-->
<!-- google_ad_section_end -->
]]></content:encoded>
			<wfw:commentRss>http://blog.justoaguilar.com/2009/08/bundle-manifest-headers-cabeceras-osgi/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>¿Qué es OSGi &#8230; y para que sirve?</title>
		<link>http://blog.justoaguilar.com/2009/08/que-es-osgi-y-para-que-sirve/</link>
		<comments>http://blog.justoaguilar.com/2009/08/que-es-osgi-y-para-que-sirve/#comments</comments>
		<pubDate>Fri, 31 Jul 2009 22:28:03 +0000</pubDate>
		<dc:creator>justo</dc:creator>
				<category><![CDATA[java]]></category>
		<category><![CDATA[profesional]]></category>
		<category><![CDATA[apache felix]]></category>
		<category><![CDATA[equinox]]></category>
		<category><![CDATA[osgi]]></category>

		<guid isPermaLink="false">http://blog.justoaguilar.com/?p=391</guid>
		<description><![CDATA[Pues sí, otro acrónimo más en el mundo Java, no sé por qué te extrañas. Y como ha pasado con muchos otros de su época, véase SOAP, el acrónimo ha pasado a ser el nombre oficial, nombre que en sus orígenes era Open Services Gateway initiative. Bueno, vamos al turrón. OSGi es la respuesta en [...]]]></description>
			<content:encoded><![CDATA[
<!-- google_ad_section_start -->
<!--INFOLINKS_ON-->
<p><p>Pues sí, otro acrónimo más en el mundo Java, no sé por qué te extrañas. Y como ha pasado con muchos otros de su época, véase <em><a href="http://en.wikipedia.org/wiki/SOAP">SOAP</a></em>, el acrónimo ha pasado a ser el nombre oficial, nombre que en sus orígenes era <em><strong>O</strong>pen <strong>S</strong>ervices <strong>G</strong>ateway <strong>i</strong>nitiative</em>. Bueno, vamos al turrón.</p>
<div id="nota">
<strong>OSGi es la respuesta en la plataforma Java a la programación modular.</strong>
</div>
<div id="titulo">¿Qué se entiende por módulo?</div>
<p>Un módulo es un componente autocontenido dentro de un sistema mayor. ¿Autocontenido?, sí, que no necesita de referencias externas para su existencia y funcionamiento.</p>
<p>Bien, detrás de esta definición de módulo tan abstracta se esconden dos de los principios fundamentales de la programación : Alta Cohesión y Bajo Acoplamiento.</p>
<p>Todo módulo debe tener una responsabilidad y funcionalidad perfectamente delimitada dentro de un sistema mayor en el que se encuentre integrado (alta cohesión). Y debe evitar cualquier dependencia con el resto de módulos para realizar la tarea que tiene asignada (bajo acoplamiento). Evidentemente, siempre existirá un determinado grado de acoplamiento, que debe estar, en la medida de lo posible, articulado mediante conceptos abstractos, lo que en <em>Java</em> vienen a ser las <em>interfaces</em>. </p>
<div id="titulo">¿Cómo gestiona <em>Java</em> actualmente la modularización?</div>
<p>Se podría decir que dentro de la plataforma <em>Java</em> la analogía de módulo sería un fichero <em>JAR</em>. Sin embargo, si conocemos el proceso mediante el cual es cargado un fichero <em>jar</em> en la <em>JVM</em>, podemos advertir una serie de carencias que invitan a pensar que la modularización en <em>Java</em> es una mera ilusión.</p>
<p>Actualmente, por defecto, una <em>JVM</em> tiene un único <em>class loader</em> (<em>system class loader</em> que es cargado por el <em>bootstrap class loader</em> escrito en código nativo) responsable de cargar en la <em>JVM</em> cualquier clase que se encuentre dentro del <em>classpath</em> de ejecución de una aplicación <em>Java</em>. Cuando existe un fichero jar en el classpath, se carga cada una de las clases que contenga ese <em>JAR</em>. Todo se carga en un espacio común, ahí, al montón. </p>
<p>Seguramente alguna vez habréis cargado el mismo jar dos o más veces, si es de la misma versión no pasa nada pero si usas distintas versiones puede que tengas un problema. Si tienes dependencias con diferentes versiones de un mismo <em>JAR</em>, que sepas que se cargará la clase del jar que aparezca primero en el <em>classpath</em>. Esto implica que no podamos usar diferentes versiones de una misma clase, al menos, con la aproximación que <em>Java</em> tiene por defecto, tendríamos que crear nuestro propio <em>classloader</em> para cargar una versión distinta de la misma clase.</p>
<p>[Extender este punto]</p>
<div id="titulo">Propuesta <em>OSGi</em></div>
<p>La <a href="http://www.osgi.org/About/HomePage">Alianza <em>OSGi</em></a> empezó a definir allá por el año 1999 (diez añitos ya) una especificación que permitiera la construcción de aplicaciones modulares en Java.<br />
<center><br />
<img src="http://www.osgi.org/wiki/uploads/About/layering-osgi.png" /><br />
</center></p>
<p>La especificación <em>OSGi</em> propone una arquitectura montada sobre una JVM que permita la instalación, actualización y eliminación de módulos. Cada módulo podrá publicar las clases que serán visibles por otros módulos de la máquina virtual y podrá consumir cualquier clase que haya sido publicada por otros módulos de la VM. Entre los módulos cliente y proveedor existe un registro central, en el que los proveedores registran los servicios que desean hacer públicos al resto de módulos y los clientes solicitan los servicios que desean consumir. Evidentemente un módulo puede ser, a la vez, cliente y proveedor. Este paradigma es similar a la solución definida en SOA, por ello se dice que:   </p>
<div id="nota">
<strong>OSGi = SOA en una JVM</strong>
</div>
<p>Dentro de la especificación <em>OSGi</em>, un módulo tiene el nombre de bundle. Un bundle <em>OSGi</em> es un fichero <em>JAR</em> con un conjunto de metadatos que especifican las características de <em>bundle</em>. Estos metadatos deben de incluidos dentro del fichero <em>MANIFEST.MF</em>, por tanto : </p>
<div id="nota">
<strong>Bundle OSGi = JAR + MANIFEST.MF con METADATOS OSGi</strong>
</div>
<p>Un ejemplo de <em>MANIFEST.MF</em> adaptado para ser utilizado dentro de un entorno <em>OSGi</em> puede ser el siguiente :</p>
<pre class="brush: java; title: ; notranslate">
Manifest-Version: 1.0
Export-Package: org.apache.commons.logging;version=&quot;1.1.1&quot;,org.apache.
 commons.logging.impl;version=&quot;1.1.1&quot;;uses:=&quot;javax.servlet,org.apache.
 avalon.framework.logger,org.apache.commons.logging,org.apache.log,org
 .apache.log4j&quot;
Implementation-Title: Jakarta Commons Logging
Implementation-Version: 1.1.1
Bundle-Classpath: .
Built-By: dlg01
Specification-Vendor: Apache Software Foundation
Bundle-Name: Apache Commons Logging
Created-By: Apache Maven
X-Compile-Source-JDK: 1.2
Implementation-Vendor: Apache Software Foundation
Bundle-Vendor: SpringSource
Implementation-Vendor-Id: org.apache
Build-Jdk: 1.4.2_16
Bundle-Version: 1.1.1
Specification-Title: Jakarta Commons Logging
Bundle-ManifestVersion: 2
Import-Package: javax.servlet;version=&quot;[2.1.0, 3.0.0)&quot;;resolution:=opt
 ional,org.apache.avalon.framework.logger;version=&quot;[4.1.3, 4.1.3]&quot;;res
 olution:=optional,org.apache.log;version=&quot;[1.0.1, 1.0.1]&quot;;resolution:
 =optional,org.apache.log4j;version=&quot;[1.2.15, 2.0.0)&quot;;resolution:=opti
 onal
Bundle-SymbolicName: com.springsource.org.apache.commons.logging
Specification-Version: 1.0
Extension-Name: org.apache.commons.logging
Archiver-Version: Plexus Archiver
X-Compile-Target-JDK: 1.2
</pre>
<p>El código anterior correponde al fichero <em>MANIFEST.MF</em> contenido en el <em>JAR</em> de <em>Apache Commons Logging</em> 1.1.1 descargado del repositorio <em>OSGi</em> de <a href=" http://www.springsource.com/repository/app/bundle/version/detail?name=com.springsource.org.apache.commons.logging&#038;version=1.1.1&#038;searchType=bundlesByName&#038;searchQuery=logging">SpringSource</a>.</p>
<p>De todas las cabeceras (<em>headers</em>) que componen el fichero vamos a explicar las más significativas:</p>
<ul>
<li><strong>Bundle-SymbolicName</strong> : Única cabecera obligatoria y que debe identificar unívocamente dentro de la <em>VM</em> al <em>bundle</em>.</li>
<li><strong>Bundle-Version</strong>: Versión del <em>bundle</em>. La dupla (Bundle-SymbolycName,BundleVersion) permite referenciar una versión de un <em>bundle</em>.</li>
<li><strong>Export-Package</strong> : Lista de paquetes que el <em>bundle</em> publica para que puedan ser consumidos por otros <em>bundles</em> dentro de la misma <em>VM</em>.</li>
<li><strong>Import-Package</strong> : Lista de paquetes que el <em>bundle</em> necesita consumir y que por tanto deben ser exportados por otros <em>bundles</em>.</li>
<li><strong>Bundle-Classpath</strong> : Lista de rutas donde se encuentran las clases del <em>bundle</em></li>
</ul>
<p>Como habréis notado, todas las referencias a paquetes van acompañadas de un rango de versiones, no es obligatorio pero si recomendable. Por ejemplo, este <em>bundle</em> consume, de manera opcional, el paquete <em>javax.servlet</em> desde la versión 2.1.0 en adelante y menor a la 3.0.0. Esto permite a un <em>bundle</em> importar paquetes de varias versiones de un <em>bundle</em>, lo que puede ser necesario en ocasiones cuando tengamos dependencias con sistemas antiguos y que tengan dependencias con versiones antiguas mientras que la parte nueva del desarrollo esté vinculada a las versiones más modernas. </p>
<p>Otra ventaja es la posibilidad publicar únicamente lo que quiero que sea público a otros <em>bundles</em> sin necesidad de utilizar modificadores de ámbito a nivel de código como <em>private</em>, <em>protected</em> o <em>package</em>. </p>
<p>Por último os dejo con la definición que <em>Craig Walls</em> hace en su libro, <em>Modular Java</em>.</p>
<blockquote><p>
OSGi is a component framework speciﬁcation that brings modularity to<br />
the Java platform. OSGi enables the creation of highly cohesive, loosely<br />
coupled modules that can be composed into larger applications. What’s<br />
more, each module can be individually developed, tested, deployed,<br />
updated, and managed with minimal or no impact to the other modules.</p>
<p>Craig Walls,<br />
Modular Java
</p></blockquote>
<p>Próximamente más.</p>
<div id="referencias">
<b>Referencias:</b></p>
<ul>
<li><a href="http://www.amazon.com/Modular-Java-Applications-Pragmatic-Programmers/dp/1934356409/ref=sr_1_1?ie=UTF8&#038;s=books&#038;qid=1249077129&#038;sr=8-1">Modular Java: Creating Flexible Applications with OSGi and Spring</a>, Craig Walls</li>
<li><a href="http://www.osgi.org/Specifications/HomePage">OSGi Specification</a></li>
<li><a href="http://static.springsource.org/osgi/docs/current/reference/html-single/">Spring DM Reference</a></li>
</ul>
</div>

<!--INFOLINKS_OFF-->
<!-- google_ad_section_end -->
]]></content:encoded>
			<wfw:commentRss>http://blog.justoaguilar.com/2009/08/que-es-osgi-y-para-que-sirve/feed/</wfw:commentRss>
		<slash:comments>2</slash:comments>
		</item>
		<item>
		<title>Configuración Equinox, OSGi, Spring DM y Maven</title>
		<link>http://blog.justoaguilar.com/2009/07/configuracion-osgi-spring-dm-y-maven/</link>
		<comments>http://blog.justoaguilar.com/2009/07/configuracion-osgi-spring-dm-y-maven/#comments</comments>
		<pubDate>Mon, 20 Jul 2009 22:00:17 +0000</pubDate>
		<dc:creator>justo</dc:creator>
				<category><![CDATA[java]]></category>
		<category><![CDATA[profesional]]></category>
		<category><![CDATA[eclipse]]></category>
		<category><![CDATA[equinox]]></category>
		<category><![CDATA[howto]]></category>
		<category><![CDATA[maven]]></category>
		<category><![CDATA[maven-bundle-plugin]]></category>
		<category><![CDATA[osgi]]></category>
		<category><![CDATA[spring]]></category>
		<category><![CDATA[spring dm]]></category>

		<guid isPermaLink="false">http://blog.justoaguilar.com/?p=350</guid>
		<description><![CDATA[Actualmente estoy trabajando en un proyecto en el que OSGi va a jugar un papel determinante. En próximos posts hablaré sobre OSGi, y sobre que papel juega dentro de las aplicaciones Java. Cuando se habla de OSGi, se habla de programación modular donde un módulo es un fichero JAR (Java ARchive). En OSGi un módulo [...]]]></description>
			<content:encoded><![CDATA[
<!-- google_ad_section_start -->
<!--INFOLINKS_ON-->
<p><p>Actualmente estoy trabajando en un proyecto en el que <a href="http://www.osgi.org"><em>OSGi</em></a> va a jugar un papel determinante. En próximos <em>posts</em> hablaré sobre <em>OSGi</em>, y sobre que papel juega dentro de las aplicaciones <em>Java</em>. </p>
<p>Cuando se habla de <em>OSGi</em>, se habla de programación modular donde un módulo es un fichero <em>JAR</em> (<em>Java ARchive</em>). En <em>OSGi</em> un módulo tiene el nombre de <em>bundle</em>.</p>
<p>Para poder entender este <em>post</em> se requieren conocimientos de :</p>
<ul>
<li>Maven 2</li>
<li>Spring Framework</li>
<li>Eclipse, Equinox</li>
<li>OSGi</li>
<li>Spring Dynamic Modules  (Spring DM)</li>
</ul>
<p>La prueba de concepto que vamos a describir a continuación, paso a paso, es la siguiente:</p>
<ul>
<li><strong>Crear</strong> un <em><strong>bundle</strong></em> e instalarlo en Equinox. </li>
<li><strong>Asociar</strong> al <em>bundle</em> una clase <em><strong>BundleActivator</strong></em> para comprobar que el <em>bundle</em> ha sido inicializado correctamente.</li>
<li><strong>Instalar</strong> el <em>bundle</em> <strong><strong>Extender</strong></strong> de <em><strong>Spring DM</strong></em> y sus dependencias dentro de <em>Equinox</em>.</li>
<li><strong>Configurar</strong> el <em>bundle</em> para que arranque un <em><strong>ApplicationContext</strong></em> de <em>Spring</em>.</li>
</ul>
<p>La idea de este <em>post</em> es mostrar de manera rápida el proceso de configuración, por ello no voy a explicar detalladamente el conjunto de herramientas y <em>plugins</em> utilizados a lo largo del proceso. Esto quedará pendiente para próximos <em>posts</em>.</p>
<div id="titulo">1. Creación de un <em>bundle OSGi</em></div>
<p><strong>Creamos</strong> un nuevo <strong>proyecto Java</strong> dentro de nuestro espacio de trabajo de <em>Eclipse</em>. En mi caso, <em>es.jmac.springdm</em> </p>
<p><strong>Mavenizamos</strong> ese proyecto, yo he usado el <em>plugin</em> de <em>Sonatype</em> <strong><a target="_blank" href="http://www.sonatype.com/products/m2eclipse">M2Eclipse</a></strong>, pero vamos puede hacerse a mano, basta crear un fichero <em>pom.xml</em>.</p>
<p>La información del fichero Maven de configuración del proyecto será la siguiente : </p>
<pre class="brush: xml; title: ; notranslate">
&lt;project xmlns=&quot;http://maven.apache.org/POM/4.0.0&quot; xmlns:xsi=&quot;http://www.w3.org/2001/XMLSchema-instance&quot; xsi:schemaLocation=&quot;http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd&quot;&gt;

  &lt;modelVersion&gt;4.0.0&lt;/modelVersion&gt;
  &lt;groupId&gt;es.jmac&lt;/groupId&gt;
  &lt;artifactId&gt;springdm&lt;/artifactId&gt;
  &lt;packaging&gt;jar&lt;/packaging&gt;
  &lt;name&gt;springdm&lt;/name&gt;
  &lt;version&gt;1.0.0-SNAPSHOT&lt;/version&gt;
  &lt;description&gt;Prueba de concepto de Spring Dynamic Modules sobre Equinox&lt;/description&gt;

&lt;/project&gt;
</pre>
<p><em>Creamos</em> una nueva <em>source folder</em>, <strong>src/main/java</strong>.</p>
<p>Al final del proceso, nuestro proyecto en Eclipse debería tener una estructura similar a la siguiente:</p>
<div id="attachment_376" class="wp-caption aligncenter" style="width: 310px"><a href="http://blog.justoaguilar.com/wp-content/uploads/2009/07/estructura_inicial_proyecto.jpg"><img src="http://blog.justoaguilar.com/wp-content/uploads/2009/07/estructura_inicial_proyecto-300x176.jpg" alt="Estructura inicial del proyecto" title="Estructura inicial del proyecto" width="300" height="176" class="size-medium wp-image-376" /></a><p class="wp-caption-text">Estructura inicial del proyecto</p></div>
<p>Una vez que tenemos nuestro proyecto <em>Java</em> <em>mavenizado</em> vamos a incorporarle la información necesaria que lo identifique como un <em>bundle OSGi</em>.</p>
<p>Al inicio del <em>post</em> se ha comentado que un <em>bundle OSGi</em> es un fichero <em>JAR</em>. Sin embargo no basta con eso. Ese fichero <em>JAR</em> debe contener un información concreta que permita a la plataforma <em>OSGi</em> identificar ese fichero como un <em>bundle OSGi</em>. Esta información viene especificada en el fichero <em>MANIFEST.MF</em>. Esa será nuestra siguiente tarea generar ese fichero con la información necesaria para ser identificado como un <em>bundle OSGi</em>.</p>
<p></p>
<div id="nota">
Bundle OSGi = JAR + MANIFEST.MF con CABECERAS OSGi
</div>
<p></p>
<p>Bueno, ahora hay que generar el <em>MANIFEST.MF</em>. Hay varias formas de hacerlo, se puede hacer mano, si se conocen el conjunto de cabeceras <em>OSGi</em> o bien a través de alguna herramienta. En mi caso he utilizado un plugin para <em>Maven</em> que permite realizar varias operaciones sobre <em>bundles OSGI</em>, <a target="_blank" href="http://felix.apache.org/site/apache-felix-maven-bundle-plugin-bnd.html">maven-bundle-plugin</a>. Como he comentado anteriormente, no voy a explicar como funciona el plugin, os indico que hay que poner en el <em>pom.xml</em> y la orden que hay que lanzar.</p>
<p>Actualizar el pom.xml con el siguiente código : </p>
<pre class="brush: xml; title: ; notranslate">
...
&lt;packaging&gt;bundle&lt;/packaging&gt;
...
&lt;build&gt;
  &lt;plugins&gt;
    &lt;plugin&gt;
      &lt;groupId&gt;org.apache.felix&lt;/groupId&gt;
      &lt;artifactId&gt;maven-bundle-plugin&lt;/artifactId&gt;
      &lt;version&gt;2.0.0&lt;/version&gt;
      &lt;extensions&gt;true&lt;/extensions&gt;
      &lt;configuration&gt;
        &lt;instructions&gt;
	  &lt;Bundle-SymbolicName&gt;${groupId}.${artifactId}&lt;/Bundle-SymbolicName&gt;
	  &lt;Bundle-Version&gt;${pom.version}&lt;/Bundle-Version&gt;
	&lt;/instructions&gt;
      &lt;/configuration&gt;
    &lt;/plugin&gt;
  &lt;/plugins&gt;
&lt;/build&gt;
...
</pre>
<p><strong>ATENCIÓN</strong> : No olvidéis cambiar el valor de elemento <em><em>packaging</em></em> a <em><em>bundle</em></em>.</p>
<p>Ejecutad por consola o a través de <em>Eclipse</em>, el siguiente comando <em>Maven</em> :</p>
<pre class="brush: bash; title: ; notranslate">

mvn clean package -Declipse.pde
</pre>
<p>Esta orden generará un fichero <em>JAR</em> con nuestro proyecto, que por ahora no tiene nada, y que contendrá un fichero <em>MANIFEST.MF</em> con las cabeceras necesarias para ser reconocido como un <em>bundle OSGi</em>. Lo que tenéis que hacer es iros a la carpeta target donde se ha copiado el fichero <em>JAR</em> y sacar de él fichero <em>MANIFEST.MF</em> para copiarlo a vuestro proyecto en <em>Eclipse</em>, ojo, dentro de la carpeta <em>META-INF</em>. Ya sé que no es muy sutil, pero es rápido.</p>
<div style="padding:5px;border:1px solid #ccc;background-color:#fafafa">
<strong>ACTUALIZACIÓN</strong><br />
Una forma más automática de generar el fichero <em>MANIFEST.MF</em> sería utilizando el <em>goal <strong>manifest</strong></em> del anterior <em>plugin</em>. Tendríamos que cambiar nuestra definición anterior del <em>plugin</em> por la siguiente:</p>
<pre class="brush: xml; title: ; notranslate">
...
&lt;plugin&gt;
  &lt;groupId&gt;org.apache.felix&lt;/groupId&gt;
  &lt;artifactId&gt;maven-bundle-plugin&lt;/artifactId&gt;
  &lt;version&gt;2.0.0&lt;/version&gt;
  &lt;extensions&gt;true&lt;/extensions&gt;
  &lt;configuration&gt;
    &lt;manifestLocation&gt;META-INF&lt;/manifestLocation&gt;
    &lt;instructions&gt;
      &lt;Bundle-SymbolicName&gt;${groupId}.${artifactId}&lt;/Bundle-SymbolicName&gt;
      &lt;Bundle-Version&gt;${pom.version}&lt;/Bundle-Version&gt;
    &lt;/instructions&gt;
  &lt;/configuration&gt;
  &lt;executions&gt;
    &lt;execution&gt;
      &lt;id&gt;genera-manifest&lt;/id&gt;
      &lt;phase&gt;generate-resources&lt;/phase&gt;
      &lt;goals&gt;
        &lt;goal&gt;manifest&lt;/goal&gt;
      &lt;/goals&gt;
    &lt;/execution&gt;
    &lt;execution&gt;
      &lt;id&gt;genera-bundle&lt;/id&gt;
      &lt;phase&gt;package&lt;/phase&gt;
      &lt;goals&gt;
        &lt;goal&gt;bundle&lt;/goal&gt;
      &lt;/goals&gt;
    &lt;/execution&gt;
  &lt;/executions&gt;
&lt;/plugin&gt;
...
</pre>
<p>El plugin anterior tiene dos ejecuciones en dos fases distintas del ciclo de vida de <em>Maven</em>. En la primera fase, <em><strong>generate-resources</strong></em>, se ejecuta el <em>goal</em> <em><strong>manifest</strong></em>. En la segunda fase, <strong><em>package</em></strong>, se ejecuta el <em>goal</em> definido por defecto en el <em>plugin</em>, <em><strong>bundle</strong></em>.</p>
<p>No olvidar añadir el elemento : </p>
<pre class="brush: xml; title: ; notranslate">
&lt;manifestLocation /&gt;
</pre>
</div>
<p>Vuestro fichero <em>MANIFEST.MF</em> debería contener algo parecido a esto:</p>
<pre class="brush: xml; title: ; notranslate">
Manifest-Version: 1.0
Built-By: justo
Created-By: Apache Maven Bundle Plugin
Import-Package: es.jmac.springdm,org.osgi.framework;version=&quot;1.3&quot;
Export-Package: es.jmac.springdm;uses:=&quot;org.osgi.framework&quot;
Bundle-Version: 1.0.0.SNAPSHOT
Bundle-Name: springdm
Bundle-Description: Prueba de concepto de Spring Dynamic Modules sobre
 Equinox
Build-Jdk: 1.5.0_19
Private-Package: es.jmac.springdm,
Bundle-ManifestVersion: 2
Bundle-SymbolicName: es.jmac.springdm
Tool: Bnd-0.0.311
</pre>
<div id="titulo">2. Ejecución del <em>bundle</em> dentro de <em>Equinox</em></div>
<p>Ya hemos creado un <em>bundle OSGi</em>, ahora vamos a ver si lo hemos creado correctamente. Para ello, vamos a lanzar una instancia de <em>Equinox</em> y especificaremos que dentro de esa instancia se cargue nuestro bundle recien creado.</p>
<p>Nos vamos al menún <strong>Run</strong> &#8211;> <strong>Run Configurations&#8230;</strong> de <em>Eclipse</em></p>
<div id="attachment_374" class="wp-caption aligncenter" style="width: 310px"><a href="http://blog.justoaguilar.com/wp-content/uploads/2009/07/run_configurations.jpg"><img src="http://blog.justoaguilar.com/wp-content/uploads/2009/07/run_configurations-300x165.jpg" alt="Run Configurations..." title="Run Configurations..." width="300" height="165" class="size-medium wp-image-374" /></a><p class="wp-caption-text">Run Configurations...</p></div>
<p>Selecciónamos <em><strong>OSGi Framework</strong></em> y pulsando el botón derecho seleccionamos <strong><strong>New&#8230;</strong></strong></p>
<div id="attachment_372" class="wp-caption aligncenter" style="width: 287px"><a href="http://blog.justoaguilar.com/wp-content/uploads/2009/07/new_osgi_framework.jpg"><img src="http://blog.justoaguilar.com/wp-content/uploads/2009/07/new_osgi_framework-277x300.jpg" alt="New OSGi Framework" title="New OSGi Framework" width="277" height="300" class="size-medium wp-image-372" /></a><p class="wp-caption-text">New OSGi Framework</p></div>
<p>En la parte derecha de la ventana aparecerán la lista de bundles que tenemos actualmente, dividos en dos grupos, los <em>bundles</em> creados en nuestro workspace y los bundles instalados en la <em>Target Platform</em>, que suele ser <em>$ECLIPSE_HOME/plugins</em>. </p>
<div id="attachment_378" class="wp-caption aligncenter" style="width: 310px"><a href="http://blog.justoaguilar.com/wp-content/uploads/2009/07/bundles.jpg"><img src="http://blog.justoaguilar.com/wp-content/uploads/2009/07/bundles-300x217.jpg" alt="Lista de Bundles" title="Lista de Bundles" width="300" height="217" class="size-medium wp-image-378" /></a><p class="wp-caption-text">Lista de Bundles</p></div>
<p>Los siguientes pasos serán seleccionar nuestro <em>bundle</em>, deseleccionar todos los bundles de la sección <em>Target Platform</em> y pichar sobre el botón <strong>Add Required Bundles</strong>, es nos marcará un único bundle el <strong>core</strong> de <em>OSGi</em>, <strong>org.eclipse.osgi</strong>. Pinchamos en <strong><em>Apply</em></strong> para salvar los cambios y pichamos en <strong><em>Run</em></strong>.</p>
<p>A continuación se debe abrir la ventana de <em>Console</em> en la que aparecerá el <em>shell</em> de <em>OSGi</em>, algo como :<br />
<code>osgi></code><br />
Para comprobar si nuestro bundle ha sido inicializado correctamente, introducimos el comando <em>ss</em> y pulsamos intro. Debería aparecer lo siguiente, si no, algo hemos mal : </p>
<div id="attachment_375" class="wp-caption aligncenter" style="width: 310px"><a href="http://blog.justoaguilar.com/wp-content/uploads/2009/07/ss_osgi.jpg"><img src="http://blog.justoaguilar.com/wp-content/uploads/2009/07/ss_osgi-300x193.jpg" alt="Short Status (ss)" title="Short Status (ss)" width="300" height="193" class="size-medium wp-image-375" /></a><p class="wp-caption-text">Short Status (ss)</p></div>
<p>Acabamos de ejecutar nuestro <em>bundle</em> dentro de <em>Equinox</em>. Tampoco tiene mucho mérito porque es un <em>bundle</em> vacío, pero bueno, vamos poco a poco. </p>
<div id="titulo">3. Añadimos una clase <em>BundleActivator</em> a nuestro <em>bundle</em></div>
<p>Una clase <em>activator</em> se encarga de inicializar/liberar un <em>bundle</em>. Esta pequeña clase nos avisará que el bundle ha sido inicializado/parado correctamente. </p>
<p>La clase debe implementar la interfaz <em>BundleActivator</em> definida en la <em><a href="http://www.osgi.org/javadoc/r4v41/org/osgi/framework/BundleActivator.html">API</a</em> de <em>OSGi</em>. Para que nuestro código compile correctamente será necesario añadir un par de dependencias a nuestro <em>pom.xml</em>. Basta con añadir el siguiente fragmento de código:</p>
<pre class="brush: xml; title: ; notranslate">
&lt;dependencies&gt;
  ...
  &lt;dependency&gt;
    &lt;groupId&gt;org.osgi&lt;/groupId&gt;
    &lt;artifactId&gt;osgi_R4_core&lt;/artifactId&gt;
    &lt;version&gt;1.0&lt;/version&gt;
    &lt;scope&gt;provided&lt;/scope&gt;
  &lt;/dependency&gt;
  &lt;dependency&gt;
    &lt;groupId&gt;org.osgi&lt;/groupId&gt;
    &lt;artifactId&gt;osgi_R4_compendium&lt;/artifactId&gt;
    &lt;version&gt;1.0&lt;/version&gt;
    &lt;scope&gt;provided&lt;/scope&gt;
  &lt;/dependency&gt;
...
&lt;/dependencies&gt;
</pre>
<p>El código de la clase <em>SpringDMActivator</em> es el siguiente:</p>
<pre class="brush: java; title: ; notranslate">
package es.jmac.springdm;

import org.osgi.framework.BundleActivator;
import org.osgi.framework.BundleContext;

/**
 * @author Justo Aguilar
 * Creation date : 20/07/2009
 *
 * &lt;p&gt;
 */
public class SpringDMActivator implements BundleActivator {

	public void start(BundleContext context) throws Exception {
		System.out.println(&quot;Arrancando bundle &quot; + context.getBundle().getSymbolicName() + &quot;...&quot;);
	}

	public void stop(BundleContext context) throws Exception {
		System.out.println(&quot;Parando bundle &quot; + context.getBundle().getSymbolicName() + &quot;...&quot;);
	}
}
</pre>
<p>Un último paso necesario  es añadir al fichero <em>MANIFEST.MF</em> la siguiente línea:</p>
<pre class="brush: xml; title: ; notranslate">
Bundle-Activator: es.jmac.springdm.SpringDMActivator
</pre>
<p>con la ruta de nuestro clase <em>BundleActivator</em>.</p>
<p>La próxima vez que arranquemos la instancia del <em>OSGi Framework</em> creada anteriormente debería aparecer el mensaje definido por el método <em>start</em> de la clase <em>SpringDMActivator</em>.</p>
<div id="titulo">4. Spring DM entra en juego</div>
<p>En este post se hace un uso trivial e insignificante de <em>Spring DM</em>. En próximos post se mostrará la verdadera potencia de integrar <em>Spring Framework</em> con <em>OSGi</em>. </p>
<p>Nuestro objetivo en este último paso es inicializar un <em>ApplicationContext</em> de <em>Spring</em> al iniciar un bundle en <em>OSGi</em>.</p>
<p>Lo primero que haremos será crear un <em>POJO</em> para referenciarlo posteriormente en el fichero <em>applicationContext.xml</em> de <em>Spring</em>. Yo he creado una clase <em>Entity</em> : </p>
<pre class="brush: java; title: ; notranslate">
package es.jmac.springdm.domain;

import java.io.Serializable;

/**
 * @author Justo Aguilar
 * Creation date : 20/07/2009
 *
 */
public class Entity implements Serializable {

	/** Identificador de Serialización  */
	private static final long serialVersionUID = 147003001012587224L;

	/** Identificador de la entidad */
	private Long id;

	/** Nombre de la entidad */
	private String name;

	/**
	 * Constructor de Entity. Necesita el identificador.
	 * @param id Identificador de la entidad.
	 */
	public Entity(Long id) {
		this.id = id;
		System.out.println(&quot;Creando entidad con id : &quot; + id);
	}

}
</pre>
<p>A continuación hay que crear y definir el fichero <em>applicationContext.xml</em> de <em>Spring</em>. Este fichero se situará dentro de una carpeta <em>spring</em> que a su vez debe estar contenida en la carpeta <em>META-INF</em> del proyecto.</p>
<pre>
Proyecto
|__META-INF
    |__spring
        |__applicationContext.xml
</pre>
<p>Este fichero contendrá la siguiente información :</p>
<pre class="brush: xml; title: ; notranslate">
&lt;?xml version=&quot;1.0&quot; encoding=&quot;UTF-8&quot;?&gt;
&lt;beans xmlns=&quot;http://www.springframework.org/schema/beans&quot;
   xmlns:xsi=&quot;http://www.w3.org/2001/XMLSchema-instance&quot;
   xsi:schemaLocation=&quot;http://www.springframework.org/schema/beans   

http://www.springframework.org/schema/beans/spring-beans.xsd&quot;&gt;

    &lt;bean id=&quot;entity&quot; class=&quot;es.jmac.springdm.domain.Entity&quot;&gt;
      &lt;constructor-arg&gt;&lt;value&gt;1&lt;/value&gt;&lt;/constructor-arg&gt;
    &lt;/bean&gt;
&lt;/beans&gt;
</pre>
<p>Ahora es cuando entra en juego <em>Spring DM</em>. Evidentemente, por si solo, <em>Equinox</em> no tiene ni idea para que sirve la carpeta spring creada bajo <em>META-INF</em> ni los ficheros contiene ésta, por tanto si volvemos a arrancar <em>Equinox</em> veremos que no sucede nada nuevo. Para dar un tratamiento especial a este tipo de <em>bundles</em>, <em>Spring-Powered bundle</em> según la documentación de <em>Spring DM</em>, es necesario que alguién esté pendiente de cuando se inicializa o se para este tipo de <em>bundles</em>. Ese alguien es conocido como <strong>Extender</strong>. <em>Extender</em> no es más que un <em>bundle</em> que está a la escucha del ciclo de vida del restos de <em>bundles</em>, cuando un <em>bundle</em> se inicializa o se para, comprueba si se trata de un <em>Spring-Powered bundle</em> para realizar la tarea de inicialiar/parar el <em>ApplicationContext</em> asociado al éste.</p>
<p>Por tanto será necesario instalar el <em>bundle Extender</em> y todos sus dependencias dentro de la <em>Target Platform</em> y marcarlos para que se ejecuten dentro de <em>Equinox</em>. Bueno, vamos por pasos.</p>
<p><strong>Primero</strong>, nos descargamos todos los <em>bundles</em> que necesitamos. Para ello acudimos al repositorio de <em>springsource</em>, http://www.springsource.com/repository/app/ y nos descargamos el <em>jar</em> <a href="http://www.springsource.com/repository/app/bundle/version/detail?name=org.springframework.osgi.extender&#038;version=1.2.0&#038;searchType=bundlesByName&#038;searchQuery=extender">org.springframework.osgi.extender</a>, en mi caso la versión 1.2.0. En la parte inferior de esa ventana aparece una sección plegada que dice <em>Required Dependencies (9)</em>, la desplegamos y nos bajamos cada uno de los <em>bundles</em> que ahí se indican.</p>
<div id="attachment_377" class="wp-caption aligncenter" style="width: 310px"><a href="http://blog.justoaguilar.com/wp-content/uploads/2009/07/dependencias_extender.jpg"><img src="http://blog.justoaguilar.com/wp-content/uploads/2009/07/dependencias_extender-300x272.jpg" alt="Dependencias Bundle Extender de Spring DM" title="Dependencias Bundle Extender de Spring DM" width="300" height="272" class="size-medium wp-image-377" /></a><p class="wp-caption-text">Dependencias Bundle Extender de Spring DM</p></div>
<p><strong>Segundo</strong>, copiamos los <em>JARs</em> descargados al directorio de nuestra <em>Target Platform</em>, si no sabes a lo que me refiero, no lo habrás cambiado asi que el directorio es  $INSTALACION_ECLIPSE/plugins. Reiniciamos <em>Eclipse</em>.</p>
<p><strong>Tercero</strong>, abrimos la ventana <em>Run Configurations&#8230;</em> y comprobamos que entre los <em>bundles</em> de la sección <em>Target Platform</em> se encuentran los que acabamos de copiar. <strong>Marcamos</strong> el bundle de <strong>Extender</strong> y volvemos a pinchar en <strong>Add Required Bundles</strong>, veremos como se seleccionan el resto de bundles copiados en el paso anterior. <strong>Aplicamos</strong> y <strong>ejecutamos</strong>.</p>
<p>Si no os aparece el mensaje definido en el constructor del <em>POJO</em>, algo habréis hecho mal.  </p>
<p></p>
<div id="referencias">
<b>Enlaces relacionados:</b> </p>
<ul>
<li><a href="http://blog.justoaguilar.com/2009/08/que-es-osgi-y-para-que-sirve/">¿Qué es OSGi &#8230; y para qué sirve?</a></li>
<li><a href="http://www.osgi.org/Main/HomePage">OSGi Foundation</a></li>
<li><a href="http://www.springsource.org/osgi">Spring Dynamic Modules</a></li>
<li><a href="http://www.eclipse.org/equinox/">Equinox</a></li>
<li><a href="http://maven.apache.org">Apache Maven</a></li>
<li><a href="http://www.springsource.org/about">Spring Framework</a></li>
<li><a href="http://www.eclipse.org/">Eclipse</a></li>
<li><a href="http://felix.apache.org/site/apache-felix-maven-bundle-plugin-bnd.html">maven-bundle-plugin</a></li>
<li><a href="http://felix.apache.org/site/index.html">Apache Felix</a></li>
<li><a href="http://www.aqute.biz/Code/Bnd">BND</a></li>
<li><a href="http://wiki.ops4j.org/display/ops4j/Pax">Pax</a></li>
</ul>
</div>

<!--INFOLINKS_OFF-->
<!-- google_ad_section_end -->
]]></content:encoded>
			<wfw:commentRss>http://blog.justoaguilar.com/2009/07/configuracion-osgi-spring-dm-y-maven/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Particularidades de Java2WSDL en Axis</title>
		<link>http://blog.justoaguilar.com/2008/08/particularidades-de-java2wsdl-en-axis/</link>
		<comments>http://blog.justoaguilar.com/2008/08/particularidades-de-java2wsdl-en-axis/#comments</comments>
		<pubDate>Wed, 20 Aug 2008 20:15:27 +0000</pubDate>
		<dc:creator>justo</dc:creator>
				<category><![CDATA[howto]]></category>
		<category><![CDATA[java]]></category>
		<category><![CDATA[axis]]></category>
		<category><![CDATA[axis2]]></category>
		<category><![CDATA[java2wsdl]]></category>
		<category><![CDATA[webservices]]></category>
		<category><![CDATA[wsdl]]></category>
		<category><![CDATA[wsdl2java]]></category>

		<guid isPermaLink="false">http://blog.justoaguilar.com/?p=112</guid>
		<description><![CDATA[Hace un par de semana me estuve peleando con los scripts de Axis que permiten crear documento WSDL a partir de una clase Java, Java2WSDL y viceversa, generar el skeleton o stub Java a partir de un documento WSDL, WSDL2Java. Particularmente la herramienta java2wsdl tiene ciertos detalles que pueden llegar a ser molestos si no [...]]]></description>
			<content:encoded><![CDATA[
<!-- google_ad_section_start -->
<!--INFOLINKS_ON-->
<p><p>Hace un par de semana me estuve peleando con los scripts de Axis que permiten crear documento <a href="http://es.wikipedia.org/wiki/WSDL">WSDL</a> a partir de una clase Java, <em><a href="http://ws.apache.org/axis/java/user-guide.html#Java2WSDLBuildingWSDLFromJava">Java2WSDL</a></em> y viceversa, generar el <em>skeleton</em> o <em>stub</em> Java a partir de un documento WSDL, <em><a href="http://ws.apache.org/axis/java/user-guide.html#WSDL2JavaBuildingStubsSkeletonsAndDataTypesFromWSDL">WSDL2Java</a></em>. Particularmente la herramienta <em>java2wsdl</em> tiene ciertos detalles que pueden llegar a ser molestos si no se sabe como abordarlos. A continuación os cuento los problemas que tuve y las soluciones que adopté.</p>
<p>&nbsp;</p>
<p><strong>1. WSDL es un <a href="http://www.w3.org/TR/wsdl">estándar web</a></strong>.<br />
La afirmación anterior implica que la información que puede contener este documento debe ser independiente de cualquier plataforma, arquitectura, sistema operativo o lenguaje de programación, por tanto, en la sección donde se define los tipos <code><types>...</types></code> <em>no puede haber</em> referencia a ningún tipo específico Java, tal como <em>List, ArrayList, Map, &#8230;</em>.</p>
<p><strong>Solución:</strong><br />
Cualquier método que quiera ser publicado a través de un servicio web debe tener como parámetros de entrada y/o salida, tipos estándares, esto es en Java, tipos primitivos, arrays u objetos propios serializables que contengan tipos estándares.</p>
<p><strong>Ejemplo</strong><br />
Supongamos que tenemos una clase <code>Foo</code> que queremos publicar como servicio web y que contiene un tipo no estándar, por ejemplo una Lista.</p>
<pre class="brush: java; title: ; notranslate">
public class Foo {
  private List&lt;String&gt; lstStrings;

  public List&lt;String&gt; getLstStrings() {
    return this.lstStrings;
  }

  public void setLstStrings(List&lt;String&gt; lstStrings){
    this.lstStrings = lstStrings;
  }
  ...
}
</pre>
<p>Bien, pues si queremos que la herramienta <code>java2wsdl</code> genere un documento <code>WSDL</code> correcto debemos transformar el código anterior en el siguiente:</p>
<pre class="brush: java; title: ; notranslate">
public class Foo {
  private List&lt;String&gt; lstStrings;
  private String[] arrayStrings;     // Según la versión de axis, es necesario declararlo o no

  public List&lt;String&gt; getLstStrings() {
    return this.lstStrings;
  }

  public void setLstStrings(List&lt;String&gt; lstStrings) {
    this.lstStrings = lstStrings;
  } 

  public String[] getArrayStrings() {
    if ( getLstStrings() != null ) {
      return getLstStrings().toArray(new String[getLstStrings().size()]);
    }
    return new String[0];
  }

  public void setArrayStrings(String[] arrayStrings) {
    this.setArrayStrings(Arrays.asList(arrayStrings));
  }
  ...
}
</pre>
<p>&nbsp;</p>
<p><strong>2. No están soportados los modelos de datos con clases internas</strong>.<br />
Actualmente el script no genera correctamente el conjunto de tipos del documento <code>WSDL</code> si el método de la clase a publicar contiene parametros cuyos tipos sean clases internas.</p>
<p>Este error es debido a un fallo en el jar <em>annogen-0.1.0</em>; existe un <a href="http://blogs.deepal.org/2008/02/axis2-pojo-with-inner-classes-are.html">parche oficioso</a> que genera el <code>WSDL</code> correctamente pero si usamos <code>wsdl2java</code> para generar el cliente, la invocación al servicio dará un fallo.</p>
<p><strong>Solución:</strong</p>
<ol>
<li>Sacar las clases internas de la clase del servicio.</li>
<li>Utilizar el <a href="http://people.apache.org/%7Edeepal/m2repo/">jar</a> <em>oficioso</em> de Deepal Jayasinghe y crear el stub sin utilizar WSDL2Java.</li>
</ol>
<p>&nbsp;</p>
<p><strong>3. Nuestro servicio lanza una excepción.</strong><br />
Siempre que nuestro servicio lanza una <code>Exception</code> la herramienta <code>java2wsdl</code> intenta mapear el tipo de la excepción a un tipo <code>WSDL</code>. Esto siempre produce el código :</p>
<pre class="brush: xml; title: ; notranslate">
...
&lt;xs:complexType name=&quot;MyException&quot;&gt;
  &lt;xs:complexContent&gt;
    &lt;xs:extension base=&quot;xs:Exception&quot; /&gt;
    &lt;!-- o &lt;xs:extension base=&quot;xs:RuntimeException&quot; /&gt; --&gt;
    &lt;xs:sequence /&gt;
  &lt;/xs:complexContent&gt;
&lt;/xs:complexType&gt;
...
</pre>
<p>Al final <code>java2wsdl</code> busca la clase padre de la que extiende nuestra <code>exception</code> para mapearla según la semántica que permite <code>WSDL</code>, sin embargo, ni <code>xs:Exception</code> ni <code>xs:RuntimeException</code> existen como tipo estándares de <code>XML-Schema</code> lo que provoca la existerncia de warnings/errores de los editores XML que validan los documentos. </p>
<p><strong>Solución:</strong<br />
La única solución que he encontrado es hacer el siguiente cambio :</p>
<pre class="brush: xml; title: ; notranslate">
&#8230;
&lt;xs:complexType name=&quot;MyException&quot;&gt;
  &lt;xs:complexContent&gt;
    &lt;xs:extension base=&quot;xs:anyType&quot; /&gt;
    &lt;!&#8211; o &lt;xs:extension base=&quot;xs:anyType&quot; /&gt; &#8211;&gt;
    &lt;xs:sequence /&gt;
  &lt;/xs:complexContent&gt;
&lt;/xs:complexType&gt;
&#8230;
</pre>
<p>Eso si, si se genera un cliente a partir de este documento <code>WSDL</code> se deberán modificar las clases asociadas a las excepciones para que extiendan de <code>RuntimeException</code> o <code>Exception</code> o cualquier otra.</p>
<p>&nbsp;</p>
<p><strong>4. &lt;soap12:fault /&gt;</strong>.<br />
Cuando se crea un documento <code>WSDL</code> para que pueda ser utilizado por las dos versiones de <code><a href="http://www.w3.org/TR/soap/">SOAP</a></code> existentes, 1.1 y 1.2, la parte del binding de <code>SOAP 1.2</code> es incorrecta.</p>
<p>La herramienta <code>java2wsdl</code> genera un código tal como este : </p>
<pre class="brush: xml; title: ; notranslate">
...
&lt;wsdl:binding...&gt;
  &lt;soap12:opertation ... /&gt;
  &lt;wsdl:operation ...&gt;
    ...
    &lt;wsdl:fault ...&gt;
      &lt;soap12:fault use=&quot;literal&quot; name=&quot;MyException&quot; /&gt;
    &lt;/wsdl:fault&gt;
  &lt;/wsdl:operation&gt;
&lt;/wsdl:binding&gt;
...
</pre>
<p>pues el código anterior falla, el elemento <code>&lt;soap12:fault use="literal" name="MyException" /&gt;</code> no es correcto y, por ejemplo, la herramient <code>wsdl2java</code> no puede crear un cliente correcto a partir del <code>WSDL</code> </p>
<p><strong>Solución:</strong<br />
Cambiar <code>&lt;soap12:fault use="literal" name="MyException" /&gt;</code> por <code>&lt;soap:fault use="literal" name="MyException" /&gt;</code></p>
<p>&nbsp;</p>
<p>Si habéis tenido problemas parecidos y los habéis resuelto de una forma distinta a la mía, por favor, no dudéís es compartirlo para que todos salgamos ganando.</p>
<p>&nbsp;</p>

<!--INFOLINKS_OFF-->
<!-- google_ad_section_end -->
]]></content:encoded>
			<wfw:commentRss>http://blog.justoaguilar.com/2008/08/particularidades-de-java2wsdl-en-axis/feed/</wfw:commentRss>
		<slash:comments>3</slash:comments>
		</item>
		<item>
		<title>Javadoc con Diagramas de Clases usando Graphviz + UMLGraph + Maven</title>
		<link>http://blog.justoaguilar.com/2008/07/javadoc-con-diagramas-de-clases-usando-graphviz-umlgraph-maven/</link>
		<comments>http://blog.justoaguilar.com/2008/07/javadoc-con-diagramas-de-clases-usando-graphviz-umlgraph-maven/#comments</comments>
		<pubDate>Tue, 15 Jul 2008 19:24:23 +0000</pubDate>
		<dc:creator>justo</dc:creator>
				<category><![CDATA[howto]]></category>
		<category><![CDATA[graphviz]]></category>
		<category><![CDATA[java]]></category>
		<category><![CDATA[javadoc]]></category>
		<category><![CDATA[maven]]></category>
		<category><![CDATA[umlgraph]]></category>

		<guid isPermaLink="false">http://blog.justoaguilar.com/?p=59</guid>
		<description><![CDATA[Una de las herramientas más importantes con las que cuenta la plataforma Java es, sin duda, el javadoc. A través de ella es posible crear una referencia completa y homogénea de todo el conjunto de clases definidas es nuestro proyecto. Toda página de una clase del javadoc está dividida en distintas zonas, la primera de [...]]]></description>
			<content:encoded><![CDATA[
<!-- google_ad_section_start -->
<!--INFOLINKS_ON-->
<p><p>Una de las herramientas más importantes con las que cuenta la plataforma <i>Java</i> es, sin duda, el <a href="http://java.sun.com/j2se/javadoc/">javadoc</a>. A través de ella es posible crear una referencia completa y homogénea de todo el conjunto de clases definidas es nuestro proyecto. </p>
<p>Toda página de una clase del <i>javadoc</i> está dividida en distintas zonas, la primera de ellas muestra la situación jerárquica de la clase respecto a la clase padre <a href="http://java.sun.com/j2se/1.5.0/docs/api/java/lang/Object.html">Object</a>. Bajo la situación jerárquica aparece la lista de interfaces que implementa si existe alguna y, por último, la lista de clases que la extienden. Para las interfaces aparece la lista de superinterfaz en primer lugar y la lista de subinterfaces en segundo.</p>
<p>Aunque esta información pueda ser suficiente en muchos casos, no estaría de más completarla con diagramas de clases que permitan una rápida visualización de la jerarquía de clases de nuestro proyecto. Ese es el objetivo de este post, añadir <em>diagramas de clases UML</em> a nuestro <em>javadoc</em>.</p>
<p><a class="thickbox" rel="" href='http://blog.justoaguilar.com/wp-content/gallery/umlgraph/nivel01.png' title=''><img src='http://blog.justoaguilar.com/wp-content/gallery/umlgraph/thumbs/thumbs_nivel01.png' alt='Diagrama UML generado con UMLGraph' class='ngg-singlepic ngg-center' /></a></p>
<p>Para ello necesitaremos dos cosas básicamente :</p>
<ul>
<li><a href="http://www.graphviz.org">Graphviz</a>, librería que permite la generación de gráficos.</li>
<li><a href="http://wiki.wsmoak.net/cgi-bin/wiki.pl?UMLGraph">UMLGraph</a> plugin para maven que sobreescribe el <a href="http://java.sun.com/j2se/1.4.2/docs/tooldocs/javadoc/overview.html">doclet</a> que genera el javadoc para insertar los gráficos generados por Graphviz.</li>
<li><a href="http://maven.apache.org">Maven</a></li>
</ul>
<p>A continuación vamos a ver como instalar <em>Graphviz</em> y como configurar el <a href="http://maven.apache.org/pom.html">pom.xml</a> del proyecto maven para que use el plugin <em>UMLGraph</em>.</p>
<h3>Instalación de Graphviz</h3>
<p>Hay varias formas de hacerlo, bajando el fuente y compilando, haciendo un <em>checkout</em> del repositorio <a href="http://www.nongnu.org/cvs/">CVS</a> o dejando que el sistema de paquetes de tu sistema operativo lo haga todo por tí <img src='http://blog.justoaguilar.com/wp-includes/images/smilies/icon_wink.gif' alt=';-)' class='wp-smiley' /> . Esto último es lo que haremos pues la finalidad de éste post es otra.</p>
<pre class="brush: bash; title: ; notranslate">
sudo apt-get update
sudo apt-get install graphviz
man dot
</pre>
<p></p>
<p>La última entrada es sólo para comprobar que el paquete ha sido instalado correctamente. </p>
<h3>Modificación del fichero <em>pom.xml</em> del proyecto</h3>
<p>Simplemente se debe añadir al fichero las siguientes líneas de código</p>
<pre class="brush: xml; title: ; notranslate">
...
&lt;reporting&gt;
  &lt;plugins&gt;
    &lt;plugin&gt;
      &lt;groupId&gt;org.apache.maven.plugins&lt;/groupId&gt;
      &lt;artifactId&gt;maven-javadoc-plugin&lt;/artifactId&gt;
      &lt;configuration&gt;
        &lt;doclet&gt;gr.spinellis.umlgraph.doclet.UmlGraphDoc&lt;/doclet&gt;
        &lt;docletArtifact&gt;
          &lt;groupId&gt;gr.spinellis&lt;/groupId&gt;
          &lt;artifactId&gt;UmlGraph&lt;/artifactId&gt;
          &lt;version&gt;4.6&lt;/version&gt;
        &lt;/docletArtifact&gt;
        &lt;additionalparam&gt;
          -inferrel -inferdep -quiet -hide java.* -collpackages java.util.* -qualify
          -postfixpackage -nodefontsize 9
          -nodefontpackagesize 7
        &lt;/additionalparam&gt;
      &lt;/configuration&gt;
    &lt;/plugin&gt;
  &lt;/plugins&gt;
&lt;/reporting&gt;
...
</pre>
<p>Para generar el <em>javadoc</em> se ejecutará el comando <code>mvn javadoc:javadoc</code> dentro del directorio<br />
donde se encuentra el fichero <code>pom.xml</code>.</p>
<p><a class="thickbox" rel="" href='http://blog.justoaguilar.com/wp-content/gallery/umlgraph/umlgraph_javadoc.png' title=''><img src='http://blog.justoaguilar.com/wp-content/gallery/umlgraph/thumbs/thumbs_umlgraph_javadoc.png' alt='Javadoc de la clase Nivel 1' class='ngg-singlepic ngg-center' /></a></p>

<!--INFOLINKS_OFF-->
<!-- google_ad_section_end -->
]]></content:encoded>
			<wfw:commentRss>http://blog.justoaguilar.com/2008/07/javadoc-con-diagramas-de-clases-usando-graphviz-umlgraph-maven/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>

