Git: Difference between revisions

From Peters wiki
Jump to navigation Jump to search
mNo edit summary
mNo edit summary
Line 331: Line 331:
When you have resolved the conflicts, commit the merge result.
When you have resolved the conflicts, commit the merge result.


== Rebase ==
== Rebase:a ==


Rebase is a way to forward-port your commits to a new upstream HEAD. Assume we have made 2 commits on a test branch. After we branched, someone has made a new commit on master:
Rebase är ett sätt att framåt-porta dina commit:s till en nyare HEAD. Antag att vi har skapat 2 commit:s på en branch som heter test. Efter att vi branch:at så har någon annan gjort en nya commit master:


[[Image:git-rebase-before.png]]
[[Image:git-rebase-before.png]]


Instead of merging from master to test, we can forward-port our 2 commits. While on test branch, run the command:
Istället för att merga:a från master till test, vi kan rebase:a våra 2 commit:s. När man är på test branch:en, kör kommandot:
{{Cmd|git rebase master}}
{{Cmd|git rebase master}}
In this example we will not have any merge conflicts, so git will automatically do the rebase for us:
I detta exempel uppstår inga merge konflikter, git kommer automatiskt göra rebase:en åt oss:
{{Bc|1=
{{Bc|1=
First, rewinding head to replay your work on top of it...
First, rewinding head to replay your work on top of it...
Line 346: Line 346:
}}
}}


Our 2 commits have now been modified and based on the new HEAD on master:
Våra 2 commit:s har nu blivit modifierade och baseras nu på den nya HEAD master.


[[Image:git-rebase-after.png]]
[[Image:git-rebase-after.png]]


