Zach Olivare - 2019 May 03
This post is the first installment of what I hope to be a comprehensive explanation of the most effective git strategies and workflows
In this post, I hope to discuss what is in my opinion the most powerful feature of Git that no one ever uses: The Log. Understanding the log is essential to understanding Git as a whole. In many ways they are one in the same, yet, when's the last time you typed git log
? We're about to change that!
Many people think about branches in Git the wrong way. They think of branches being these heavy constructs that "contain" commits. This is understandable because language like commits being "on a branch" or your repo being "on a branch" is common, and that fits with prior understanding from other version control tools. But branches in Git are very simple. The only thing they do is point to commits, that's it! Every branch in Git is a pointer to a particular commit hash, and because they're just pointers, branches can easily be created, moved and deleted. You can switch
to a branch, but all that really means is that when you make a commit, the branch you have "checked out" will automatically be moved forward to your newly created commit.
It is possible to use Git (locally, anyway) without any branches at all, but that would be extremely inconvenient and I don't recommend it. Branches serve as convenient aliases for hard to remember and at times fluid commit hashes, but they also mark which commits are important. If a commit is somewhere in the ancestry of a commit that is pointed to by a branch, then Git considers it important enough to show it in the log. Any "hanging chads", if you will, commits that are not in the ancestry of any commit pointed to by a branch (or a tag), do not show up in the log. If that language is confusing to you, don't worry, keep reading!
Your HEAD (in Git) is like a weird kind of branch in the sense that it points to a particular commit. But unlike a branch, it can't be deleted. That's because your HEAD represents the files that currently exist in your file system, i.e. the files that you see in your IDE. When you have a branch checked out and you run git log
, you'll see that git shows your HEAD pointing to that branch: HEAD -> dev
. Like I mentioned before, that just means that when you create a new commit, both your HEAD and the branch that you have checked out (dev
in this example) will both move forward to that commit.
If the atom is the is the building block of the universe, then the commit is the building block of Git. Everything in Git is based around commits.
But what is a commit exactly? A Git commit is made up of three parts
Okay okay, enough of this talky talky nonsense. Go ahead, cd
to your most convenient Git repo and type git log
. You'll see something like the following:
Those, as you're probably aware, are your Git commits. For every commit, you're shown:
.gitconfig
when they made the commitYou can also see branches listed after the commit hash that point to this particular commit.
The commits are shown with the most recent at the top, and oldest at the bottom.
Now we've seen the log. But in its raw form, it's not super useful most of the time. We want to use the log to see the relationships between our commits to better understand the structure of our repository. Git can generate a graph which does exactly that by adding a --graph
flag to the git log
command.
The graph is nice, but the formatting of the information makes it difficult to extract the most useful information at a glance. Let's add a couple more flags to this command to modify the formatting.
We will add the --oneline
flag to make each commit display on only a single line instead of five or more. Then we'll add the --decorate
flag to add some nice colors to the graph and commit text. Finally we'll add the --all
flag. This flag shows the history of all of the branches in your repository, instead of only showing the history of the commit that you currently have checked out.
Great! This is a much more useful format of the log than what we started with. One final tweak though, I think it's useful to still display the author or each commit. In order to accomplish that, you have to specify a custom --pretty
format. This command becomes extremely arduous to type, and I would strongly recommend that you create an alias for it
This is one of many git aliases that I use. You can find the all of them here.
In this post, we've discussed some intermediate Git topics that are often misunderstood. Then we came up with a git alias to display a graph of the commits in your repository. This graph shows you the structure of your repository, and in future parts of this series, I will talk about what this graph should look like, and the techniques that are needed to structure commits in that fashion.