Get started with Git CLI and use these workflows in your everyday projects

radualexandrub

Radu-Alexandru B

Posted on October 5, 2020

Get started with Git CLI and use these workflows in your everyday projects

I know, I know, there are a lot of mini-tutorials, cheatsheets, and howto's about Git, but some of those are really crazy! I can't see how a beginner that gives Git a try would start learning first about cherry-pick and commit amends or whatever. So, presuming that you know Git as a version control system (if not, just watch this 12min intro from Telusko), but you never started using CLI commands, I'll write here some notes that got me going into using Git in all my personal projects!

Here's the contents:


git help

Display help (all commands) in CLI:

git help
Enter fullscreen mode Exit fullscreen mode

Check Git version installed on PC:

git --version
Enter fullscreen mode Exit fullscreen mode

Git help used on commands will display the help info in a new Browser tab (Chrome/Firefox/Edge etc). This will be really handy to understand all those parameters used in various commands.

git help <command> # is the same as git <command> --help

# eg:
git help config
git config --help # is the same as git help config
git add --help
Enter fullscreen mode Exit fullscreen mode

Set up global configuration variables

  • Needed to push local repository to GitHub remote server
  • Also useful when working in a team to see who changed the code (with the blame command aka 'see who and when was this file last modified')
  • For name: you can choose any name, it can be different from your github account
  • For email: it must be the exact same email used on your github account
git config --global user.name "John Doe" 
git config --global user.email "JohnDoe97@gmail.com" 

git config --list # list all configurations you made
Enter fullscreen mode Exit fullscreen mode

Create own local empty repository init VS. clone an existing repository as a local repository in your PC

  • Create own local empty repo (init) (it will create a new folder .git with all the informations about that repo):
cd my_new_project_folder_name
git init
Enter fullscreen mode Exit fullscreen mode
  • Copy/Download an existing repo (clone) (this will also copy all of the earlier commits/history made to that repository)
git clone <url> <optional:where_to_clone>
# eg:
git clone https://github/username/android-app AndroidAppFolder
Enter fullscreen mode Exit fullscreen mode

git status

git status
Enter fullscreen mode Exit fullscreen mode

Git keeps track of modified/added/removed files and also which files are [are not] tracked.
However, sometimes you don't want to track some files (eg. personal files, personal configurations files, cache files auto-generated after each build) => create .gitignore file where you can write the files you don't want to include (ignore them forever):

mystuff.txt
*.pyc
.DS_Store
Enter fullscreen mode Exit fullscreen mode

^^ These files won't show up anymore when calling git status.
Also, .gitignore should be included (git add .gitignore) to prevent a team collaborating on a project from committing generated cache files => don't add .gitignore to .gitignore itself lol.

git diff

Git diff shows the changes made to the code within modified files (git status shows only which files have been modified/created).

git diff
Enter fullscreen mode Exit fullscreen mode

Git work-flow (on master branch)

  • Add files to the staging area (= add all the files that are ready to be commited except the files from .gitignore)
git add -A 
Enter fullscreen mode Exit fullscreen mode

  • Commit all the files added (tracked files) to the local repository.
git commit -m "Message from this commit"
Enter fullscreen mode Exit fullscreen mode
  • Add all the files to the staging area then commit:
git commit -a -m "Message from this commit"
Enter fullscreen mode Exit fullscreen mode
  • Git reset [file.ext] will make all the files [or only file.ext] untracked (out of the staging area) => The changes to that file.ext will not be committed!
git reset [file.ext]
Enter fullscreen mode Exit fullscreen mode

  • Git log shows all the made commits (with a hash number, author, date of each commit). By default, the log opens in Vim text editor, if installed.
git log
git log -3 # shows last 3 commits
Enter fullscreen mode Exit fullscreen mode


git push -u origin master
# git push -u <remote> <branch>
# -u or --set-upstream is to save/add upstream (tracking) reference(the <remote> and <branch>), in order to just write "git push" without specifying again the <remote> and <branch> 
Enter fullscreen mode Exit fullscreen mode

  • Git remote shows all the remotes (GitHub repositories) where you can push the last (local) commit. Git remote -v is for verbose (all info about the remotes).
git remote
git remote -v
Enter fullscreen mode Exit fullscreen mode

To add a remote:

git remote add origin https://github.com/username/projectname/.git
# git remote add <new_branch_name> <link_to_repository>
Enter fullscreen mode Exit fullscreen mode

OBS: When cloning a remote (a GitHub repository) with git clone, a remote named "origin" will be available by default.


Get (locally) the last state (last changes/updates) of the project if someone made changes on the global repository (the remote to github.com server):

git pull origin master
Enter fullscreen mode Exit fullscreen mode

If this error occurs when pulling: "your local changes to the following files would be overwritten by merge" and you want to drop/overwrite all the changes made from the local repository and get the latest updates from the global repository, use:

# Drop local changes, revert to the HEAD reference (last local commit in the master branch)
git reset --hard # NEVER USE: git checkout HEAD^ file/to/overwrite
git pull origin master
# HEAD^ is short for HEAD^1, which means the one commit before HEAD. You could also do HEAD^2 for the commit before that one
Enter fullscreen mode Exit fullscreen mode

