/

Squash or not Squash

¡Hola a todos! El otro día hablando con Alejandro Campos sobre el uso de Git nos surgió una duda: ¿hacer un squash es lo mismo que quedarse con el último commit? La verdad es que es algo que no sé así que vamos a descubrirlo.

¿Qué es un squash?

Por simplificar diremos que un squash es juntar los cambios de varios commits en uno solo.

En mi caso la forma que tengo de incluir código en mi rama develop es mediante Pull Request. Yo trabajo en mi rama local y hago cuantos commits quiera. Antes de hacer la Pull Request hago un squash y junto todos esos cambios en un commit.

¿Cómo trabaja Git?

Si leemos la documentación oficial de Git podemos ver que para Git los cambios son instantáneas de cómo están los archivos en ese momento. Veamos que dice la documentación de Git:

Sistemas como CVS, Subversion, Perforce, Bazaar, etc. modelan la información que almacenan como un conjunto de archivos y las modificaciones hechas sobre cada uno de ellos a lo largo del tiempo.

CVS

Cada vez que confirmas un cambio, o guardas el estado de tu proyecto en Git, él básicamente hace una foto del aspecto de todos tus archivos en ese momento, y guarda una referencia a esa instantánea.

Git

Visto que el último commit es una instantánea del estado del proyecto en ese momento ¿por qué no pensar que subir el último commit equivale a hacer el squash de todos los commits anteriores?

Nota: Conceptualmente son cosas muy diferentes pero quiero comprobar si el código que se acaba subiendo es el mismo.

Demo

Vamos a hacer una prueba sencilla reproduciendo los commits de la imagen anterior. Partimos desde la rama master con 3 ficheros: a.md, b.md, c.md.

Git

Creamos la rama changes y aplicamos todos los cambios sobre los ficheros hasta llegar al estado deseado.

Rama changes

Squash

Tenemos todos los cambios creados en esa rama. Ahora vamos a crear la rama squash en la que haremos un squash de estos commits, manteniendo a la rama changes ajena a estos cambios.

git rebase -i origin/master

pick 07f01bc Version 2
s 2636059 Version 3
s ae0ce02 Version 4
s 4ac0698 Version 5

Una vez que hemos hecho el squash y hemos subido la rama vamos a crear una Pull Request contra master a ver qué cambios nos genera.

Pull Request - Squash

Como podemos ver en la Pull Request solo tenemos 1 commit y los cambios en los ficheros nos dejan la rama master en el estado deseado.

Último Commit

Ahora vamos a ver qué pasa cuando subimos sólo el último commit. Para hacer esto posible creamos la rama lastCommit desde master y hacemos un cherry-pick para traernos solamente el último commit.

git cherry-pick 4ac0698

Al hacer un cherry-pick nos aparecen conflictos en B y C. Esto no parece muy normal, si de verdad Git guarda instantáneas del código simplemente debería sobrescribir B y C con los cambios del último commit. De hecho me extraña más que no haya un conflicto en A ya que el último commit está referenciando el estado de A del commit anterior. Recordemos el gráfico.

Git

Más raro aun es en la resolución de conflictos que me ofrece quedarme con los cambios del Source (mi último commit) o con los del Target (mi rama lastCommit) pero en Result me aparece el estado anterior al último commit de B, es decir, B1.

Conflictos

Como vemos pasan cosas muy raras cuando intentamos traernos solo el último commit de una serie de cambios. Si alguien me explica por qué pasan estas cosas le estaré agradecido.

Para seguir con el experimento vamos a resolver los conflictos y hacer la Pull Request.

Pull Request - Last Commit

Viendo esta Pull Request se me hace difícil creer que Git almacene instantáneas del estado del repositorio en vez de los diferenciales. Como vemos, hemos conseguido llevar al estado deseado tanto B como C gracias a la resolución de conflictos. Pero A parece que no ha sufrido ningún cambio respecto a master.

Tanto los conflictos como la Pull Request se explicarían perfectamente si Git almacenara los diferenciales en cada commit. Porque los conflictos se han originado con el estado inmediatamente anterior y el último commit no aplicaba cambios sobre A, por eso en la Pull Request no hay ni rastro de A.

Impresiones y conclusiones

Como comentaba antes, se me hace difícil creer que Git funcione como dice que funciona. Para más información os dejo la documentación. El comportamiento que he observado es más propio de un sistema que almacene los diferenciales entre commits. Entiendo que Git no nos está mintiendo así que me gustaría que alguien me explicara todo esto.

Os he dejado el repositorio con este ejemplo en el GiiHub de Ikeín y yo con las ramas y las Pull Request para que podáis jugar con ellas.

Lo que si hemos demostrado es que no es lo mismo, a nivel del estado final del repositorio, subir sólo el último commit que hacer un squash de los cambios.

Me encantaría que en los comentarios alguien pudiera explicarme el porqué de estos comportamiento tan raros.

Un saludo y…

¡Nos vemos en el futuro!