Git Command Reference

This page is a non-exhaustive and growing reference of popular git commands and tips. Note: Do not include the < > brackets around the actual branch names, etc., they are used in the examples below to indicate a place holder.

Add

# add / removes all
git add .

# use `-n` to do a dry run
git add -n .

Amend

# combine any staged changes with the contents of the previous commit
git commit --amend

# same as above and change the commit message
git commit --amend -am "my new commit message"

Branch

# list local branches
git branch

# list remote and local branches
git branch -a

# list remote branches
git branch -r

# rename current branch
git branch -m <newname>

# rename a local branch
git branch -m <oldname> <newname>

# push a new local branch to remote and track
git push -u origin <branchname>

Clone

Cloning is the first step when getting a repository from the server.

# download the repository into the current directory
git clone <url>

# download the repository into the current directory with a sub directory name you supply
git clone <url> <directory>

Checkout

# checkout and create new branch
git checkout -b <branchname>

# the new branch head will point to <startpoint>
# <startpoint> can be a branchname, remote-alias/branchname, commithash or tag
git checkout -b <branchname> <startpoint>

# copy file from another branch
git checkout <branchname> -- <filename>

# checkout a tag to a new branch
git fetch origin
git checkout tags/<tagname> -b <branchname>

# restore file deleted locally that still exist on remote
git checkout -- <filename>

Cherry Pick

# merge a specific commit into the current branch
git cherry-pick <commithash>

Config

# set email for repo
cd path/to/repo
git config user.email "gilfoyle@piedpiper.com"

# set username for repo
cd path/to/repo
git config user.name "gilfoyle"

# confirm
git config user.email
git config user.name

# git config --global core.excludesfile [file]
git config --global core.excludesfile somefile

In Git version 2.16+, git command output is sent to the editor. To change this, set pager.branch false. more info

# set paged output for git branch to off
git config --global pager.branch false

Delete

# delete local branch
git branch -d <branchname>

# delete remote branch
git push origin :<branchname>

# Git 1.7+
git push origin --delete <branchname>

Diff

# diff of what is changed but not staged
git diff

# diff of what is staged but not yet commited
git diff --staged

# diff between two local branches
git diff branch_a..branch_b

# write the diff to a file
git diff branch_a..branch_b > C:\diff.txt

# common ancestor diff
git diff branch_a...branch_b

Notes

  • Press q to exit diff
  • You can also add a file or folder name after the commands

Fetch

# checkout a remote branch that does not exist on local
git fetch origin
git checkout --track origin/<remote branch name>

# checkout a remote branch to a new local branch
git fetch origin <remote branch name>:<local branch name>

Init

# initialize an existing directory as a Git repository
git init

Log

# commit logs for the active branch.
git log

# list commits on branchA that are not on branchB
git log branchB...branchA

# list commits that changed a file
git log  --follow somefile

# not in origin
git log origin..HEAD

# one line
git log --pretty=oneline

# last commit only with truncated hash
git rev-parse --short HEAD

# find in commit messages
git log --oneline | grep sometext

# list commit logs with indication of any paths that moved
git log --stat -M

# show any object in Git in human-readable format
git show <SHA>

Notes

  • git log outputs with the less program, making it scrollable. Type q to exit. Type h for help.
  • origin..HEAD specifies all the commits reachable from the current commit (i.e. HEAD), but not from origin.

Pull

# fetch and merge any commits from the tracking remote branch
git pull

# get rid of 'your branch is ahead of origin <branchname> by X commits
git pull origin <branchname>

Rebase

# reapply commits on top of another base tip
git rebase

# squash four commits into one
git rebase -i HEAD~4

Here’s an easy way to non-interactively merge the last X commits into the first. In this example, we’re merging the last 3 commits into commit 1.

# note that this is HEAD-2, not HEAD-3
git reset --soft HEAD~2

# this automatically uses `commit 1` message
git commit --amend -C HEAD