Or, discard/give up all the changes (modified files) and go back to the last (local) commit state of files:

git checkout -- . 
Enter fullscreen mode Exit fullscreen mode

NOTE1: git checkout HEAD^ filename (will overwrite the file to the state of commit before last local commit in the master branch)

NOTE2: git checkout HEAD^ without specifying a file will DROP THE WHOLE LAST COMMIT!!! and will revert to the commit before last commit (however you will be in a detached head state branch, so you can revert this action by just changing back to the master branch: git checkout master)

NOTE3: git checkout HEAD^1 is roughly the same as git reset HEAD^1, but:

  • Use reset if you want to undo the staging of a modified file !!!
  • Use checkout if you want to discard changes to unstaged file/s !!! (however it is still possible to recover lost files with git reflog and cherry-pick, check locally mistakes section.

Schema: Working Directory, Staging Area, Git Remote Repository


Create a new branch from CLI

  • git branch, git checkout To create and move to a new branch:
git branch <new_branch_name>
git checkout <new_branch_name>
Enter fullscreen mode Exit fullscreen mode

Show all branches/active branch with:

git branch
git branch -v
Enter fullscreen mode Exit fullscreen mode
  • Merge a branch: git merge (mini-workflow) If all your modifications to the code are great and pass all the (unit) tests, merge your branch with the master branch:
git checkout master # change to master branch
git pull origin master # get last updates before making any changes to master
git branch --merged # show branches that are/aren't merged with master branch
git merge <my_new_branch_ive_worked_on>
git push origin master
Enter fullscreen mode Exit fullscreen mode
  • Delete a branch (mini-workflow) After you added the features from your branch and merged with master, you can delete your branch you worked on:
git branch --merged
git branch -d <my_branch_ive_worked_on> # locally delete the branch
git branch -a # show all branches: we still have <my_branch> globally
git push origin --delete <my_branch_ive_worked_on> # globally/definitely delete the branch
Enter fullscreen mode Exit fullscreen mode

Git Complete Workflow - Work from another branch

git config --global user.name
git config --global user.email
git clone <url> <where_to_clone>
git branch <my_new_branch_name>
git checkout my_new_branch
# (make changes to the code ...)
git status
git commit -m "Add @function in views.py | Solve bug in models.py that fixes #8"
git push -u origin my_new_branch
# (wait for unit tests to complete)
# (if all unit tests pass, then do these)
git checkout master
git pull origin master
git merge my_new_branch
git push origin master
# (now time to delete my_new_branch)
git branch -d my_new_branch
git branch -a
git push origin --delete my_new_branch
Enter fullscreen mode Exit fullscreen mode

Create a new repo from a locally existing/completed project (mini-workflow)

  • On GitHub.com website:
    • Create a new repository (write name & description)
    • (optional) Create a Readme.MD file
  • On CLI (locally):
    • Open the terminal in that folder/project's path
    • Write the following commands:
git init
git remote add origin https://github.com/username/projectname/.git
git remote -v
git pull origin master # needed to update the commit history of new repo (especially if Readme.MD or LICENSE was created)
git status
git add -A
git status
git commit -m "Initial commit from local project"
git push origin master
Enter fullscreen mode Exit fullscreen mode

Locally Mistakes that could've been made

If we made changes to a single_file but then we don't want to keep the changes to that file anymore (we want to undo/go back):

git checkout single_file.py
Enter fullscreen mode Exit fullscreen mode

And if we want to discard all changes/modifications to our files:

git checkout -- .
Enter fullscreen mode Exit fullscreen mode

Also, if we want to delete/get rid of untracked files (newly created files):

git clean -fd # force directories
Enter fullscreen mode Exit fullscreen mode

We mess up a commit -m message. We want to modify the last commit message without doing another commit

WARNING: The following commands in this section will change the hash of previous commits => THIS WILL CHANGE GIT HISTORY => IF OTHER PEOPLE WILL PULL THE CHANGES AFTER EXECUTING THESE COMMANDS, THE CHANGED HISTORY COULD CAUSE BIG PROBLEMS TO THEIR REPOSITORIES. We can only change git history when we're the only one owners of the repository.

git commit --amend -m "Corrected commit message"
Enter fullscreen mode Exit fullscreen mode

We forgot to add a file to the last commit. We want the add the file without committing again.

git add file.c # get the file in the staging area
git commit --amend # this will add the new file to last commit, also it opens a log in Vim, exit with :wq
git log --stat # show file changes in commits
# The last commit hash will be changed, so the git history will be changed
Enter fullscreen mode Exit fullscreen mode

We made commits to the master branch instead of our working branch. Fix: we "move" a commit(hash) to the master and return the state of the master branch

# from master's branch
git log
# grab/copy (the first 6-7 characters of) the commit hash that we want to cherry-pick
# change to our working branch
git checkout [my-branch-name]
git cherry-pick 1b818d3b
git log

# Now delete the commit from master
git checkout master 
git log
# grab/copy the hash of the commit before our wrong commit
git reset --hard 2e75207
git log
git clean -fd
Enter fullscreen mode Exit fullscreen mode

WARNING: Again, this will change git history and will cause consequences when working in a team!!! I'll write some alternatives in the next sections.

Types of git resets

SOFT RESET

git log # grab the hash of the commit we want to keep, the commits after that commit will be removed
git reset --soft 2e7520782

git log
git status
Enter fullscreen mode Exit fullscreen mode

Git soft reset will set us back to the specified commit BUT will keep the modified and new files from the unwanted commits (the ones we removed) in the staging area ("added files ready to be committed" area) - we still didn't lose our work, but we can discard with git reset HEAD -- ..

MIXED RESET (DEFAULT)

git log # grab the hash of the commit we want to keep, the commits after that commit will be removed
git reset 2e7520782

git log
git status
Enter fullscreen mode Exit fullscreen mode

Git mixed/default reset will set us back to the specified commit BUT will keep the modified and new files from the unwanted commits (the ones we removed) in the working directory ("untracked files, before executing add -A" area) - we still didn't lose our work, but we can discard with git checkout -- ..

HARD RESET

git log # grab the hash of the commit we want to keep, the commits after that commit will be removed
git reset --hard 2e7520782

git log
git status
Enter fullscreen mode Exit fullscreen mode

Git hard reset will set us back to the specified commit AND will make all the changes in files to match the state that they were in the specified commit - we've lost our work.

However, a hard reset will not affect untracked files (newly created files from the unwanted commits, but it's irrelevant if we didn't create any new files). We can get rid of these untracked files with git clean -fd.



NOTE: git clean -fd could be useful when we accidentally unzip an archive in a project directory (local repo) and we don't want to manually delete all the new files created.

Fatal: We did a hard reset on some changes but we realized that we actually need them: git reflog (or we deleted last commits)

This "fix" works if we screwed up with git checkout HEAD^1 or git reset --hard HEAD^. (HEAD^ is short for/same with HEAD^1).

Luckily, git garbage collector (gc) collects/deletes (forever) lost commits after 30 days (IF WE DIDN'T ALREADY RAN git gc COMMAND).

git reflog
# grab the hash before executed reset command
git checkout [0c8189]
git log # happily see our changes back 

git branch
# HOWEVER, we're in a detached head state - we are on a branch that would be trashed in the future, so we need to save those changes in a newly-created branch
git branch backup
git checkout master
git branch
Enter fullscreen mode Exit fullscreen mode

Now we've successfully recovered our lost changes, we can merge the backup branch with master (git merge backup) OR if our changes are already in master branch (do check), we can delete the backup branch (git branch -d backup).


Undoing a commit after pushing to remote server. Fix without changing the git history

Undo a commit (when other people already pulled the changes), without rewriting the git history. We use git revert to create a new commit on top that reverses the changes of earlier commits.

git log # select the commit hash THAT WE WANT TO UNDO (the wrong commit)
git revert [1b818d3] # will also show a message in Vim, :wq to exit
git log # you can see the new revert commit

# You can also see the revert diff
git diff [1b818d3] [hash from revert commit]
Enter fullscreen mode Exit fullscreen mode

Using the git stash command ("temporary" commits)

Useful for changes that you are not ready to commit yet, but you need to switch branches (or revert back to another code) work temporarily in another part of the project.
NOTE: If you don't commit your changes (modified files) and you switch to another branch, your code will be lost.

git branch my_branch
git checkout my_branch
# Make changes to the code, realize you have to switch branch for a moment
git stash save "Worked on login function"
# git diff / git status will show "working tree clean" -> after pushing to stash, all modifications to files are gone.

# You can now switch branches / cherry-pick commits / work on other part of project, when you come back:
# Option 1:
git stash list
git stash apply stash@{0} # after executing this, the saved stash will still be listed in stash list
# Option 2:
git stash pop # grabs the very first (top) stash, applies changes then drops that stash from stash list
Enter fullscreen mode Exit fullscreen mode

You can also drop/delete stashes in stash list:

git stash list
git stash drop stash@{2}

# Or delete all the stashes in the list (assume that all those changes were junk/no longer needed)
git stash clear
Enter fullscreen mode Exit fullscreen mode

NOTE: You can't merge two stashes (eg. git stash pop twice) -> will show Error: files would be overwritten by merge, please commit your changes or stash them before you merge.

NOTE: The same stash list is accessible to every branch => Useful scenario: If you've written all your changes to code in a wrong branch (master) you need to commit to another branch, just stash the changes git stash save "Worked on login function", git checkout another_branch, then grab changes from the stack (stash apply stash@{?}/stash pop).

Credits / Notes taken from these tutorials (which I highly recommend for more in-depth explanations):

💖 💪 🙅 🚩
radualexandrub
Radu-Alexandru B

Posted on October 5, 2020

Join Our Newsletter. No Spam, Only the good stuff.

Sign up to receive the latest update from our blog.

Related