explaining git
ivo anjo
engineer-next-door @ talkdesk
...or (probably) (hopefully) (maybe)
the last presentation on git
you'll ever need!
explaining git
...or (probably) (hopefully) (maybe)
the last presentation on git
you'll ever need!
follow along @
http://tinyurl.com/eye-doctor-here-i-go
You only need to understand one thing to get what git is all about
repository ==
bunch of Directed Acyclic Graphs
of commits
Let's leave that in a background thread for a few slides
What's in a
git commit?
• repository state hash
• parent commit hashes
• commit message
• committer
• commit date
• author
• authoring date
• (extra metadata)
What's a
git commit hash?
sha1( )
• repository state hash
• parent commit hashes
• commit message
• committer
• commit date
• author
• authoring date
• (extra metadata)
git commits are
*immutable*
You think you're editing commits?
You're not
So I promised we would go back to the DAG thingy
git operations are:
• operations on the DAG
• operations that change outside
pointers to DAG nodes
What's a
branch?
Behold the magic .git/refs folder!
~/some-git-repo/.git/refs$ tree
.
├── bisect
├── heads
│ ├── master
│ └── feature-branch
├── remotes
│ ├── origin
│ │ ├── master
│ │ └── feature-branch
│ ├── production
│ │ └── master
│ ├── qa
│ │ └── master
│ └── staging
│ └── master
├── stash
└── tags
~/some-git-repo/.git/refs$ tree
.
├── bisect
├── heads
│ ├── master -> ccc
│ └── feature-branch -> ddd
├── remotes
│ ├── origin
│ │ ├── master -> bbb
│ │ └── feature-branch -> ddd
│ ├── production
│ │ └── master -> aaa
│ ├── qa
│ │ └── master -> aaa
│ └── staging
│ └── master -> aaa
├── stash
└── tags
Just a file containing the commit hash of the tip of the branch
What's "committing to a branch"?
• create new
commit object
• hash it
• update branch file
Before
.git/refs/heads/master -> bbb
After
.git/refs/heads/master -> ccc
What's "amending a commit"?
A lie
You don't change the existing commit.
You create a new one.
• and then change the ref
file to point to from ccc to ddd
What's editing multiple commits (rebase, etc)?
rebase:
iterate over commit range & try to re-apply changes on top of new repo state
What's a
(*non fast forward)
merge?
Commit with several parents!
What's a fast-forward merge?
just changes the ref file to point to the new commit
What's a tag?
Just another ref
$ git tag bestest-release-ever master
$ ls .git/refs/tags/
bestest-release-ever
$ cat .git/refs/tags/bestest-release-ever
c50f9ddda562e2777a68976ad7afbf83a72982f8
What's a
git stash push?
Just a commit that gets listed on .git/logs/refs
.git/logs?
Whazzat?
Do you remember I said nothing gets changed?
git keeps a log of all operations!
git reflog
~/ruby/ivo-pry-debugger$ git reflog
11b1e1b HEAD@{0}: commit: Fix missing link in README
0542fa3 HEAD@{1}: rebase -i (finish): returning to refs/heads/master
0542fa3 HEAD@{2}: rebase -i (pick): Bump major version due to breaking changes (ruby compatibility)
537b2fb HEAD@{3}: rebase -i (pick): Drop support for older ruby versions (< 2.2.0)
c817891 HEAD@{4}: rebase -i (pick): Modernize codebase
03677c0 HEAD@{5}: rebase -i (pick): Add missing banners to avoid warnings in Pry code
aa67887 HEAD@{6}: rebase -i (pick): Avoid circular require when being loaded by pry
123da12 HEAD@{7}: rebase -i (fixup): README cleanups and updates
a51fbe7 HEAD@{8}: rebase -i (start): checkout origin/master
ed2e349 HEAD@{9}: rebase -i (finish): returning to refs/heads/master
ed2e349 HEAD@{10}: rebase -i (start): checkout master
ed2e349 HEAD@{11}: commit: fixup
4cf6d29 HEAD@{12}: commit: Bump major version due to breaking changes (ruby compatibility)
f741c04 HEAD@{13}: commit: Drop support for older ruby versions (< 2.2.0)
You never lose anything
(*on your local machine)
If you're low on space, you can ask git to GC and recompress commits
(e.g. remove all commits not referenced in .git/refs or reachable via them)
Back to our regularly scheduled program
What's a push?
• upload commits
• change remote refs file
to point to your latest
commit
• execute hooks
That's it
Nothing more
What's a force push?
• upload commits
• change remote refs file
to point to your latest
commit
• execute hooks
that is not a child of the previous ref
git push
git push --force
It's **dangerous** because if someone else has pushed, you remove their changes from history
Don't use it
If you really need to
--force-with-lease
Basically only lets you push force if you previously fetched the branch
1. push
2. p --force-with-lease
1. push
2. your awesome colleague pushes
3. push --force-with-lease
It doesn't solve all problems
If you just run git fetch first then...
You've just erased your colleague's work from the repository history
What's
git log?
Just follow the commits from the tip of the branch and print their messages
And finally... Some useful things you may not have heard about
git diff --word-diff
git bisect
• This was working last deploy
• It's not working now
git bisect
• git bisect does a binary search of git commits, and each time you tell it yay or nay
• the final output is the commit which
broke it
• can also run fully automated (e.g. run
rspec, watch result)