Notes

  • Squash info: Git Book Chapter 6.4: Git Tools - Rewriting History. The editor will list the 4 commits starting with the oldest. Replace the word “pick” with “squash” next to each commit you want to squash into the commit before it.
  • Editor: press i to enter inline insert mode. Use the delete key to erase from left to right, press esc to exit insert mode, then type :x (now the cursor is at the bottom) and hit enter to save and exit.
  • You can edit your last commit message before pushing with git commit --amend

If you’ve already pushed commits to GitHub, and then squash them locally after, you will have to force the push to your branch.

git push origin <branchname> --force

Reset

# can be used to undo previous uncommitted add(s)
git reset

# unstage a file while retaining the changes in working directory
git reset somefile

# reset to previous state
git reset --hard

# reset to previous commit
git reset --hard <commit hash>

# reset to origin
git reset --hard origin/<branchname>

git reflog to show what has been done with git session

Rm

# remove unstaged changes, e.g., added file or files
# note `-r` is needed to recurse
git rm -r --cached .

git rm --cached somefile

Stash

Temporarily store modified, tracked files in order to change branches

# stash changes in a dirty working directory
git-stash

# remove and apply stash
git-stash pop

# list stashes
git stash list

# apply stash
git-stash apply

# remove a stash
git stash drop <stashid>

Tag

# add a lightweight tag to by specifying the
# commit checksum (or part of it) at the end of the command
git tag sometagname ed40a9deaa0
git push origin sometagname

# delete a tag
git tag -d sometagname
git push origin :refs/tags/sometagname

If you have a local branch with the same name as the tag, you will not be able to push it to origin: error: src refspec sometagname matches more than one

Working with Forks

Forks are a concept created by GitHub. On a standard git server, the equivalent would be like cloning a repo to a git server and then cloning it from that server to your local computer. Then add the original repo as the upstream from the local computer.

Scenario, you attempt to fetch upstream and get fatal: 'upstream' does not appear to be a git repository. e.g.,

git fetch upstream
fatal: 'upstream' does not appear to be a git repository
fatal: Could not read from remote repository.

Please make sure you have the correct access rights
and the repository exists.

You will need to configure a remote that points to the upstream repository. for example, add a remote with the alias, upstream to your repo:

git remote add upstream https://github.com/wodby/docker4wordpress.git

# verify
git remote -v

Syncing a fork

Fetch the branches from the upstream repository.

git fetch upstream

Merge upstream branches using git merge upstream/<branchname>. e.g., merge master branch into your local fork master:

git checkout master

git merge upstream/master

Creating a Fork From GitHub to GitLab or Bitbucket

If you just want to track changes, first make an empty repository in GitLab, Bitbucket or whatever else you may be using and clone it to your computer.

Then add the GitHub project by declaring an alias named upstream using remote add, e.g.,

git remote add upstream https://github.com/user/repo

This will allow you to fetch and pull from the upstream repo. e.g.,

# fetch
git fetch upstream master

# pull
git pull upstream master

When there are any changes, you can fetch and pull them from the respective branch. Given your permissions on the upstream repo, you may also be allowed to push or merge.

To get the upstream branch you pulled onto your GitLab, Bitbucket, etc. server, and then merge that into your master branch and push to origin. Use this example,

# switch to origin master
git checkout master

# merge the upstream alias master into your master
git merge upstream/master

# push to your origin repo master branch
git push

GitLab also has repository mirroring to allow the mirroring of repositories to and from external sources. It can be used to mirror branches, tags, and commits between repositories.

Tips & Tricks

Share & Update

Retrieving updates from another repo and updating local repo

# clean up deleted remote git branches
git remote prune origin

# add a git URL as an alias
git remote add <alias> <url>

# fetch down all the branches from that Git remote
git fetch <alias>

# merge a remote branch into your current branch to bring it up to date
git merge <alias>/<branch>

# transmit local branch commits to the remote repository branch
git push <alias> <branch>

Discard changes / commits

# discard local changes for all unstaged file
git checkout -- .

# discard local commits
git reset --hard origin/<branchname>

Merge Repositories

