Luke M
My git configuration
2025-02-26
Some people prefer to adhere to configuration defaults. I agree with this to some extent; I generally prefer to use the original command rather than create an alias for it. That said, there are some tweaks that can make your Git experience a lot better.
I've compiled a list of my favourite git configuration options, and their explanations. Feel free to give them a try, or don't... Up to you!
Gitconfig
Basics
Pull should be in rebase mode by default.
git config --global pull.rebase true
If you push, and the remote branch doesn't exist, it should just create it.
git config --global push.autoSetupRemote true
Set default editor
I can't use Nano. Vi doesn't render text for me correctly through Tmux, and I never bothered to install Vim. So NeoVim it is.
git config --global core.editor "nvim"
Set the default branch as main
This ensures that main
is the default branch when you git init
a new
repository locally, because most Git remotes use main
as the default.
git config --global init.defaultBranch main
Show change diff in commit message editor
This is handy. When you commit, it will show a diff of changes. Scrolling through is an excellent reminder of what I need to list out in the body of the commit.
git config --global commit.verbose true
Use YYYY-MM-DD dates
ISO 8601 is the most common date format software developers are familar with. I have no idea why Git by default uses another format.
git config --global log.date iso
Stash changes before a rebase, then apply them back after
This change essentially lets you rebase from a dirty working directory. Very
handy because in addition to saving you time, ensures you won't forget to git stash apply
.
git config --global rebase.autoStash true
Use Git SSH URLs instead of HTTPS URLs
Sometimes when copying a Git repository URL from GitHub, it defaults to the HTTPS URL, which does not work with my SSH keys.
git config --global url."git@github.com:".insteadOf "https://github.com/"
Automatically clean up deleted remote branches
This deletes branches locally that have been deleted on the remote. Nobody should be forced to live with everyone's rubbish!
git config --global fetch.prune true
Fix mistyped commands
Because Git knows what you meant.
git config --global help.autocorrect prompt
Auto-squash fixup commits
A fixup
commit, is a way to mark a commit as a fixup of a previous commit,
upon making it. While you could just amend a commit, I find fixup
commits
helpful in case I need to reset
to a previous one.
git config --global rebase.autoSquash true
# Mark a certain commit as a fixup commit
git commit --fixup c8ee3f5
Git aliases
git config --global alias.o "log --oneline"
git config --global alias.adog "log --all --decorate --oneline --graph"
LazyGit fix
Git tags do not work in LazyGit if GPG signing for them is enabled
git config --global tag.gpgsign false
Git commit message
You can add a custom git commit message to keep your messages on track.
git config --global commit.template ~/.gitmessage
I added a 50-character measuring tape, template and rough guide for conventional commits, and URLs to further reading on Git messages and semantic versioning.
# -----------------------------------------------| < 50 chars length
# <type>[optional scope]: <description>
#
# [optional body]
#
# [optional footer(s)]
This is very helpful for keeping my commit messages consistent and helpful, and ensuring I don't have to double back and re-edit them later.
You can see my full .gitmessage file here. Feel free to steal it, I won't mind!
Git hooks
You can globally specify a directory for git hooks, so that you can add behaviour to git.
git config --global core.hooksPath ~/.git-hooks
Guard main branch
I like to protect against pushes to shared branches, there usually isn't a very good reason for me to be doing this.
#!/bin/bash
# Filename: ~/.git-hooks/pre-push
BRANCH=`git rev-parse --abbrev-ref HEAD`
if [[ "$BRANCH" =~ ^(master|main|staging|development|feature|testing)$ ]]; then
echo
echo "Prevented push to $BRANCH."
echo "If you really want to do this, use --no-verify to bypass this pre-push hook."
echo
exit 1
fi
exit 0
Git guard script
I've written a script that I add to my .zshrc
that intercepts the execution of
git
and checks whether I'm using a command I'm trying to unlearn, then prompts
me to use the preferred command.
You can see the full script here.
Here is an example of how it works:
luke@macbook ~/github.com/lkdm/blog (main) $ git checkout
Don't use checkout; use switch or restore.
Consider using 'git switch' instead.
Using 'git switch' is safer and more predictable for switching branches, while 'git restore' is better for discarding changes.
Do you still want to proceed? (y/N)
Here are some of the behaviours I'm trying to unlearn:
git push origin --force
This one has bitten me. --force
will just overwrite whatever is on remote with
what is on my branch. Instead, it's better to use --force-with-lease
which
first checks whether the remote has changed.
git checkout
A, in my opinion, better alternative to git checkout is to use git switch
to
deal with commits and branches, and git restore
to deal with files.
The commit that introduced the change specifies the rationale:
"git checkout" doing too many things is a source of confusion for many users (and it even bites old timers sometimes). To remedy that, the command will be split into two new ones: switch and restore. The good old "git checkout" command is still here and will be until all (or most of users) are sick of it.
While the stated goal of the change is to reduce confusion, the interface for
dealing with git switch
is much better:
--detach
is always required when switching to a detached head, whereas it was previously optional.- When using
checkout
with--force
, you can switch branches while in the middle of a merge. You can't do this withswitch
.
You are less likely to encounter a footgun while using git stash
.
git commit -m
Using the -v
flag is much better, because it opens the
commit editor in my chosen editor (nvim), shows a diff of all changes, and also
shows my commit template message. Since using -v
, I never forget the correct
commit message convention, or forget to list any of the changes I've made.