Git Notes
Here are some common git operations for which we often find ourselves searching.
Contents
- 1 Workflow
- 1.1 Checkout a remote branch 'develop' and keep it linked to the origin
- 1.2 List all branches (both local and remote)
- 1.3 List local branches and their corresponding upstream remote branches
- 1.4 Revert local changes to a file (prior to commit)
- 1.5 Revert all local changes to the current subtree (prior to commit)
- 1.6 Cute hack to revert all local changes (prior to commit)
- 1.7 Undo a commit
- 1.8 Undo multiple commits
- 1.9 Update to latest HEAD, preserving local changes and local commits on top
- 1.10 Push changes on master to origin
- 1.11 Delete untracked files and directories
- 1.12 Interactively stage patches from changed file(s)
- 1.13 Roll back the last commit's changes to a particular file
- 2 Branches
- 2.1 Branch master to a new local branch "new_branch"
- 2.2 Push local branch to remote
- 2.3 Make the current local branch start tracking a corresponding remote branch
- 2.4 List the local branches that have already been merged to this one
- 2.5 Diff a file between two branches
- 2.6 Delete a branch both locally and remotely
- 2.7 Move a commit from bad_branch to good_branch
- 2.8 Rename the current branch
- 3 Git + SVN
- 4 Searching
- 5 History
- 5.1 Display a log with colored word diffs
- 5.2 Display a diff with colored words between a file in one commit and a file in another commit
- 5.3 Display all contributing authors of a project including their e-mail
- 5.4 Viewing the history for a single file
- 5.5 See commits in branch B not present in branch A
- 6 Scripts
- 7 Advanced and/or dangerous
- 8 Rewriting history
- 9 Tutorials
Workflow
Checkout a remote branch 'develop' and keep it linked to the origin
1 | git checkout -t origin /develop |
It should report: "Branch develop set up to track remote branch develop from origin."
List all branches (both local and remote)
1 | git branch -a |
List local branches and their corresponding upstream remote branches
1 | git branch -vv |
Revert local changes to a file (prior to commit)
1 | git checkout file |
Revert all local changes to the current subtree (prior to commit)
1 | git checkout . |
Cute hack to revert all local changes (prior to commit)
1 | git stash git stash drop |
Undo a commit
1 | git reset --soft HEAD^ |
Undo multiple commits
1 | git reset --soft HEAD@{2} |
Where 2
is the number of commits to undo.
Update to latest HEAD, preserving local changes and local commits on top
1 2 3 | git stash git pull --rebase git stash pop |
Push changes on master to origin
1 | git push origin master |
Delete untracked files and directories
1 | git clean - df |
Interactively stage patches from changed file(s)
1 | git add -p <path> |
Roll back the last commit's changes to a particular file
1 | git checkout HEAD@{1} -- file |
Branches
Branch master to a new local branch "new_branch"
1 | git checkout -b new_branch master |
Push local branch to remote
1 | git push -u origin new_branch |
Make the current local branch start tracking a corresponding remote branch
1 | git branch -- set -upstream-to origin /new_branch |
This is not necessary if you used git push -u
as suggested above.
List the local branches that have already been merged to this one
1 | git branch --merged |
Diff a file between two branches
1 | git diff branch1 branch2 -- file |
Delete a branch both locally and remotely
1 2 3 | git branch -rd origin /branch_to_kill git branch -d branch_to_kill git push origin :branch_to_kill |
Move a commit from bad_branch to good_branch
1 2 3 4 5 6 7 8 | # First cherry-pick the commit onto the correct branch: git checkout good_branch git cherry-pick deadbeef # Then remove the commit from the bad branch: git checkout bad_branch git rebase -i # Change the undesirable commit to "noop" |
For more on branching, see Git topic branches.
Rename the current branch
1 | git branch -m <new branch name> |
Git + SVN
Clone an SVN repository to a local Git repository
1 | git svn clone -s http: //svn .code.sf.net /p/jhotdraw/svn/ |
Commit and push changes, even with local changes in the working copy
1 | git commit git stash git svn dcommit git stash pop |
Update to latest trunk, preserving local changes and local commits on top
1 | git stash git svn rebase git stash pop |
Searching
Recursively search for HelloWorld.file (and display the most recent commit modifying it)
1 | git ls -tree -r HEAD | grep HelloWorld. file |
Recursively search for all files containing the phrase 'import HelloWorld'
1 | git grep 'import HelloWorld' |
Recursively search for all files in any topic branch containing the phrase 'import HelloWorld'
1 | git grep 'import HelloWorld' $(git rev-list --all --no-walk) |
History
Display a log with colored word diffs
1 | git log -p --color-words |
Add -S
to less
to virtually wrap long lines.
Display a diff with colored words between a file in one commit and a file in another commit
1 | git diff <commitA>:< file > <commitB>:< file > --color-words |
Add -S
to less
to virtually wrap long lines.
Display all contributing authors of a project including their e-mail
1 | git log -- format = '%aN <%ae&>' | sort -u |
Respects .mailmap
.
Viewing the history for a single file
1 | git log --follow HelloWorld. file |
This history is algorithmically calculated and must be carefully preserved.
Simultaneous (within a single commit) significant changes + file renaming (including relocation) can prevent the algorithm from successfully tracing the file's history, or cause it to begin tracing the wrong file.
Keeping code changes separate from renames should prevent this confusion, but it is good practice to check log --follow
before pushing to a remote repository.
See commits in branch B not present in branch A
There are two main options. The first:
1 | git log A..B |
will display the different commits in full git log format. NB: the ..
between commits is important to sure only the difference in commits is considered.
The second:
1 | git cherry - v A B |
will display a simple list of the different commits, one per line, with commit message and hash.
Scripts
There are some Git-related scripts available in the scijava-scripts project.
List information about all remote branches including last author, commit date and unmerged commit count
1 | $SCIJAVA /bin/remote-branch-info .sh |
Advanced and/or dangerous
Create a repository with g+w permissions
1 | git init --shared=group |
Or for a bare repository:
1 | git init --bare --shared=group |
(Bare repositories are meant for a remote server repository that all your coworkers push into and pull/fetch from.)
Push all remote branches from one remote (e.g., "origin") to another (e.g., "github")
1 2 3 4 5 6 | git push github $(git for -each-ref refs /remotes/origin | \ grep - v HEAD | \ while read sha1 type ref do echo $ref:refs /heads/ ${ref #refs/remotes/origin/} done ) |
Another way to push all remote branches between remotes
1 2 | eval git push github $(git for -each-ref | \ sed -n 's/.*\t\(refs\/remotes\/origin\/\(.*\)\)$/\1:refs\/heads\/\2/p' ) |
Fully garbage collect and compact the repository (deletes all orphaned refs!)
1 2 | git reset --hard git for -each-ref -- format = "%(refname)" refs /original/ | \ xargs -n 1 git update-ref -d git reflog expire --expire=now --all git gc --aggressive --prune=now |
Rewriting history
Split a subdirectory into a separate git repository
See these posts on Stack Overflow:
- Detach subdirectory into separate Git repository
- Detach subdirectory (that was renamed!) into a new repo
- Split large Git repository into many smaller ones
Throw away git-svn-id metadata
1 | git filter-branch --msg-filter ' sed -e "/^git-svn-id:/d" ' |
Combine the first two commits of a Git repository
See this post on Stack Overflow:
Change the author of a commit
1 | git commit --amend --author= "Author Name" |
Change the author of many commits
See this post on Stack Overflow:
Merge multiple repositories
See these posts on Stack Overflow:
Tutorials
1 2 3 4 5 6 7 8 9 10 11 | ssh you@server mkdir repos /remote .git cd repos /remote .git git --bare init --shared=group logout cd ~ /local git remote add origin ssh : //you @server /home/you/remote .git git push origin master git config branch.master.remote origin git config branch.master.merge refs /heads/master |
Creates a bare remote repository at ssh://server/home/you/remote.git
that tracks your local repository in /home/you/local
. Adopted from Tim Lucas.
Displaying a filtered set of commits
Assume you want to see commits in branch stephan
, but only those that are not part of the history of branch saalfeld
:
1 | git log stephan ^saalfeld |
More realistically, if you want to see all the commits which are in a topic branch, but not yet merged into master:
1 | git log --all ^master |
If you want to see the changes which come from a topic branch which was merged in commit deadbeef
, use this command line:
1 | git log deadbeef^..deadbeef^2 |
Explanation: deadbeef
is a merge commit, so its first parent (deadbeef^
, can also be written as deadbeef^1
) was the current HEAD
when the merge was performed, and the second parent (deadbeef^2
) is the tip of the branch which was merged. The argument A..B
is short form of ^A B
, i.e. all commits reachable from B
excluding those which are also reachable from A
.