27/Nov/2013
|
Control de versiones utilizando Subversion |
Control de versiones utilizando Subversion
Subversion es un sistema centralizado de control de versiones. En otras palabras, es un sistema que permite llevar el control de los cambios realizados por una o mas personas a una serie de documentos a traves del tiempo.
En un proyecto de Plone podemos tener docenas o centenares de documentos de diversos tipos, como HTML o codigo de Python. Dependiendo del tamano de nuestro equipo de trabajo, los documentos pueden estar cambiando constantemente y ser modificados por diversas personas. En el ciclo de trabajo diario, esta situacion puede ocasionar cambios perdidos y severos problemas de coordinacion, lo que degenera en tiempo perdido y baja de productividad para todos los miembros del equipo de desarrollo.
Un sistema de control de versiones se encarga de llevar el control de todas estas modificaciones a los documentos del proyecto. Un desarrollador solamente tiene que encargarse de guardar sus cambios al proyecto y el sistema se ocupa de integrarlos con los cambios de los otros miembros del equipo y de asegurarse de que no existan conflictos.
Adicionalmente, utilizar un sistema de control de versiones nos permite cosas como comparar la mas reciente version de un documento con versiones anteriores, identificar quien ha realizado cuales cambios en el codigo y designar un conjunto de documentos especificos como una ‘entrega’ del proyecto que queda identificada con las versiones de cada uno de los documentos en el instante de la definicion.
Uno de los sistemas de control de versiones mas utilizados en la actualidad es Subversion. Es un sistema facil de usar y esta disponible para todas las plataformas populares, como Linux, Mac OSX y Windows.
La instalacion depende de la plataforma, pero generalmente hay paquetes disponibles para instalar con el administrador de paquetes propio de cada sistema.
Por ejemplo, en el caso de Ubuntu o Debian Linux, basta ejecutar el siguiente comando:
$ apt-get install subversion
En el caso de Red Hat o CentOS, el comando es muy similar:
Otras distribuciones de Linux y otros sistemas operativos tienen sus propios mecanismos de instalacion, pero no difieren mucho de lo anterior.
Una vez instalado Subversion, puede iniciarse el servicio utilizando el comando svnserve, de la siguiente manera:
El servicio espera peticiones en el puerto 3690, por lo que si se tiene un firewall debe abrirse dicho puerto para entrada y salida.
Por supuesto, el servicio recien iniciado no tiene ninguna utilidad hasta que es creado un repositorio para servir. Esto se hace con el comando svnadmin, el cual recibe como parametro el path dentro del sistema de archivos donde se desea crear el repositorio:
$ svnadmin create /path/del/repositorio
El path puede ser cualquiera en el sistema, pero hay que asegurarse que los usuarios tengan acceso de lectura por lo menos en todo el path elegido. Al crear el repositorio, Subversion genera una serie de directorios dentro del path elegido. Por ejemplo, si creamos el repositorio en /opt/svn:
$ svnadmin create /opt/svn
$ cd /opt/svn
$ ls
conf db format hooks locks README.txt
El contenido del repositorio se guarda dentro del directorio db, pero por supuesto nunca hay que modificar nada dentro del mismo.
El repositorio que hemos creado tiene un URL que se forma utlizando el protocolo svn con el host donde se encuentra el servicio de svnserve y el path completo al repositorio. En este caso: svn://localhost/opt/svn.
Por el momento, el unico directorio que debe interesarnos dentro del repositorio es conf, pues ahi se guardan los archivos de configuracion. El archivo pricipal de configuracion de Subversion es svnserve.conf y contiene las siguientes declaraciones, omitiendo los comentarios:
[general]
anon-access = read
auth-access = write
password-db = passwd
authz-db = authz
realm = My First Repository
[sasl]
use-sasl = false
min-encryption = 0
max-encryption = 256
La seccion general define primero el tipo de acceso permitido al repositorio. Por default, los usuarios anonimos tiene permiso para leer y solo los autenticados pueden escribir. En la mayoria de los ambientes corporativos, es deseable que los usuarios anonimos no tengan ningun acceso, por lo que la declaracion correspondiente debe cambiarse por la siguiente:
Generalmente este es el unico cambio que tiene sentido en los valores de acceso.
Las siguientes dos opciones, password-db y authz-db se refieren a nombres de archivos en el mismo directorio que contienen la base de datos de passwords y las definiciones de grupos y permisos por path, respectivamente.
La opcion realm se utiliza para nombrar al respositorio y puede tener cualquier valor deseado.
Finalmente, la seccion sasl es para definir si se desea utilizar encripcion para los passwords. En caso de que el valor de use-sasl sea true, el archivo definido arriba en password-db no se utiliza. El uso de SASL requiere tener instalado el soporte para SASL en el sistema.
El archivo passwd contiene la definicion de usuarios y passwords del repositorio. Es simplemente un archivo de texto donde se define un usuario con su password en cada linea:
En el ejemplo, se define un usuario llamado juan, con el password secreto. Para agregar usuarios simplemente hay que poner una nueva linea con el nombre del usuario y el password, separados por el signo =.
El archivo authz contiene las definiciones de grupos y permisos por path. Por default contiene solamente ejemplos:
[aliases]
# joe = /C=XZ/ST=Dessert/L=Snake City/O=Snake, Ltd./OU=Institute/CN=Joe Average
[groups]
# harry_and_sally = harry,sally
# harry_sally_and_joe = harry,sally,&joe
# [/foo/bar]
# harry = rw
# &joe = r
# * =
# [repository:/baz/fuz]
# @harry_and_sally = rw
# * = r
Dado que el repositorio que recien hemos creado no tiene contenido aun, no podemos configurar otro path mas que la raiz. Para permitir al usuario juan que definimos arriba acceso de lectura y escritura al repositorio, basta agregar las siguientes lineas:
Tambien es posible agregar grupos de usuarios en la seccion groups. Basta poner el nombre de grupo y a continuacion una lista de nombres de los que fueron definidos en el archivo passwd, separados por comas.
Una vez creado y configurado el repositorio es importante definir su estructura antes de comenzar a importar contenido. Independientemente de la organizacion de carpetas que se decida adoptar, en un repositorio de Subversion se recomienda tener una carpeta principal por cada proyecto. A su vez, dentro de cada proyecto es usual utilizar una carpeta con la version oficial del codigo, llamada trunk, asi como carpetas para etiquetar versiones y para realizar pruebas, llamadas tags y branches respectivamente.
La estructura inicial de directorios puede crearse paso a paso, pero en muchas ocasiones es mejor importarla en un solo paso utilizando ya sea un proyecto ya existente o simplemente una estructura de carpetas. El comando de Subversion para hacer esto se llama import.
Para importar una estructura de directorios al repositorio que creamos con anterioridad, podemos usar una secuencia de comandos como la que sigue:
$ mkdir proyecto_ejemplo
$ cd proyecto_ejemplo
$ mdkir trunk tags branches
$ cd ..
$ svn import proyecto_ejemplo svn://localhost/opt/svn/proyecto_ejemplo
Adding proyecto_ejemplo/trunk
Adding proyecto_ejemplo/branches
Adding proyecto_ejemplo/tags
Committed revision 1
La primera vez que nos conectamos al repositorio, Subversion nos pedira el password para entrar, asumiendo que nuestro nombre de usuario es el mismo con el que estamos conectados en nuestro sistema. Si esto no es asi, basta presionar la tecla enter sin escribir nada y Subersion nos preguntara el nombre de usuario primero.
Antes de hacer el import, Subversion abrira una ventana del editor default del sistema, para que escribamos un mensaje que explique el cambio. Esto debe hacerse en todas las operaciones de escritura al repositorio y es util ser concisos pero al mismo tiempo informativos al poner el comentario.
Notese que al final de la operacion, Subversion nos informa el numero de version que se aplica a estos cambios. Cada operacion donde se cambia el repositorio aumenta el numero de version por uno, independientemente de la cantidad de documentos modificados en ella.
Para verificar que el import funciono correctamente, podemos pedir un listado del contenido de la nueva carpeta en el repositorio, utilizando el comando ls:
$ svn ls svn://localhost/opt/svn/proyecto_ejemplo
branches/
tags/
trunk/
Otra manera de crear la estructura del repositorio es creando las carpetas directamente, utilizando el comando mkdir:
$ svn mkdir svn://localhost/opt/svn/otro_ejemplo -m 'nuevo proyecto'
Committed revision 2
En este caso, en lugar de esperar a que se nos muestre una ventana del editor, enviamos el mensaje junto con el comando utilizando la opcion -m. Esto podemos hacerlo con todos los comandos que escriben en el repositorio en lugar de utilizar el editor.
Una vez que se tiene un proyecto o estructura en el repositorio, la manera de trabajar con Suversion es extraer una copia del proyecto para realizar cambios y subirlos al terminar. Esta copia del proyecto se conoce como copia de trabajo y Subversion puede determinar exactamente que documentos se han agregado o han sido modificados mientras trabajos en ella.
El proceso de obtener del repositorio una copia del proyecto se conoce como checkout. El parametro que se pasa al comando ademas del path en el repositorio que queremos copiar es el nombre de la carpeta donde colocaremos la copia:
$ svn co svn://localhost/opt/svn/proyecto_ejemplo proyecto_ejemplo
A proyecto_ejemplo/trunk
A proyecto_ejemplo/tags
A proyecto_ejemplo/branches
Checked out revision 2
Los archivos del proyecto quedan guardados en la carpeta proyecto_ejemplo y Subversion nos informa que la version que ha obtenido es la 2. Una vez que se ha realizado el checkout podemos cambiarnos al directorio del proyecto y comenzar a trabajar.
Una vez que comenzamos a hacer modificaciones dentro del directorio del proyecto, Subversion lleva la cuenta de los cambios que hemos realizado y en cualquier momento podemos consultarlos:
$ cd trunk
$ echo "La capital de Francia es Tokio" > info.txt
$ svn status
? info.txt
En el ejemplo anterior, creamos un archivo de texto con una sola linea, llamado info.txt. Una vez creado el archivo, utilizamos el comando status para mostrar como Subversion ha detectado que existe un nuevo archivo en el directorio. El signo de interrogacion que aparece antes del nombre, significa que el archivo en cuestion no esta bajo control de versiones y Subversion lo desconoce.
Para agregar ese archivo al proyecto, utilizamos el comando add:
$ svn add info.txt
A info.txt
Subversion agrega el archivo info.txt a los que se encuentran bajo control de versiones, por lo que el status muestra ahora la letra A junto al nombre. Es importante hacer notar que este comando unicamente tiene efecto en nuestra copia de trabajo y no sube de inmediato el archivo al respositorio.
El comando add no esta limitado a agregar un solo archivo, por supuesto. Es posible incluir como parametro cualquier cantidad de archivos. Si se agrega un directorio, todos los archivos contenidos en el seran agregados recursivamente al proyecto.
Podemos hacer todos los cambios que necesitemos en nuestra copia de trabajo, si bien se recomienda subir la informacion al menos al final de cada sesion de trabajo y de preferencia cada vez que terminemos una tarea especifica de edicion. La razon es que mientras mas tiempo pasemos sin subir los cambios, mas dificil puede resultar integrarlos con otros cambios al repositorio, especialmente si muchas personas tienen acceso al mismo.
A la operacion de subir los cambios al repositorio se le llama commit. Una vez que hemos terminado nuestra sesion de trabajo, utilizamos ese comando para guardarlos en el repositorio:
$ svn commit -m 'se agrego archivo info'
Adding trunk/info.txt
Transmitting file data .
Committed revision 3.
El comando commit guarda todos los cambios realizados desde que inicio la sesion. En caso de no querer guardar todo, es posible especificar los archivos que deben subirse.
Para utilizar Subversion eficientemente, la rutina de trabajo que utilizamos debe cambiar un poco para incluir los momentos en que actualizamos o subimos archivos. Ademas, a lo largo del tiempo, el repositorio ira evolucionando y encontraremos necesidad de revisar cambios anteoriores y, si trabajamos con otras personas, de resolver conflictos.
Subversion tiene varios comandos para apoyarnos en ese ciclo basico de trabajo. En esta seccion conoceremos algunos de los mas importantes.
Lo primero que debemos hacer diariamente al iniciar una sesion de trabajo, es actualizar nuestra copia de trabajo del repositorio, para asegurarnos de trabajar con la version mas reciente de nuestros documentos. El comando para hacer esto se llama update:
$ svn update
At revision 3.
El comando actualiza los archivos que han cambiado, integrando al mismo tiempo nuestros cambios y nos muestra el status de lo que ha sido modificado, junto con la version a la que nos hemos actualizado. En el ejemplo anterior no hubo cambios que integrar.
Ahora supongamos que alguien ha agregado un titulo al archivo info.txt y ha subido sus cambios. Si hacemos un update ahora, veremos la diferencia:
$ svn update
U trunk/info.txt
Updated to revision 4.
En este caso, Subversion nos muestra el status U, que significa que un documento existente fue modificado.
Como el archivo info.txt ha sido modificado, quiza deseamos saber quien realizo la modificacion y cuando. Subversion ofrece el comando log para poder conocer la historia de commits de un archivo:
$ svn log info.txt
------------------------------------------------------------------------
r4 | predro | 2010-04-09 23:02:29 -0500 (Fri, 09 Apr 2010) | 1 line
se agrego titulo
------------------------------------------------------------------------
r3 | juan | 2010-04-09 22:41:55 -0500 (Fri, 09 Apr 2010) | 1 line
se agrego archivo info
------------------------------------------------------------------------
El comando nos muestra revision, autor, fecha y comentario por cada cambio que se ha hecho al archivo. En este caso, podemos ver que el usuario pedro agrego un titulo a nuestro archivo.
Si ademas de conocer al autor del cambio y su comentario queremos saber exactamente que texto ha cambiado en nuestro archivo, podemos utilizar el comando diff de Subversion para hacerlo:
$ svn diff -r3:4 info.txt
Index: info.txt
===================================================================
--- info.txt (revision 3)
+++ info.txt (revision 4)
@@ -1 +1,3 @@
+Sabia usted que...
+
La capital de Francia es Tokio
El comando diff acepta el parametro -r para especificar los numeros de versiones entre los que queremos conocer la diferencia. En este caso necesitamos conocer los cambios entre las revisiones 3 y 4, por lo que pasamos esos numeros. Se puede omitir el parametro -r y entonces Subversion nos dara las diferencias entre el estado actual del archivo y el estado que tenia la ultima vez que actualizamos el repositorio.
Lo que nos muestra el comando son las lineas que difieren entre una version y otra. Las lineas que tienen el simbolo + al lado izquierdo son las lineas que fueron agregadas entre la primera y la segunda version especificadas. En caso de que se hayan eliminado algunas lineas, estas tendran el simbolo - a su lado izquierdo.
Todavia podemos averiguar mas informacion sobre la historia de cambios del archivo. El comando blame nos muestra la ultima revision en que ha cambiado cada linea del archivo, junto con el nombre del autor del cambio:
$ svn blame info.txt
4 pedro Sabia usted que...
4 pedro
3 juan La capital de Francia es Tokio
Subversion nos permite tambien conocer el contenido completo de algun archivo en el momento en que determinada revision fue subida al repositorio. Por ejemplo, para ver el contenido del archivo info.txt en la revision 3:
$ svn cat -r3 info.txt
La capital de Francia es Tokio
En ocasiones, despues de haber realizado algunos cambios en un archivo, nos damos cuenta de que no queremos conservarlos, sino que deseamos volver a la version original del mismo. El comando revert anula cualquier cambio realizado a un archivo en la sesion actual, regresandolo al estado que tenia al momento de actualizar el repositorio por ultima vez:
$ svn revert info.txt
Reverted info.txt
|
|
|