<?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>Aleph &#187; https</title>
	<atom:link href="http://aleph.llull.net/tag/https/feed/" rel="self" type="application/rss+xml" />
	<link>http://aleph.llull.net</link>
	<description></description>
	<lastBuildDate>Mon, 01 Aug 2011 08:48:44 +0000</lastBuildDate>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>http://wordpress.org/?v=3.1.2</generator>
		<item>
		<title>SSL en subdominios con una única dirección IP</title>
		<link>http://aleph.llull.net/2008/10/27/ssl-en-subdominios-con-una-unica-direccion-ip/</link>
		<comments>http://aleph.llull.net/2008/10/27/ssl-en-subdominios-con-una-unica-direccion-ip/#comments</comments>
		<pubDate>Mon, 27 Oct 2008 08:03:05 +0000</pubDate>
		<dc:creator>Eduard</dc:creator>
				<category><![CDATA[SysAdmin]]></category>
		<category><![CDATA[apache]]></category>
		<category><![CDATA[http]]></category>
		<category><![CDATA[https]]></category>
		<category><![CDATA[mod_rewrite]]></category>
		<category><![CDATA[ssl]]></category>
		<category><![CDATA[web]]></category>

		<guid isPermaLink="false">http://aleph.llull.net/?p=309</guid>
		<description><![CDATA[La aparición a mediados de 1998 de los VirtualHosts basados en nombre en la versión 1.3 del servidor web de Apache significó un cambio importante en el mundo de los servidores web ya que permitía configurar webs en diferentes dominios, todos ellos compartiendo la misma dirección IP. Tampoco hay que olvidar que esa mejora fue [...]]]></description>
			<content:encoded><![CDATA[<p>La aparición a mediados de 1998 de los VirtualHosts basados en nombre en la <a href="http://httpd.apache.org/docs/1.3/">versión 1.3 del servidor web de Apache</a> significó un cambio importante en el mundo de los servidores web ya que permitía configurar webs en diferentes dominios, todos ellos compartiendo la misma dirección <abbr title="Internet Protocol">IP</abbr>. Tampoco hay que olvidar que esa mejora fue posible gracias a que los navegadores (por aquel entonces disponíamos de <a href="http://en.wikipedia.org/wiki/Netscape_Communicator">Netscape Communicator 4.5</a> e <a href="http://en.wikipedia.org/wiki/Internet_Explorer_4">Internet Explorer 4.0</a> [<a href="#nota-1">1</a>]) implementaron la versión 1.1 del protocolo <abbr title="HyperText Transfer Protocol">HTTP</abbr>, según la cual el navegador debe enviar el nombre del dominio al que se están conectando en la cabecera <em>Host</em>, usada por los servidores web para discriminar que a VirtualHost deben enviar la petición.</p>
<p>Desgraciadamente, las webs seguras nunca han podido beneficiarse de esta mejora por la forma en que funciona el protocolo HTTPS (HTTP sobre <abbr title="Secure Sockets Layer">SSL</abbr>). La cabecera Host que permite al servidor discernir el host virtual al que va dirigida la petición del usuario está a nivel HTTP, debajo del cual tenemos SSL y su negociación. En esa negociación es cuando el servidor envía al cliente su certificado X.509 y el navegador comprueba la validez del certificado comparando el campo <abbr title="Common Name">CN</abbr> del certificado con el nombre del dominio al que se está conectando. <strong>Por tanto, el servidor debe enviar el certificado correcto, y por tanto del host virtual correcto, al navegador antes que este le pueda indicar mediante la cabecera Host el dominio al que se quiere conectar</strong>. La única solución es usar VirtualHosts basados en dirección IP.</p>
<p>Sin embargo, si los diferentes dominios que tenemos en el servidor son subdominios que comparten el <a href="http://en.wikipedia.org/wiki/Second-level_domain">second-level domain</a>, o SLD, podemos conseguir que todos ellos dispongan de web segura usando una única dirección IP. Por ejemplo www.example.com y mail.example.com tienen el SLD example.com en común.</p>
<p>Para conseguirlo, también es imprescindible el uso de un <em>Wildcard SSL Certificate</em> (he decidido no traducirlo porque ninguna de las tradicciones que se me han ocurrido me acababa de gustar), que es un certificado SSL en el que el nombre del dominio contiene un comodín. Por ejemplo: *.example.com. De esta manera, indicamos al navegador que este certificado es válido para cualquier subdominio de example.com. No voy a explicar como se generan los certificados SSL ya que <a title="Busqueda 'SSL certificates Howto' en Google" href="http://www.google.com/search?q=ssl+certificates+howto">buscando un poco por google</a> se encuentran multitud de páginas al respecto. Si le vamos a dar un uso personal, podéis crear un <em>self signed certificate</em>. La única diferencia con los tutoriales que podáis encontrar es que como dominio introduciremos &#8220;*.&lt;dominio&gt;&#8221; (Por ejemplo: *.example.com).</p>
<h3>Configuración del servidor web de Apache</h3>
<p>Voy a suponer que en nuestro servidor tenemos configurado un servidor web Apache2 con varios host virtuales (www.example.com, mail.example.com y un servidor WebDAV para el Subversion en svn.example.com), y que el servidor sólo dispone de una dirección IP.</p>
<p>Lo primero que hay que hacer es poner el servidor web a escuchar en el puerto 443 (puerto estándar de HTTPS) y configurar un host virtual sobre ese puerto:</p>

<div class="wp_syntax"><div class="code"><pre class="apache" style="font-family:monospace;"><span style="color: #00007f;">Listen</span> <span style="color: #ff0000;">443</span>
&lt;<span style="color: #000000; font-weight:bold;">VirtualHost</span> *:<span style="color: #ff0000;">443</span>&gt;
  <span style="color: #00007f;">ServerName</span> *.example.com
  <span style="color: #00007f;">ErrorLog</span> /var/log/apache2/https-error.log
&nbsp;
  <span style="color: #adadad; font-style: italic;"># Possible values include: debug, info, notice, warn, error, crit,</span>
  <span style="color: #adadad; font-style: italic;"># alert, emerg.</span>
  <span style="color: #00007f;">LogLevel</span> warn
&nbsp;
  <span style="color: #00007f;">CustomLog</span> /var/log/apache2/https-access.log combined
  <span style="color: #00007f;">ServerSignature</span> <span style="color: #0000ff;">Off</span>
&nbsp;
  <span style="color: #00007f;">SSLEngine</span> <span style="color: #0000ff;">On</span>
  <span style="color: #00007f;">SSLCertificateFile</span> /path/to/certs/example.com.pem
  <span style="color: #00007f;">SSLCertificateKeyFile</span> /path/to/private-keys/example.com.key
&lt;/<span style="color: #000000; font-weight:bold;">VirtualHost</span>&gt;</pre></div></div>

<p>Con esto ya tenemos la capa SSL configurada. El certificado indicado en el parámetro <code>SSLCertificateFile</code> es el <em>Widlcard SSL Certificate</em> que hemos creado anteriormente. A partir de aquí, <strong>el &#8220;truco&#8221; está en el uso creativo del mod_proxy</strong>, que conseguiremos añadiendo la siguiente configuración dentro del hosts virtual que acabamos de crear:</p>

<div class="wp_syntax"><div class="code"><pre class="apache" style="font-family:monospace;">  &lt;<span style="color: #000000; font-weight:bold;">Proxy</span> *&gt;
    <span style="color: #00007f;">Order</span> <span style="color: #00007f;">deny</span>,<span style="color: #00007f;">allow</span>
    <span style="color: #00007f;">Allow</span> from <span style="color: #0000ff;">all</span>
  &lt;/<span style="color: #000000; font-weight:bold;">Proxy</span>&gt;
&nbsp;
  <span style="color: #adadad; font-style: italic;"># Proxy requests to ourselves preserving the Host header</span>
  <span style="color: #00007f;">ProxyPass</span> / http://localhost/
  <span style="color: #00007f;">ProxyPassReverse</span> / http://localhost/
  <span style="color: #00007f;">ProxyPreserveHost</span> <span style="color: #0000ff;">on</span></pre></div></div>

<p>Con esta configuración estamos reenviando todas las peticiones que nos lleguen a este host virtual por HTTPS hacia el propio servidor, pero cambiando el protocolo de HTTPS a HTTP. <strong>Es importante destacar la última línea, que le indica al servidor web que debe mantener la cabecera Host al hacerse la petición a sí mismo</strong>, gracias a la cual no tendrá ningún problema a la hora de determinar que página quiere ver el usuario.</p>
<p>Para que las aplicaciones web puedan distinguir si la petición que les llega a entrado por HTTP o HTTPS, configuraremos el host virtual para que añada la cabecera <code>X_FORWARDED_PROTO</code> a las peticiones que se hace a sí mismo:</p>

<div class="wp_syntax"><div class="code"><pre class="apache" style="font-family:monospace;">  <span style="color: #adadad; font-style: italic;"># Add the X_FORWARDED_PROTO=https to allow applications to identify</span>
  <span style="color: #adadad; font-style: italic;"># proxyed https connections</span>
  <span style="color: #00007f;">RequestHeader</span> set X_FORWARDED_PROTO https</pre></div></div>

<p>Ya sólo queda un detalle, necesario únicamente si vamos a usar esta configuración para proteger un WebDAV. El problema de esta configuración con WebDAV viene a la hora de intentar hacer un &#8216;<code>svn mv</code>&#8216;, por ejemplo. Este comando indica el destino en la cabecera HTTP <code>Destination</code>. Como el cliente se esta conectando a https://svn.example.com/repo/etc, la cabecera tendrá ese valor. Pero el host virtual que tiene la configuración de WebDAV sólo está configurado para HTTP por lo que espera que la cabecera <code>Destination</code> empiece por http://svn&#8230; <strong>Para solucionar la inconsistencia entre la cabecera esperada y la recibida echaremos mano del mod_rewrite</strong>:</p>

<div class="wp_syntax"><div class="code"><pre class="apache" style="font-family:monospace;">  <span style="color: #adadad; font-style: italic;"># Workarrounf for WebDAV Destination header</span>
  <span style="color: #00007f;">RewriteEngine</span> <span style="color: #0000ff;">on</span>
  <span style="color: #00007f;">RewriteCond</span> %{HTTP:Destination} ^https://(.*)$
  <span style="color: #00007f;">RewriteRule</span> . - [env=DESTINATION:http://%1,PT]
  <span style="color: #00007f;">RequestHeader</span> set Destination %{DESTINATION}e env=DESTINATION</pre></div></div>

<h3>Resumen de la configuración</h3>
<p>Una vez explicadas las diferentes partes de la configuración, os pongo toda la configuración tal cual se debe escribir en los ficheros de configuración del servidor web para que no haya problemas de &#8220;¿y esto donde va?, ¿dentro o fuera del virtualhost?&#8221; <img src='http://aleph.llull.net/wp-includes/images/smilies/icon_wink.gif' alt=';-)' class='wp-smiley' />  :</p>

<div class="wp_syntax"><div class="code"><pre class="apache" style="font-family:monospace;"><span style="color: #00007f;">Listen</span> <span style="color: #ff0000;">443</span>
&lt;<span style="color: #000000; font-weight:bold;">VirtualHost</span> *:<span style="color: #ff0000;">443</span>&gt;
  <span style="color: #00007f;">ServerName</span> *.example.com
  <span style="color: #00007f;">ErrorLog</span> /var/log/apache2/https-error.log
&nbsp;
  <span style="color: #adadad; font-style: italic;"># Possible values include: debug, info, notice, warn, error, crit,</span>
  <span style="color: #adadad; font-style: italic;"># alert, emerg.</span>
  <span style="color: #00007f;">LogLevel</span> warn
&nbsp;
  <span style="color: #00007f;">CustomLog</span> /var/log/apache2/https-access.log combined
  <span style="color: #00007f;">ServerSignature</span> <span style="color: #0000ff;">Off</span>
&nbsp;
  <span style="color: #00007f;">SSLEngine</span> <span style="color: #0000ff;">On</span>
  <span style="color: #00007f;">SSLCertificateFile</span> /path/to/certs/example.com.pem
  <span style="color: #00007f;">SSLCertificateKeyFile</span> /path/to/private-keys/example.com.key
&nbsp;
  &lt;<span style="color: #000000; font-weight:bold;">Proxy</span> *&gt;
    <span style="color: #00007f;">Order</span> <span style="color: #00007f;">deny</span>,<span style="color: #00007f;">allow</span>
    <span style="color: #00007f;">Allow</span> from <span style="color: #0000ff;">all</span>
  &lt;/<span style="color: #000000; font-weight:bold;">Proxy</span>&gt;
&nbsp;
  <span style="color: #adadad; font-style: italic;"># Workarrounf for WebDAV Destination header</span>
  <span style="color: #00007f;">RewriteEngine</span> <span style="color: #0000ff;">on</span>
  <span style="color: #00007f;">RewriteCond</span> %{HTTP:Destination} ^https://(.*)$
  <span style="color: #00007f;">RewriteRule</span> . - [env=DESTINATION:http://%1,PT]
  <span style="color: #00007f;">RequestHeader</span> set Destination %{DESTINATION}e env=DESTINATION
&nbsp;
  <span style="color: #adadad; font-style: italic;"># Add the X_FORWARDED_PROTO=https to allow applications to identify</span>
  <span style="color: #adadad; font-style: italic;"># proxyed https connections</span>
  <span style="color: #00007f;">RequestHeader</span> set X_FORWARDED_PROTO https
&nbsp;
  <span style="color: #adadad; font-style: italic;"># Proxy requests to ourselves preserving the Host header</span>
  <span style="color: #00007f;">ProxyPass</span> / http://localhost/
  <span style="color: #00007f;">ProxyPassReverse</span> / http://localhost/
  <span style="color: #00007f;">ProxyPreserveHost</span> <span style="color: #0000ff;">on</span>
&lt;/<span style="color: #000000; font-weight:bold;">VirtualHost</span>&gt;</pre></div></div>

<h3>Conclusiones</h3>
<p>Personalmente veo está configuración como una solución válida para servidores personales en los que se suelen usar <em>self signed certificates</em> o de <a href="http://www.cacert.org/">CAcert</a>. Para pequeñas empresas (con unos pocos dominios) puede ser una solución, pero los <em>Widlcard SSL Certificates</em> de <a title="Thawte Wildcard SSL Certificates" href="https://www.thawte.com/ssl-digital-certificates/wildcardssl/index.html">autoridades</a> <a title="Verisign Wildcard SSL Certificates" href="http://www.verisign.com/ssl-certificates/wildcard-ssl-certificates/">certificadoras</a> <a title="Go Daddy SSL Certificates" href="http://www.godaddy.com/gdshop/ssl/ssl.asp?ci=9039">comerciales</a> (cuyos certificados raíz vienen en los navegadores) no son baratos. Dependiendo del <em>hosting</em> creo que saldría más barato contratar IPs adicionales para el servidor y certificados individuales que no usar el <em>Widlcard SSL Certificate</em> en un servidor con una sóla IP.</p>
<p>En empresas medianas o grandes, puede justificarse el uso de <em>Widlcard SSL Certificates</em> si tienen un gran número de subdominios que quieren asegurar ya que puede suponer un ahorro, pero no veo el motivo de tenerlo todo sobre una sóla dirección IP. Por tanto, no veo que la configuración aquí explicada sea aplicable en este tipo de empresas.</p>
<p>A mi, esta solución me está funcionando muy bien y de momento no he encontrado ningún problema.</p>
<h3>Notas adicionales</h3>
<p>[<span id="nota-1">1</span>] Para los nostálgicos, ¿os acordáis de la <a href="http://es.wikipedia.org/wiki/Guerra_de_navegadores">guerra de navegadores</a>?</p>
]]></content:encoded>
			<wfw:commentRss>http://aleph.llull.net/2008/10/27/ssl-en-subdominios-con-una-unica-direccion-ip/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>

