# Git ## Glossary **GIT**: an open source, distributed version-control system **GITHUB**: a platform for hosting and collaborating on Git repositories **TREE**: directory that maps names to blobs or trees **BLOB**: any file **COMMIT**: snapshot of the entire repository + metadata, identified it's SHA-1 hash **HEAD**: represents the current working directory, the HEAD pointer can be moved to different branches, tags, or commits when using git checkout **BRANCH**: a lightweight movable pointer to a commit **CLONE**: a local version of a repository, including all commits and branches **REMOTE**: a common repository on GitHub that all team member use to exchange their changes **FORK**: a copy of a repository on GitHub owned by a different user **PULL REQUEST**: a place to compare and discuss the differences introduced on a branch with reviews, comments, integrated tests, and more **REPOSITORY**: collection of files and folder of a project aka repo **STAGING AREA/STASH**: area of temporary snapshots (not yet commits) ## Data Model Data Model Structure: ```txt (tree) | |_ foo (tree) | |_ bar.txt (blob, contents = "hello world") | |_ baz.txt (blob, contents = "git is wonderful") ``` Data Model as pseudocode: ```py # a file is a bunch of bytes blob = array # a directory contains named files and directories tree = map # a commit has parents, metadata, and the top-level tree commit = struct { parent: array author: string message: string snapshot: tree } # an object is either a blob, tree or commit object = map # commit identified by it's hash (unmutable) def store(object): id = sha1(object) # hash repo objects = object # store repo w/ index hash # load the commit def load(id): return objects # human-readable names for SHA-1 hashes (mutable) references = map # bind a reference to a hash def update_reference(name, id): references = id def read_reference(name): return references def load_reference(name_or_id): if name_or_id in references: return load(references) else: return load(name_or_id) ``` ## Commands `git help `: get help for a git command ### Create Repository `git init []`: initialize a brand new Git repository and begins tracking `.gitignore`: specify intentionally untracked files to ignore ### Config `git config --global user.name ""`: set name attached to commits `git config --global user.email ""`: set email attached to commits `git config --global color.ui auto`: enable colorization of command line output ### Making Changes `git status`: shows the status of changes as untracked, modified, or staged `git add `: add files to the staging area `git add -p `: interactively stage chunks of a file `git blame `: show who last edited which line `git commit`: save the snapshot to the project history `git commit -m "message"`: commit and provide a message `git commit -a`: automatically notice any modified (but not new) files and commit `git commit -v|--verbose`: show unified diff between the HEAD commit and what would be committed `git diff `: show difference since the last commit `git diff `: show differences in a file since a particular snapshot `git diff `: show differences in a file between two snapshots `git diff --cached`: show what is about to be committed `git diff ...`: show content diff between two branches `git bisect`: binary search history (e.g. for regressions) ### Stashes `git stash [push -m|--message]`: add all changes to the stash (and provide message) `git stash list` list all stashes `git stash show []`: show changes in the stash `git stash pop`: restore last stash `git stash drop []`: remove a stash from the list `git stash clear`: remove all stashes ### Remotes `git remote`: list remotes `git remote -v`: list remotes names and URLs `git remote show `: inspect the remote `git remote add `: add a remote `git branch --set-upstream-to=/`: set up correspondence between local and remote branch `git push `: send objects to remote `git push :`: send objects to remote, and update remote reference `git fetch []`: retrieve objects/references from a remote `git pull`: update the local branch with updates from its remote counterpart, same as `git fetch; git merge` `git pull -ff`: when possible resolve the merge as a fast-forward (only update branch pointer, don't create merge commit). Otherwise create a merge commit. `git fetch && git show /`: show incoming changes `git clone []`: download repository and repo history from remote `git clone --shallow`: clone only repo files, not history of commits `git remote remove `: remove the specified remote `git remote rename `: rename a remote ### Viewing Project History `git log`: show history of changes `git log -p`: show history of changes and complete differences `git log --stat --summary`: show overview of the change `git log --follow `: list version history fo file, including renames `git log --all --graph --decorate`: visualizes history as a DAG `git log --oneline`: compact log `git shortlog`: list commits by author `git show `: output metadata and content changes of commit `git cat-file -p `: output commit metadata ### Tag Git supports two types of tags: *lightweight* and *annotated*. A lightweight tag is very much like a branch that doesn't change—it's just a pointer to a specific commit. Annotated tags, however, are stored as full objects in the Git database. They're checksummed;contain the tagger name, email, and date; have a tagging message; and can be signed and verified with GNU Privacy Guard (GPG). It's generally recommended creating annotated tags so it's possible to have all this information. `git tag`: list existing tags `git tag -l|--list `: list existing tags matching a wildcard or pattern `git tag []`: create a *lightweight* tag on the commit `git tag -a [ -m ]`: create am *annotated* tag on the commit `git push `: push a tag to the remote `git push --tags`: push commits and their tags (both types) to the remote `git tag -d `: delete a tag `git push :refs/tags:`: remove a tag from the remote `git push --delete `: remove a tag from the remote `git checkout `: checkout a tag - **WARNING**: will go into *detached HEAD* ### Branching And Merging `git branch`: shows branches `git branch -v`: show branch + last commit `git branch `: create new branch `git checkout -b `: create a branch and switches to it, same as `git branch ; git checkout ` `git branch`: show list of all existing branches (* indicates current) `git checkout `: change current branch (update HEAD) and update working directory `git branch -d `: delete specified branch `git branch -m `: rename a branch without affecting the branch's history `git merge `: merges into current branch `git merge --continue`: continue previous merge after solving a merge conflict `git mergetool`: use a fancy tool to help resolve merge conflicts `git rebase`: rebase set of patches onto a new base `git rebase -i`: interactive rebasing `git cherry-pick `: bring in a commit from another branch `git cherry-pick ^..`: bring in a range of commits from another branch (first included) `git cherry-pick ..`: bring in a range of commits from another branch (first excluded) ### Undo & [Rewriting History](https://www.themoderncoder.com/rewriting-git-history/) `git commit --amend`: replace last commit by creating a new one (can add files or rewrite commit message) `git commit --amend -m "amended message"`: replace last commit by creating a new one (can add files or rewrite commit message) `git commit --amend --no-edit`: replace last commit by creating a new one (can add files or rewrite commit message) `git reset HEAD `: unstage a file `git reset `: undo all commits after specified commit, preserving changes locally `git checkout `: discard changes `git checkout -- `: discard changes, no output to screen `git reset --soft `: revert to specific commit but keep changes and staged files `git reset --hard `: discard all history and changes back to specified commit `git rebase -i HEAD~`: modify (reword, edit, drop, squash, merge, ...) *n* commits `git rm --cached `: remove a file from being tracked **WARNING**: Changing history can have nasty side effects --- ## How To ### Rebase Branches ```ps1 git checkout git pull # get up to date git checkout git rebase # rebase commits on master (moves branch start point on last master commit) git checkout git rebase # moves commits from the branch on top of master ``` ![branch](../img/git_branches.png "how branches work") ### Clone Branches ```ps1 git clone # clone the repo git branch -r # show remote branches git checkout # checkout remote branch (omit /) git pull # clone branch ``` ### [Sync Forks](https://docs.github.com/en/free-pro-team@latest/github/collaborating-with-issues-and-pull-requests/syncing-a-fork) ```ps1 git fetch upstream # Fetch the branches and their respective commits from the upstream repository git checkout main # checkout fork's main primary branch git merge upstream/main # Merge the changes from the upstream default branch into the local default branch git push # update fork on GitHub ```