{{Warning|Do not rebase commits that you have pushed to a public repository!}}
{{Watning|Gör inte rebase på commit:s som du har push:at till publikt repository!}}
However, rebasing your local branch is preferred over merge, since you will create a cleaner history tree. It does not matter if you merge or rebase, any conflicts have to be resolved. If we now do a merge on master from test, git will default do a fast-forward merge (see [[#No merge needed (default)]]
Men en rebase på en lokal branch är att föredra framför en merge, eftersom du kommer skapa en renare historia. Det spelar ingen roll om du merge:ar eller rebase:ar, varje konflikt måste lösas ändå. Om vi nu merge:ar till master, så kommer git default att göra en fast-forward merge.


== Cherry-pick ==
== Cherry-pick ==

Revision as of 06:49, 20 April 2013


Denna artikel beskriver några kommandon och användarfall med git. (Jag håller på att skriva om sidan från engelska till svenska då jag nu insåg att alla mina andra sidor är på svenska.)

Inställningar

Git inställningar kan var global eller per repository (sparade under .git mappen i ett repository). Globala inställningar är sparade i filen ~/.gitconfig. Det går att läsa och editera filen ~/.gitconfig manuellt, eller använda kommandot git config --global. Några exempel hur man sätter, tar bort och läser inställningar:

user $ git config --global core.pager cat
user $
git config --global --unset core.pager
user $
git config --global --get core.pager


Notering: Inställningar sparade i repositoryt är bara temporära. Dom inkluderas inte när du t.ex. klonar ett repository.


Vem är du?

Ställ in namn och email adress:

user $ git config --global user.name "Förnamn Efternamn"
user $
git config --global user.email fornamn.efternamn@exempel.com

Pager

Du kan välja vilken 'pager' (t.ex. more, less, cat) Git ska använda. Kör detta kommando om du vill att Git ska använda cat:

user $ git config --global core.pager cat

Snyggare log

För att skapa alias i Git, som t.ex. att skriva ut en kompakt och snygg log med kommandot git hist, kör kommandot:

user $ git config --global alias.hist 'log --pretty=tformat:"%h %ad | %s%d [%an]" --graph --date=short'

Vi använder tformat istället för format för att få med ett avslutande 'carriage return' ifall du använder cat som din pager. Utskriften från git hist kommer att se ut så här:

* 40ec614 2012-12-10 | The best commit ever (HEAD, master) [Peter Kerwien]
* 4b732d0 2012-12-10 | Initial commit [Peter Kerwien]

Repository

Skapa ett repository

För att skapa ett repository i en icke-existerande katalog, kör:

user $ git init repo

Eller om du vill skapa ett repository i nuvarande katalog:

user $ cd <path>/repo
user $
git init

Båda kommandona kommer skapa en dold katalog som heter .git. Dessa repositoryn kan också innehålla filerna som man jobbar med dvs ett 'workspace'. Om du vill skapa ett repository som ska fungera som ett centralt repository eller dela mellan flera användare och alla ska kunna pusha till repositoryt så måste man skapa ett s.k. bare repository. Ett bare repository saknar workspace och alla git-filer ligger synliga direkt i repositoryt och inte i en katalog som heter .git. För att skapa ett bare repository, kör::

user $ git init --bare repo.git

Namnkonventionen är att bare repositorys slutar med ändelsen .git i namnet.

Klona ett repository

För att klona ett repository, dvs skapa en kopia av alla dess versioner, kör:

user $ git clone repo dir

Det spelar ingen roll om du klonar ett vanligt eller bare repository, du kommer få en klon med ett workspace. Men om du vill klona utan ett workspace, kör kommandot:

user $ git clone --bare repo dir

Din klon med namnet dir kommer nu inte innehålla något workspace.

Rensa workspace

Efter en del arbete kommer ditt workspace innehålla versionshanterade, ändrade men också genererade och backup-filer. En del av de icke-versionshanterade filerna listas med git status, men en del filer ignoreras via en .gitignore fil. För att rensa hela ditt workspace från alla icke-versionshanterade inklusive ignorerade filer, gå till toppen av repositoryt och kör:

user $ git clean -fdx

Använd -n optionen för att endast lista vilka filer som skulle ha raderats.

Versionshantera, komitta och sluta versionshantera förändringar

Versionshantera förändringar

För att addera en eller flera filer till git, kör:

user $ git add <pattern>

Om en fil redan versionshanteras, kommer kommandot att addera förändringar till index. Om du vill addera alla gjorda förändringar till index, kör kommandot:

user $ git add -u

Commit:a förändringar

För att commit:a förändringar redan adderade till index, kör:

user $ git commit -m "Commit message"

Om man vill commit:a alla ändringar som är gjorda i redan versionshanterade filer, men ej adderade till index, kör kommandot:

user $ git commit -am "Commit message"

Sluta versionshantera förändringar

För att ta bort en eller flera filer från workspace och index, kör kommandot:

user $ git rm <pattern>

Filerna kommer att tas bort från repositoryt när du commit:ar till repositoryt.

Lista förändringar

Lista förändringar i workspace

För att lista förändringar du har gjort i workspace, men som inte än är adderade till index, kör kommandot:

user $ git diff

För att lista förändringar mellan ditt index och HEAD, dvs versionen du ser i repositoryt, kör kommandot:

user $ git diff --cached

Om du vill se alla dina icke-commit:ade förändringar (både adderade till index eller inte), kör kommandot:

user $ git diff HEAD

Graphical diff using meld

If you have set up meld as your graphical diff-tool (See BBI Git#Using meld as graphical diff and merge tool (Optional)), you can perform graphical diff by running:

user $ git difftool -g

Just replace git diff with git difftool -g in the examples above. A diff with meld can look like this:

(To the left we have the original version and to the right our modified Makefile).

Jämföra commits

Antag att vi har följande historia:

* 2634685 2012-12-18 | Commit 4 (HEAD, master) [Peter Kerwien]
* d0afcbd 2012-12-18 | Commit 3 [Peter Kerwien]
* aaa2574 2012-12-18 | Commit 2 [Peter Kerwien]
* b61568c 2012-12-18 | Initial commit [Peter Kerwien]

För att visa alla förändringar mellan commit 2 och 3, kör:

user $ git diff b61568c d0afcbd

Du kan också köra kommandot:

user $ git diff HEAD~3 HEAD~1

Välja revision

Släkt-referenser

Antag att vi har följande historia:

* 5b38862 2013-01-16 | Commit 5 (HEAD, master) [Peter Kerwien]
*   9940a50 2013-01-16 | Commit 4 [Peter Kerwien]
|\ 
| * fc796ce 2013-01-16 | Commit on test (test) [Peter Kerwien]
* | f7935db 2013-01-16 | Commit 3 [Peter Kerwien]
|/ 
* dc53825 2013-01-16 | Commit 2 [Peter Kerwien]
* bea9676 2013-01-16 | Initial commit [Peter Kerwien]

Istället för att ange hela hashen för att välja version, kan man ange släkt-referenser genom att addera ^ eller ~ efter versionen. HEAD^ betyder den första föräldern till HEAD (=9940a50). Commit:en 9940a50 har två föräldrar, därför väljer 9940a50^ den föräldern som finns på nuvarande branch. Om vi är på master så kommer 9940a50^ välja commit:en f7935db. 9940a50^2 kommer välja den andre föräldern, i detta fall fc796ce. Man kan också lägga till flera ^ efter varandra. HEAD^^ kommer välja commit f7935db.

~ däremot betyder alltid den första föräldern av en commit. Så HEAD~ och HEAD^ betyder alltid samma sak. Men HEAD~2 betyder "första föräldern till första föräldern", så 9940a50~2 väljer commit:en dc53825. Detta är inte samma sak som 9940a50^2.

Man kan kombinera dessa två, t.ex. HEAD~^2. Detta betyder "den andra föräldern till den första föräldern till HEAD", dvs commit:en fc796ce.

Ångra och ändra

Notering: Ändra inte eller ta bort commit:s som du redan har delat med andra. Det är bara säkert att ändra historian om det bara påverkar dig själv.


Ångra ändringar som inte har adderats till index

För att ångra ändringar som inte har adderats till index, kör kommandot:

user $ git checkout -- <file>

Ångra ändringar som har adderats till index

För att ångra ändringar som har adderats till index, kör:

user $ git reset HEAD <file>

Reset kommandot återställer index till vad som pekas ut av HEAD. Reset kommandot återståller inte workspace. Så workspace kan fortfarande innehålla oönskade ändringar.

Rätta senaste commit

Om du upptäcker att någonting saknas i den senaste commit:en och du vill inte göra en ny, då kan du rätta till ditt misstag med kommandot:

user $ git commit --amend -m "..."

Du måste dock först addera vad du saknade till index innan du kör kommandot ovan.

Ångra en commit

För att ångra en commit kan vi skapa en commit som tar bort dom förändringar som gjordes i den oönskade commit:en. För att ånga den senaste gjorda commit:en kör kommandot:

user $ git revert HEAD

Om du vill ångra någon annan commit, ersätt HEAD med hashen. Använd optionen --no-edit om du vill acceptera standard commit meddelandet för en revert.

Ta bort en commit på en branch

När man ångrar en commit med git revert kommer både den felaktiga och återställningen synas i historian. Om man vill ångra commit:en och radera den från historian kan man flytta HEAD pekaren tillbaka till commit:en före den felaktiga och börja om från början. Antag att man har en historia som ser ut så här:

* e5f1c95 2012-12-10 | Another good commit (HEAD, master) [Peter Kerwien]
* e491e6d 2012-12-10 | Good commit [Peter Kerwien]
* ba30da7 2012-12-10 | Bad commit [Peter Kerwien]
* 4b732d0 2012-12-10 | Initial commit [Peter Kerwien]

Om man vill slänga bort dom 3 senaste commit:erna kan man flytta HEAD till commit 4b732d0 med kommandot:

user $ git reset --hard 4b732d0
HEAD is now at 4b732d0 Initial commit

När man sen genomför en ny commit, kommer historian se ut så här:

* 40ec614 2012-12-10 | The best commit ever (HEAD, master) [Peter Kerwien]
* 4b732d0 2012-12-10 | Initial commit [Peter Kerwien]

git reset --hard kommer även radera ev. icke-commit:ade ändringar i workspace. Om du vill spara alla förändringar (oavsett om de adderats till index eller inte), använd istället --soft optionen. De commit:s som blev slängda kommer fortfarande finnas kvar tills man städar repositoryt. Man kommer åt dessa commit:s via deras hash eller taggar. Man kan köra kommandot git reflog för att se var HEAD varit den senaste tiden.

Branchning

Skapa och byta branch

Om man vill skapa och byta till en branch som heter test, kör kommandot:

user $ git checkout -b test

Man kan också specifiera en annan branchnings punkt än HEAD via:

user $ git checkout -b test tag

Eller:

user $ git checkout -b test hash

Lista brancher

För att lista alla lokala brancher, kör:

user $ git branch

Den aktuella branchen är markerad med *. Om man också vill se remote brancher, kör:

user $ git branch --all

Om man bara vill lista alla lokala brancher som börjar på te, kör kommandot:

user $ git branch --list te*

Växlar mellan brancher

Det är väldigt enkelt att växla mellan brancher, kör bara kommandot:

user $ git checkout <branch name>

Git kommer att varna och stoppa dig om du har förändringar i workspace som inte är commit:ade.

Byta namn på en branch

För att döpa om branchen test till dev, kör kommandot:

user $ git branch -m test dev

Radera en branch

För att radera en branch som heter test, kör:

user $ git branch -d test

Git kommer varna dig om branchen inte har mergats ordentligt till annan branch. För att radera den trots detta, kör kommandot:

user $ git branch -D test

Följa en lokal branch i ett annat repository

Två användare har klonat samma repository (helloworld.git). Användare A skapar en lokal branch som heter test:

user $ git checkout -b test

Sen skapar A en ändring och commit:ar den till sitt repository:

user $ git commit -am "Commit made on test branch"

Nu vill B följa denna branch. B måste då först lägga till As repository som en remote:

user $ git remote add <local name> <url>

I detta exempel kommer B kalla As repository för hello_a och URL:en är en absolut sökväg till repositoryt:

user $ git remote add hello_a /home/a/git/helloworld

Visa detaljer om hello_a med kommandot:

user $ git remote show hello_a
* remote hello_a
  Fetch URL: /home/a/git/helloworld
  Push  URL: /home/a/git/helloworld
  HEAD branch: test
  Remote branches:
    master new (next fetch will store in remotes/hello_a)
    test   new (next fetch will store in remotes/hello_a)
  Local ref configured for 'git push':
    master pushes to master (up to date)

Innan B kan följa test branschen, måste B hämta heads, taggar, och objekt från As repository:

user $ git fetch hello_a
remote: Counting objects: 5, done.
remote: Compressing objects: 100% (3/3), done.
remote: Total 3 (delta 2), reused 0 (delta 0)
Unpacking objects: 100% (3/3), done.
From /home/a/git/helloworld
* [new branch]      master     -> hello_a/master
* [new branch]      test       -> hello_a/test

B kan nu skapa en lokal branch som B också kallar test som följer As test branch:

user $ git branch --track test hello_a/test
Branch test set up to track remote branch test from hello_a.

Byt till test branchen:

user $ git checkout test

Merge

Olika typer av merge i git

I följande exempel har vi gjort 2 kommit:s på en branch som heter test:

Ingen merge behövs (standard)

Om ingen commit har gjorts på master branch:en sen test skapades, kommer git göra en s.k. fast-forward merge om vi merge:ar test till master:

Git bara flyttar HEAD framåt. Så alla commit:s vi har gjort på test branch:en kommer "flyttas" till master. Om vi pushar till origin kommer vi push:a två commit:s.

Ingen merge behövs men med optionen --no-ff

Om ingen commit har gjorts på master branch:en sen test skapades, kommer git göra en s.k. fast-forward merge om vi merge:ar test till master. Men i detta fall använder vi --no-ff optionen:

Git skapar nu en merge-commit på master.

Notering: Om vi push:ar master till origin, kommer vi inte bara push:a denna merge-commit, utan vi kommer push:a tre commit:s.


Merge behövs

Antag att någon har commit:at till master efter att vi branch:ade:

I detta fall måste git göra en vanlig merge, så vi behöver inte ange --no-ff optionen:

Git skapar nu en merge-commit på master.

Notering: Om vi push:ar master till origin, kommer vi inte bara push:a denna merge-commit, utan vi kommer push:a tre commit:s.


Resolve merge conflicts with meld

If you have set up meld as your graphical merge-tool (See ?), you can perform graphical merge by running:

user $ git mergetool

Example: After a merge from test branch to master, git prints out the following:

Auto-merging main.c
CONFLICT (content): Merge conflict in main.c
Auto-merging Makefile
CONFLICT (content): Merge conflict in Makefile
Automatic merge failed; fix conflicts and then commit the result.

Start the graphical merge:

user $ git mergetool

Each file will be opened in meld for resolving the conflicts. With the configuration where meld is given 4 arguments, it will resolve all non-conflicting changes, and then leave the rest for the user to resolve. The meld window is divided into three parts. From left to right we have:

  • The version on master branch (to-version)
  • The merge result
  • The version on test branch (from-version)

Resolve the merge by selecting the code from the left or right section into the middle section. You can also edit the code in the middle section to create the correct merge result. When done save the result.

When you have resolved the conflicts, commit the merge result.

Rebase:a

Rebase är ett sätt att framåt-porta dina commit:s till en nyare HEAD. Antag att vi har skapat 2 commit:s på en branch som heter test. Efter att vi branch:at så har någon annan gjort en nya commit på master:

Istället för att merga:a från master till test, vi kan rebase:a våra 2 commit:s. När man är på test branch:en, kör kommandot:

user $ git rebase master

I detta exempel uppstår inga merge konflikter, så git kommer automatiskt göra rebase:en åt oss:

First, rewinding head to replay your work on top of it...
Applying: Added unused attribute on argc & argv
Applying: Replaced __attribute__((unused)) with -Wno-unused-parameter

Våra 2 commit:s har nu blivit modifierade och baseras nu på den nya HEAD på master.

Template:Watning Men en rebase på en lokal branch är att föredra framför en merge, eftersom du kommer skapa en renare historia. Det spelar ingen roll om du merge:ar eller rebase:ar, varje konflikt måste lösas ändå. Om vi nu merge:ar till master, så kommer git default att göra en fast-forward merge.

Cherry-pick

Simple cherry-picking without conflict

If you want to apply changes from one branch to another, but you want to select exactly which commits to choose, you can do a so called cherry-pick. Assume we have the following history:

* b5df0bb 2012-12-17 | Added main() header (HEAD, test) [Peter Kerwien]
| * 12ad7c3 2012-12-17 | Whitespace edit in main.c (master) [Peter Kerwien]
| * f39ba2f 2012-12-14 | Add some file headers [Peter Kerwien]
| * 02d1638 2012-12-14 | Change hello message [Peter Kerwien]
|/ 
* 0e1085d 2012-12-14 | Made some cleanup [Peter Kerwien]
* 10213e2 2012-12-14 | Initial commit [Peter Kerwien]

We want to apply the commit f39ba2f to our test branch. To do that, switch to the test branch and run:

user $ git cherry-pick f39ba2f
[test 81d9476] Add some file headers
2 files changed, 4 insertions(+)

The history will now look like:

* 81d9476 2012-12-14 | Add some file headers (HEAD, test) [Peter Kerwien]
* b5df0bb 2012-12-17 | Added main() header [Peter Kerwien]
| * 12ad7c3 2012-12-17 | Whitespace edit in main.c (master) [Peter Kerwien]
| * f39ba2f 2012-12-14 | Add some file headers [Peter Kerwien]
| * 02d1638 2012-12-14 | Change hello message [Peter Kerwien]
|/ 
* 0e1085d 2012-12-14 | Made some cleanup [Peter Kerwien]
* 10213e2 2012-12-14 | Initial commit [Peter Kerwien]

Note that the commit f39ba2f is called 81d9476 on our test branch.

Multiple cherry-picking with conflict

During cherry-picking, conflicts might have to be resolved. In this example, we will cherry-pick two commits, where the first one will conflict with our changes on the test branch. Assume the following history:

* e108d4a 2012-12-17 | Hello test! (HEAD, test) [Peter Kerwien]
| * fdb65d7 2012-12-17 | Return EXIT_SUCCESS instead of 0 (master) [Peter Kerwien]
| * 69e7150 2012-12-17 | Added argc & argv parameters to main() [Peter Kerwien]
| * 413565e 2012-12-17 | Changed hello message [Peter Kerwien]
|/ 
* 3ff2e68 2012-12-17 | Initial commit [Peter Kerwien]

Switch to the test branch, and cherry-pick commits 413565e and fdb65d7. The first one will conflict with our commit e108d4a:

user $ git cherry-pick 413565e fdb65d7
error: could not apply 413565e... Changed hello message
hint: after resolving the conflicts, mark the corrected paths
hint: with 'git add <paths>' or 'git rm <paths>'
hint: and commit the result with 'git commit'

Resolve the conflict and commit the merge result. Now, to continue the cherry-picking, run the command:

user $ git cherry-pick --continue
$ git cherry-pick --continue
[test 9eac083] Return EXIT_SUCCESS instead of 0
1 file changed, 2 insertions(+), 1 deletion(-)

Now our history looks like:

* 9eac083 2012-12-17 | Return EXIT_SUCCESS instead of 0 (HEAD, test) [Peter Kerwien]
* bfc1ad9 2012-12-17 | Changed hello message [Peter Kerwien]
* e108d4a 2012-12-17 | Hello test! [Peter Kerwien]
| * fdb65d7 2012-12-17 | Return EXIT_SUCCESS instead of 0 (master) [Peter Kerwien]
| * 69e7150 2012-12-17 | Added argc & argv parameters to main() [Peter Kerwien]
| * 413565e 2012-12-17 | Changed hello message [Peter Kerwien]
|/ 
* 3ff2e68 2012-12-17 | Initial commit [Peter Kerwien]

We have now applied the two commits, with the new hash values bfc1ad9 and 9eac083.

Squash

When doing a push, several commits might be pushed. This is not always a wanted behaviour, since it might be a lot of commits from our local development branch. If you push to a central repository, all you intermediate commits will be visible to others that clones this repository. If you want to push a feature as one single commit, you can use squashing. A squash will create a new commit which is the summary of several commits.

Assume we have developed a feature on a branch called test. Everything is rebased to latest commit on master:

Create a new branch from master called test-squashed:

user $ git checkout -B test-squash master

Merge all commits made on test, but create one single commit of them:

user $ git merge --squash test

Commit the changes:

user $ git commit -am "Added input arguments to main()"

We have now created one commit on the test-squash branch, which has the complete contents of the commits on the test branch:

Switch to master and merge from test-squash (fast-forward merge):

user $ git checkout master
user $
git merge test-squash

If we push to origin/master, only one commit will be added:

File:Git-squash-4.png

Terminology

Term Description
Bare repository Ett git repository utan workspace.
Klon En kopia av ett repository. En klon innehåller alla versioner av repositoryt.
Workspace Katalogen som innehåller filerna som versionshanteras. Det är här man kan editera, skapa nya filer och commit:a sina ändringar.
HEAD En pekare som pekar ut den just nu valda versionen av repositoryt.
Staged (Saknar bra svensk översättning). Adderad till index men inte commit:ad.
Cached (Saknar bra svenska översättning). Se staged.
Versionshanterad fil En fil i workspace som är adderad och versionshanterad i git repositoryt.
Ej versionshanterad fil En fil i workspace som inte är adderad och versionshanterad i git repositoryt.
Index Innehåller alla förändringar i versionshanterade filer som kommer commit:as när man kör kommandot git commit.