Usando rebase do Git
Neste exemplo, abordaremos todos os comandos git rebase
disponíveis, exceto exec
.
Vamos começar nossa troca de base inserindo git rebase --interactive HEAD~7
no terminal. O editor de texto exibirá as seguintes linhas:
pick 1fc6c95 Patch A
pick 6b2481b Patch B
pick dd1475d something I want to split
pick c619268 A fix for Patch B
pick fa39187 something to add to patch A
pick 4ca2acc i cant' typ goods
pick 7b36971 something to move before patch B
Neste exemplo, vamos:
- Faça uma mesclagem squash do quinto commit (
fa39187
) no commit"Patch A"
(1fc6c95
) usandosquash
. - Mova o último commit (
7b36971
) para cima antes do commit"Patch B"
(6b2481b
) e mantenha-o comopick
. - Mescle o commit
"A fix for Patch B"
(c619268
) no commit"Patch B"
(6b2481b
) e desconsidere a mensagem do commit usandofixup
. - Divida o terceiro commit (
dd1475d
) em dois commits menores usandoedit
. - Corrija a mensagem do commit digitado incorretamente (
4ca2acc
) usandoreword
.
Ufa! Parece muito trabalho, mas, executando uma etapa de cada vez, podemos fazer essas alterações facilmente.
Para começar, precisamos modificar os comandos no arquivo para que fiquem assim:
pick 1fc6c95 Patch A
squash fa39187 something to add to patch A
pick 7b36971 something to move before patch B
pick 6b2481b Patch B
fixup c619268 A fix for Patch B
edit dd1475d something I want to split
reword 4ca2acc i cant' typ goods
Alteramos o comando de cada linha de pick
para o comando no qual estamos interessados.
Agora, salve e feche o editor. Isso iniciará o rebase interativo.
O Git ignora o primeiro comando de troca de base, pick 1fc6c95
, pois ele não precisa fazer nada. Ele vai para o próximo comando, squash fa39187
. Como essa operação requer entrada de dados, o Git abre o editor de texto novamente. O arquivo aberto é parecido com este:
# This is a combination of two commits.
# The first commit's message is:
Patch A
# This is the 2nd commit message:
something to add to patch A
# Please enter the commit message for your changes. Lines starting
# with '#' will be ignored, and an empty message aborts the commit.
# Not currently on any branch.
# Changes to be committed:
# (use "git reset HEAD <file>..." to unstage)
#
# modified: a
#
Este arquivo é a maneira do Git de dizer: "Ei, é isto o que vou fazer com este squash
". Ele lista a mensagem do primeiro commit ("Patch A"
) e a mensagem do segundo commit ("something to add to patch A"
). Se você estiver satisfeito com essas mensagens, salve o arquivo e feche o editor. Caso contrário, tem a opção de alterar a mensagem do commit simplesmente mudando o texto.
Depois que o editor é fechado, o rebase continua:
pick 1fc6c95 Patch A
squash fa39187 something to add to patch A
pick 7b36971 something to move before patch B
pick 6b2481b Patch B
fixup c619268 A fix for Patch B
edit dd1475d something I want to split
reword 4ca2acc i cant' typ goods
O Git processa os dois comandos pick
(para pick 7b36971
e pick 6b2481b
). Ele também processa o comando fixup
(fixup c619268
), pois não exige nenhuma interação. fixup
mescla as alterações de c619268
no commit antes dele, 6b2481b
. As duas alterações terão a mesma mensagem de commit: "Patch B"
.
O Git chega à operação edit dd1475d
, para e imprime a seguinte mensagem no terminal:
You can amend the commit now, with
git commit --amend
Once you are satisfied with your changes, run
git rebase --continue
Neste ponto, você pode editar qualquer arquivo no projeto para fazer outras alterações. Para cada alteração feita, você precisará executar um novo commit e poderá fazer isso inserindo o comando git commit --amend
. Quando terminar de fazer todas as alterações, execute git rebase --continue
.
Em seguida, o Git chega ao comando reword 4ca2acc
. Ele abre o editor de texto mais uma vez e apresenta as seguintes informações:
i cant' typ goods
# Please enter the commit message for your changes. Lines starting
# with '#' will be ignored, and an empty message aborts the commit.
# Not currently on any branch.
# Changes to be committed:
# (use "git reset HEAD^1 <file>..." to unstage)
#
# modified: a
#
Como antes, o Git mostra a mensagem do commit para você editar. Você pode alterar o texto ("i cant' typ goods"
), salvar o arquivo e fechar o editor. O Git terminará o rebase e retornará para o terminal.
Fazer push de código com rebase para o GitHub
Como você alterou o histórico do Git, a git push origin
normal não funcionará. É preciso modificar o comando forçando o push das alterações mais recentes:
# Don't override changes
$ git push origin main --force-with-lease
# Override changes
$ git push origin main --force
Forçar push tem implicações sérias, pois ele muda a sequência histórica de commits para o branch. Use-o com cuidado, especialmente se o repositório estiver sendo acessado por várias pessoas.