Merge repo_a into repo_b

cd path/to/repo_b

git remote add repo_a path/to/repo_a

git fetch repo_a

git merge repo_a/master

git remote remove repo_a

Merge a branch from another repository into a branch in a different local repository. For two repositories, foo and bar located in the current directory, this example merges master from bar into a new branch in foo.

cd foo

# add the bar repo to foo
git remote add bar ../bar

# fetch --all from remote
git remote update

# create a new branch in foo
git checkout -b barmaster

# merge master from bar into the new branch
git merge bar/master

since git 2.9 you can use the option --allow-unrelated-histories with git merge

Change Remote Origin

When the remote origin repository has been renamed, the URL needs to be reset.

List existing remotes to get the address format, ssh or https of the remote to change.

Example

git remote -v
origin	git@bitbucket.org:gilfoyle/gilfoylehacks.git (fetch)
origin	git@bitbucket.org:gilfoyle/gilfoylehacks.git (push)

Use git remote set-url to change the origin url. e.g.,

git remote set-url origin git@bitbucket.org:gilfoyle/gilfoylehacks2.git

Another example - suppose you cloned a repo on GitHub using https and you’re ready to push changes up to the remote origin. You might receive this message output after entering your username and password:

remote: Support for password authentication was removed on August 13, 2021. Please use a personal access token instead.
remote: Please see https://github.blog/2020-12-15-token-authentication-requirements-for-git-operations/ for more information.

In addition to updating your configuration to connect to GitHub with SSH, you will need to change the remote from HTTPS to SSH, For example,

git remote set-url origin git@github.com:gilfoyle/gilfoylehacks.git

Git Basics - Working with Remotes

Clone into an existing folder?

This can’t be done, however, workaround by cloning to a new directory, then moving the .git directory into the existing directory.

git clone https://gitlab.com/myrepo.git temp

# move .git from temp to myrepo directory
mv temp/.git myrepo/.git

# delete temp directory
rm -rf temp

Remove Commits after Specified Commit

If for some reason you need to remove recent commits, you can force a push to reset the origin to a particular commit hash. This will remove any commits that were pushed after it that commit hash.

git push origin +<commithash>:<branchname>

# for example
git push origin +ed40a9deaa0:master

The + is interpreted as forced push.

Another way to do the same thing is with git reset. For example,

# reset to previous commit in current branch
git reset --hard <commit hash>

# force push
git push origin -f

SSH Keys

How to create ssh keys for multiple git accounts on a the same host. Generate a ssh key pair for each account using the respective e-mail address associated with it.

ssh-keygen -t rsa -b 4096 -C "username@example.com"

When you’re prompted to “Enter a file in which to save the key,” Edit the filename so the key pair can be mapped to the account. e.g., id_rsa_gitlab

Enter a file in which to save the key (/Users/you/.ssh/id_rsa_gitlab): [Press enter]
ssh-agent
# start the ssh-agent
eval $(ssh-agent -s)

# add the key
ssh-add ~/.ssh/id_rsa_gitlab

More info: Generating a new SSH key and adding it to the ssh-agent

Create a Config file

touch ~/.ssh/config

Edit the config file to map the ssh key to its unique Host value. For example, github.com for the personal account and github.com-work for the work account. e.g.,

.ssh/config
# github personal
Host github.com
    HostName github.com
    User richard.hendricks
    IdentityFile ~/.ssh/id_rsa_github

# github work account
Host github.com-work
    HostName github.com
    User richard.hendricks@piedpiper.com
    IdentityFile ~/.ssh/id_rsa_github_piedpiper

Host gitlab.com
    HostName gitlab.com
    User richard.hendricks
    IdentityFile ~/.ssh/id_rsa_gitlab

Clone repos using the unique host value. e.g.,

git clone git@github.com-work:githubaccount/somerepo.git

This will set the repo’s .git/config remote origin as needed. e.g.,

[remote "origin"]
    url = git@github.com-work:githubaccount/somerepo.git

Additional Resources

comments powered by Disqus