Git Flashcards
What am I going to learn?
Everything I need to know to use Git like a pro
Work effectively with others
Fundamental concepts
Creating snapshots of code
browsing project history
collaborating using Github
rewriting history
What is Git and why is it popular?
Hint: track history, work together
data:image/s3,"s3://crabby-images/378de/378de7a3e812514e8b91138e8eed531b352d0ed3" alt=""
most popular version control system in the world
More than 90% of software projects globally use Git
Almost every job description for software developer mentions Git
records changes made to code over time in a repository
can look at project history to see who made which changes when
can revert project back to earlier state
data:image/s3,"s3://crabby-images/6ceaa/6ceaa486f22c282ca74c0fc55a08b25f37241788" alt=""
What is a centralized version control system?
All team members connect to central server to get latest code + share code
Problems?
Single point of failure
if server goes down, cannot collaborate
must wait for server to go back up to work together and save snapshots of code
data:image/s3,"s3://crabby-images/afe37/afe374ffc2bc54b0367ad46e1da98d68c2e7a72b" alt=""
What is a de-centralized version control system?
every team member has a copy of the project with it’s history on their machine
we can save snapshots of the project locally, on our machine
if server is offline, can synchronize directly with others
What must be in your toolbox if you want to get hired as a software developer?
Know the in’s and out’s of Git
How it works
How to track project history
How to work together effectively
How can we use Git?
Command line
fastest way, many people use
this course will focus on command line
Code Editors and IDEs
built in support for Git features w/ plugins for special features (GitLens, VS code, etc.)
Graphical user interfaces
tools for windows, mac, windows, Linux
GitKraken (GUI for Git) integrates with GitKraken boards (error tracking), timelines (project roadmaps)
have limitations, not always available
What are the different levels we can config Git?
System - all users
Global - all repos of the current user
Local - current repo
We can have different settings for different projects or repositories
data:image/s3,"s3://crabby-images/8bee4/8bee4759d0767690cc66756042731aa1905f21cb" alt=""
What settings do we need to specify?
Name, Email, Default Editor
Line Ending (auto.crlf)
data:image/s3,"s3://crabby-images/dc6d1/dc6d146a7261a191f072b3cd7c002b957da982f0" alt=""
Why do we have to handle “end of line” settings properly?
data:image/s3,"s3://crabby-images/f9b86/f9b8632f6af2bdd40defc880ce3a44eef8150bc8" alt=""
Windows and Mac have different end of line characters
If we don’t setup line ending properly, will end up with issues down the road
Solution?
Configure variable corecrlf (carriage return line feed)
Windows Users (Git)
remove carriage return line characters (check into repo)
add carriage return line characters (check out repo)
Mac Users
don’t add carriage return characters
remove carraige return line characters if present before (check into repo)
Input - only modify end of lines, when storing into repository
data:image/s3,"s3://crabby-images/1ccc2/1ccc28359bf4ffa4467e5129b1cbd2da52300067" alt=""
How do we create a new repository?
Commands
$mkdir RepoName
$cd RepoName
$git init
Under the Hood
(implementation detail)
creates a .git file w/in directory (hidden)
.git file stores info about our project
project history, branches, hooks, objects, info, references
purely implementation details
don’t corrupt this file (will lose project history)
data:image/s3,"s3://crabby-images/0ac34/0ac34501ab97545cefbcf9e74a3921a5db62d4f2" alt=""
What is the git workflow?
data:image/s3,"s3://crabby-images/f0820/f082053c2b159669816334c3996633f0777d4b93" alt=""
Working Directory (Project Directory)
local to my machine
Index (Staging area)
.add command
intermediate step
review code before commit
Git Repository
subdirectory (hidden) in working directory
data:image/s3,"s3://crabby-images/79110/791105651c67e39c92b07cc9390bb74b7b36f6f6" alt=""
What does each commit contain?
Hint: complete snapshot
each commit contains…
a unique identifier
generated by git
information about change
Message
Date/Time
Author
complete snapshot of project (not only changes)
can revert to earlier version quickly without computing changes
data:image/s3,"s3://crabby-images/65ed9/65ed998cbe6a7e84a36f6b45f13369720cd983d6" alt=""
What is the echo command?
A standard Unix / Linux command for writing content to a file
echo hello > file1.txt
data:image/s3,"s3://crabby-images/1c567/1c56789c58fb63e67eaab4667dd230e857078716" alt=""
What does $git status show us?
see status of working directory or staging area
data:image/s3,"s3://crabby-images/636b7/636b78ce3cf0b1e6d7fe24470fac34d90a366f36" alt=""
How do we add files to staging area?
git add * .txt
all files with .txt ending
git add file1.txt file2.txt
multiple files separated by space
git add .
adds all files in current directory
beware some files we don’t want to add (.gitignore)
data:image/s3,"s3://crabby-images/f9f0a/f9f0adc009fb401c9d18815771b125e4c59f4edd" alt=""
If we stage a file, then make a change locally, what must we do?
$git add fileName.ext
add to staging area again
Why?
Git stores a snapshot of file when $git add . to staging area
data:image/s3,"s3://crabby-images/8a165/8a16511d8b42d9e3c15d8da817a0b7353c43e267" alt=""
How do we store our snapshot permanently in our git repository?
$git commit -m “meaningful snapshot description”
bug constraints explained
data:image/s3,"s3://crabby-images/dbefb/dbefb29b8534bd761f80f56d8a3f01300cd6b663" alt=""
What are best practice guidlines for commits?
data:image/s3,"s3://crabby-images/60d8a/60d8a468d9c2247860c69fba296fe1d1348f0e49" alt=""
whole point of commits is to record checkpoints
if a mistake is made, can go back to last checkpoint
each commit represents a single unit of work
DO LIST
commit often (5-10 times per day, depending on work)
commit at each checkpoint
As I reach a state I want to record, THEN make a commit
each commit should represent a logically chained step
give each commit message a meaningful comment (messages show in history)
Use present tense in commit message
DO NOT DO
make a commit everytime a file is updated
wait three days, then make a commit
build a feature end to end before making a commit
Do not wait until BIG commit to make a commit
commit two logical steps together
Ex. don’t commit bug fix / typo fix together (separate these)
How do we skip staging?
99% of time, stage code before commiting
Beware (only due if sure, don’t need to review)
-a all
-m message
$git commit -am
How do we remove a file from working directory?
Hint: remove from working directory,
.add + commit to remove from staging directory
$rm file2.txt
linux cmmd removes from working directory
still in staging directory (ls)
Soln?
Must stage changes using add command
$git add file2.txt
stages deletion
$git commit -m
data:image/s3,"s3://crabby-images/e3dcd/e3dcd8a83228757432a1dca1428020935048fbe8" alt=""
How do we remove a file from working directory and staging files?
$git rm file2.txt
How do we move or rename files?
Hint: $git mv newFileName oldFileName
$ mv file1.txt main.js
rename file1.txt to main.js
Two step process:
- Modify working directory
- Stages two changes (add / delete)
$git mv
does this for us vs. standard unix command
changes are applied to both working directory and staging directory
$git commit -m
data:image/s3,"s3://crabby-images/db3f9/db3f927c64571facd2aa481029cf7c27f9daa491" alt=""
How do we ignore files?
data:image/s3,"s3://crabby-images/67837/67837fe0efc7845ea05df4c7c8e24415c4eec484" alt=""
certain files we want to ignore:
log files: individual for each developer, don’t want to include
node_modules: increase size of code, etc.
Solution?
Don’t want to add to staging area, don’t want git to track these
.gitignore (in root of project)
Beware?
If you accidentally add a file to your repository
THEN, add to gitignore, it won’t work
How do we remove a file from the staging area that we don’t want git to track?
must remove from staging area (index)
$git rm
removes from working directory and staging area
$git rm –cached -r bin/
removes only from staging area (index)
data:image/s3,"s3://crabby-images/bec32/bec327244ba91a2f59d42396ed7da17f7d868f81" alt=""
What’s a shortcut for status?
Hint: git status -s
data:image/s3,"s3://crabby-images/fb86c/fb86c5965afa20bbfad697836c89cf269cdfa468" alt=""
$git status -s
short version
less wordy
easier on the eyes
Symbols?
M - modified
A - added
left column
staging area
right column
working directory
Ex. red M -> modified a file in working directory, not yet in staging
green M -> some changes in staging area
data:image/s3,"s3://crabby-images/35ae6/35ae69f762c2b2e28ef92099d9b1b5dacc0ed557" alt=""
How can we see the exact lines of code we’ve stagged?
Hint: $git diff –staged
Best practice - always review changes before commit
Terminal:
$git diff –staged
$git diff
compares what is in working directory vs. staged directory
visual dev tools (VSCode):
$git config –global diff.tool vscode
$git config –global difftool.vscode.cmd “code –wait –diff $LOCAL $REMOTE”
data:image/s3,"s3://crabby-images/fac82/fac823eb7be6c118903ed1d4420991df40937342" alt=""
How do we setup VScode for viewing differences between files?
Hint: $git difftool –staged
data:image/s3,"s3://crabby-images/ebb98/ebb98257f0d5ec643be06588f4cae03316d9b3d1" alt=""
$git difftool –staged
ex.
old copy in working directory
vs
old copy in staging area
data:image/s3,"s3://crabby-images/2a902/2a9027407690c2989bdb340dab5f91ac6a1ceb08" alt=""
How do we look at history?
Hint: $git log
data:image/s3,"s3://crabby-images/3fcb8/3fcb8e91df025c2b827428e5ac64561256d32409" alt=""
$git log –oneline
newest commit on top
$git log –onlineline –reverse
initial commit on top
data:image/s3,"s3://crabby-images/b0374/b037490a08a5157e22dee455ac1aae717a5c9dd7" alt=""
How do we see the content of a specific commit?
Hint: $git show ~HEAD1
$git show commit 7c6df80
$git show HEAD~#steps
$git show HEAD~1:.gitignore
data:image/s3,"s3://crabby-images/43031/43031f1e581fc9f0584b533199229f0b4119920e" alt=""
How do we see all the files and directories in a commit (not only changes)?
Hint: $git ls-tree HEAD~1
data:image/s3,"s3://crabby-images/ef3f0/ef3f01dda479b8f7bad3ccd25acd15166aea446f" alt=""
directory in filesystem represented as a tree
data:image/s3,"s3://crabby-images/8c0c0/8c0c0fd7b3b23219e731c46a57ca55201ce2f74f" alt=""
each directory can have children
children can be files / sub directories
How do we remove a file from the staging area?
Hint: $git restore –staged file1.js
removes all changes from staging directory
only changes left in the working directory
Restore implementation?
grabs last copy from Git repository
puts that back into staging area (last commit)
if not in Git repository?
Removes from staging
reverts back to untracked file
data:image/s3,"s3://crabby-images/ec9a1/ec9a1095c7474781506256fb1c547c38c772baaa" alt=""
How can we discard changes made in working directory ?
undue local changes
$ git restore .
will take previous version from staging area and restore that version in working directory
$ git clean -fd
removes untracked files from working directory
data:image/s3,"s3://crabby-images/31758/317587c5427f7df5826a85c920136214529f29d8" alt=""
How do we restore a file to a previous version?
$git restore –source=HEAD~1 fileName
removes file from working directory and staging area
restore file to previous version vs undue commit
data:image/s3,"s3://crabby-images/25997/25997cc8b6befab5e82b8e80a565854f370eaf2f" alt=""
How do we create snapshots using VScode?
data:image/s3,"s3://crabby-images/6cad5/6cad52eed60c624b30a365f86cec5d275f5bf642" alt=""
Open VScode
go to git extension
data:image/s3,"s3://crabby-images/62460/62460da5220941d20580e526f9238f31bebee337" alt=""
What is GitKraken?
A GUI tool for Git
can see all commits - author, date/time
data:image/s3,"s3://crabby-images/2f79c/2f79c73f9d37afc4cfa0dbc4546a2ec8d20bf0e0" alt=""
What are we going to learn in the history section?
data:image/s3,"s3://crabby-images/c026d/c026db1ade93de94a19aaf2cb60549edb365e8fc" alt=""
What does $git –oneline –stat do for us?
data:image/s3,"s3://crabby-images/18d67/18d67bcff489b37bafad881cbdd2ac999d04349a" alt=""
$git –oneline -stat
shows us all the files changed in each commit
$git –oneline –patch
data:image/s3,"s3://crabby-images/e6fe6/e6fe6d7f6556389af585f6cf739c31d88ad13af6" alt=""
How do we filter history of commits when looking at Git repository?
data:image/s3,"s3://crabby-images/0f600/0f6005ef6c883e25b87c703481417dda12705f14" alt=""
In real world, hundreds or thousands of commits
Don’t want to look through them all
filter by author, commit message, date, etc.
GIT COMMANDS:
$git log –oneline -3
returns last three commits
$git log –oneline –author=”arsalon”
returns commits made by author=arsalon
$git log –oneline –after=”2020-08-17”
$git log –oneline –after=”yesterday”
$git log –oneline –after=”one week ago”
returns commits made after specified date
git log –oneline –grep=”GUI”
case sensitive, returns commits with message containing “GUI” term
data:image/s3,"s3://crabby-images/eaf69/eaf69ea1c5e80b3f5f2c102abc43e28eccfa2cda" alt=""
How can we filter history by all commits that have modified a specific function?
Hint: $git log –oneline -S”functionName”
Ex.
$ git log –oneline -S”getJwt” –patch
brings up all commits that modify getJwt ( ) and shows changes
data:image/s3,"s3://crabby-images/e4a10/e4a1069129bad23dfa5b1e3da3436c47c62ffa58" alt=""
How do we filter history by a range (within a range)?
Hint: $ git log –oneline commitIdentifier1…commitIdentifier2
data:image/s3,"s3://crabby-images/2674c/2674cd3e507155b71b148b4ea6bc5706fc204444" alt=""
Example
$ git log –oneline 56a3436..600400c
gives all commits after 56a3436 up to 600400c (included)
data:image/s3,"s3://crabby-images/c9a33/c9a339a3648a18e258c7647b58b451a83349ca3a" alt=""
How do we customize the log output?
Hint: $git log –pretty=format:”%an commited %h on %cd”
data:image/s3,"s3://crabby-images/ae389/ae3892ecaca1623a89f78ad359f49487a2afceea" alt=""
$ git log –pretty=format:”%an commited %h on %cd”
%an - author name
%h - abbreviated commit identifier
%cd - commit date
data:image/s3,"s3://crabby-images/87446/874464ae4adf49ec3987c64bc32773e523a59a59" alt=""
What are git aliases?
Hint: $git config –global alias.unstage “restore –staged .”
data:image/s3,"s3://crabby-images/9309f/9309f72062efcb2d989d4ca29e818b652ff81c97" alt=""
$ git config –global alias.lg“log –pretty=format:’%Cgreen%an%Creset commited %Cred% %h%Creset% %Cblue% on %cd’“
Can abstract git commands into shorter alias
data:image/s3,"s3://crabby-images/304dc/304dc333deffa541a6702f3a34fbe9fadc13b5b9" alt=""
How do we see a specific commit?
Hint $git show HEAD~2
data:image/s3,"s3://crabby-images/a4c99/a4c99e4592e33f0b425a79f355a6c2b7557250fe" alt=""
How can we see the list of file names that have been changed across commits?
Hint: $git diff HEAD~1 HEAD –name-only
$git diff HEAD~2 HEAD –name-status
shows status of files changed (modified, added, etc.)
data:image/s3,"s3://crabby-images/ed054/ed054017604d9a4878534134bb77e4dc2adbead2" alt=""
How do we restore our working directory to a particular snapshot from a commit?
(ie. restore all files to a previous snapshot version)
data:image/s3,"s3://crabby-images/28ac7/28ac7a6f951cf6cef9287f8ab6461a23cc4d177b" alt=""
$git checkout 6c325d5
working directory will look exactly like an earlier point in time
data:image/s3,"s3://crabby-images/2bd4c/2bd4c22484643daf147b7cc94d3a2bb5b1f4b039" alt=""
What is a detached head?
data:image/s3,"s3://crabby-images/120e8/120e81abe77e434769cb267c51595cc7f4b5a637" alt=""
$git checkout master
restores head to master branch
git When we checkout a particular commit
head will point to that commit
Head isn’t pointing to a branch (master)
it’s pointing to a particular commit
DON’T MAKE CHANGES
ONLY LOOK AROUND
data:image/s3,"s3://crabby-images/d8bbd/d8bbd5c70ea9a3da7c12aa54080493f4d0094de6" alt=""
What is bisect?
data:image/s3,"s3://crabby-images/02cf4/02cf42ed4e9c4458d24e94b9f06f2072a8ba00fe" alt=""
GIT COMMANDS:
$git bisect start
$git bisect good ###
$git bisect bad ###
$git bisect reset
an incredibly powerful tool for finding bugs
can quickly find the commit that introduced the issue
How?
Give Git the bad commit code (one with bug)
give Git a good commit code (one without bug)
like binary search (log n )
ref/bisect/bad -> bad commit labeled
(Head) -> in middlepoint between two*
refs/bisect/good -> good commit
*working directory restored to middlepoint snapshot (head)
*run tests to see if issue is still there
*if issue is not there, tell bisect commit is good, continue
data:image/s3,"s3://crabby-images/c19a5/c19a598c364fadfcaee83b243b8e19c491d7268c" alt=""
How do we view an authors contribution to a git repository?
Hint: $git shortlog
data:image/s3,"s3://crabby-images/c3228/c32280f5bff1a054ce6898698a1fb64cf34740dd" alt=""
$git shortlog -n -s -e –before=”” –after ““
-n number of commits (filter author by)
-s suppress commit messages
-e show email address
-before/after to see contributors for certain date rangs
data:image/s3,"s3://crabby-images/f7016/f701698bf80b4d9feb808990e5c03b0fa02b406a" alt=""
How do we view history of a file?
Hint: $git log –oneline –stat fileName.ext
data:image/s3,"s3://crabby-images/2c7ab/2c7ab1058ce405736f2e9a9cea1f93f7e619acb0" alt=""
$cd
$ls
$git log –oneline –stat index.js
data:image/s3,"s3://crabby-images/4c148/4c14827d4aa0c2d859018ae9e7aaf31335f04d34" alt=""
How do we restore a deleted file?
$git checkout previousCommit pathName.ext
$git commit -m “restore pathName.ext”
How?
Look at parent of file, get ID
checkout only deleted file from commit
Situation?
remove file fixing a bug or implementing a new feature
want to restore the file
data:image/s3,"s3://crabby-images/ba0be/ba0be0cba999bd8db65ba2256a8175f67f14c56a" alt=""
How do we find the author of a particular line of code?
$git blame filename.ext -L 1, 11
data:image/s3,"s3://crabby-images/dbdc1/dbdc19eb357ba640d0ef1c85eccf046ff4df8567" alt=""
$git blame filename.ext -L 1,11
-L lines
1, 11 line 1 through 11
Part of troubleshooting issues, want to know who wrote a line of code
$git blame fileName.ext
in front of each line of code, additional information
commit ID
author name
date/time inserted
Ex.
git blame -e -L 1,11 authService.js
data:image/s3,"s3://crabby-images/b17a2/b17a2961a825625c32ea9726a809a7b6e920d1f1" alt=""
What are tags?
$git tag -a v1.1 -m “message”
$git tag -d v1.1
data:image/s3,"s3://crabby-images/c2c0d/c2c0dadbaaf4fd9863c556272287cb9b1a9a31f5" alt=""
allow us to bookmark commits like a certain version
lightweight vs annotated tage
(light weight tag)
$git tag v1.1 commitID
a reference or pointer to particular commit
(annotated tag)
$git tag -a v1.1 -m “my version 1.1”
$git tag -n
creates annotated tag (object) with properties
can associate a message with tag
$git tag -d tagID
deletes tag
data:image/s3,"s3://crabby-images/5bdf1/5bdf18028c4731ee77c49bafc57b869fcca97961" alt=""
What is gitlens?
data:image/s3,"s3://crabby-images/c0ef7/c0ef746d528991cc3a474df3806387a723164f03" alt=""
VSCode plugin
Pros
can view commits
can view diff
can compare commits
can see files changed
can search by messages, etc.
Cons
cannot search by multiple criterias
cannot use bisect to find bugs
Mosh Advice
GUI tools are great for diff
seeing how two files are different
data:image/s3,"s3://crabby-images/f8206/f820662190f937260c910afed6c5f2d731d9a4be" alt=""
How do we view history in GitKraken?
Can right click, add tags, checkout, view diffs
What is branching?
Hint: totally change how you develop software
data:image/s3,"s3://crabby-images/f14c8/f14c832c3a74f141d92669910b1fa7f379133e0c" alt=""
One of the most powerful Git tools
the tools learned in this section
can totally change how I develop software
use branches
to work on a feature in isolation
like a separated isolated workspace
data:image/s3,"s3://crabby-images/bfdd2/bfdd2ef27e72579570a5c72a0da659e94a97ab5a" alt=""
How do we use branches to develop features in isolation?
without messing with main line of work (master)
work on various work items (features)
after testing and verifying feature works
bring code back to master (merging)
What is a branch in git?
just a pointer to a commit
tiny file with a 40 char bit ID
Head tracks which branch currently working
Ex.
master branch is just pointer to last commit in main line of work
git moves this pointer forward automatically
data:image/s3,"s3://crabby-images/84056/840564bf0e6c83168f46ab7bcacd991d9b40fb33" alt=""
How do we fix a bug?
$git branch bugfix
create a branch to fix bug
$git branch -m oldName newName
rename it to specific bug
Ex. changes only visible in bugfix branch
make changes
$git add .
$git commit -m “fix bug”
$git branch -d branchName
if we switch back to master branch (update head pointer)
won’t see changes
in future, will have to merge branches
delete branch
data:image/s3,"s3://crabby-images/573ef/573ef90b92e21405f0be2f03cc691cec5ae37f97" alt=""
How can we assess how a branch is divering from master?
ie. comparing branches (master w/ sub branch)
$git diff bugfix/signup-form
shows differences between current branch and listed branch
when branches are merged can see expected changes
$git diff –name-status bugfix/signup-form
shows on the file that have been changed in branch
$git log master ..branchName
$git log master ..bugfix/signup-form
data:image/s3,"s3://crabby-images/1148b/1148b33540c432efd4c604ad732616bb5087ec7e" alt=""
What is stashing?
data:image/s3,"s3://crabby-images/85898/858984d9864697a8474489bbc05c2f1a5883367a" alt=""
Hint:
$git stash push -am “message”
$git switch master
$git stash show 1
$git stash apply 1
$git stash drop 1
$git stash clear
COMMANDS:
$git stash push -am “message”
$git switch master
$git stash show 1
$git stash apply 1
$git stash drop 1
$git stash clear
When we modify a file in working directory of a branch
and we want to switch to a new branch but have not commited changes
we don’t want to commit because we are not finished
we can stash changes (store in a safe place)
NOTE
new, untracked files are not included in stash (default)
$git stash push -am
to push all files (untracked) into stash
data:image/s3,"s3://crabby-images/7f4c5/7f4c574ecc6b32fa34f749dd8f8a2a672e0243ff" alt=""
What is merging?
bringing changes from one branch to another
Two methods:
Fast-forward merge
moving the master pointer to the updated branch
if two branches have not diverged
there is a direct linear path from target to source
Three way merge
if branches have diverged
additional commit that combines versions is created
What is a fast-forward merge?
data:image/s3,"s3://crabby-images/cff60/cff607f563c6a51dbc56e0400f63c56d4b060d74" alt=""
bring master pointer forward
if two branches have not diverged
there is a direct linear path from target to source
brings pointer of master forward
Ex. Create a bugfix branch
code is identical to master
make updates/changes to bugfix directory
copy all files from bugfix to master?
if thousands of files??
no additional changes in master
files in master are first version of bugfix
simply, rename bugfix to master
data:image/s3,"s3://crabby-images/d62bb/d62bb4c6e3b7d40dde6d42ed2a4f9098ae823b2c" alt=""
What is a three way merge?
additional commits on master (after branch)
changes in master that don’t exist in bugfix branch
soln?
git creates a new commit that combines changes from both branches
git looks at before commit and after snapshots (two)
How do we create and switch to a branch simultaneously?
data:image/s3,"s3://crabby-images/ae573/ae57310ccb722cd820b6f6c159d4cf8bb4869992" alt=""
$ git switch -C bugfix
How do we force a merge-commit?
Hint:
$git merge –no-ff bugfix/login-form
data:image/s3,"s3://crabby-images/153b9/153b99b823bd253768987ba8f61705748e180bb5" alt=""
$git merge –no-ff bugfix/login-form
allows us a true history of a project
allows us to revert back to previous version
data:image/s3,"s3://crabby-images/c18d2/c18d263fc876901a519bd6646c35b8df1bbed51e" alt=""
What’s the benefit of not using a fast-forward merge?
data:image/s3,"s3://crabby-images/05395/053951ea30fa433770d4ba0c3a360fa508da1daf" alt=""
new merge commit combines all changes in branch
can revert to an old commit
if using a merge commit, a single commit to revert
vs.
if fast-forward merge, have to undue more commits (two)
data:image/s3,"s3://crabby-images/f429a/f429a3b10cbf48d395ba3dc5b1056444f630f896" alt=""
How do we disable fast-forward merging in the current repository?
$git config ff no
Some companies may have a policy of no fast forward merging
How do we run a three way merge commit?
data:image/s3,"s3://crabby-images/6ee1b/6ee1b106c4968fd83c7c218c766e20f3a0db1c59" alt=""
$git merge branchName
looks at tips (last commit in each branch)
and common ancestor
creates a merge
data:image/s3,"s3://crabby-images/7b5a9/7b5a98e0d9b57597e2c63eb051b894797071d903" alt=""
How do we see which branches have already been merged with master?
Hint: $git branch –merged (no–merged)
data:image/s3,"s3://crabby-images/5d73b/5d73b67f43100c866765299bd54cfaff255e2202" alt=""
$git branch –merged
shows branches merged
$git branch -d branchName
deletes branch
$git branch –no-merge
shows unmerged branches
Best Practice:
When done with a branch, delete it
otherise pointer will sit / create confusion in future
data:image/s3,"s3://crabby-images/d47bf/d47bfdd6f487ae81fa9cdb96167d4c2a75756cd2" alt=""
What is a merge conflict?
data:image/s3,"s3://crabby-images/8bf49/8bf49c7f248e426b2f4d9ace7666fab42f789fcd" alt=""
Merge Conflicts:
Same line of code has been changed in different ways in different branches
One branch changes a file, another deletes a file
Two branches add same file twice, content is different in both files
Soln?
goog
data:image/s3,"s3://crabby-images/5eb67/5eb679dfdf7219ebabfbbb0f0ec0fc1d4aafda72" alt=""
How do we resolve a merge conflict during a merge commit?
Hint: Don’t add new lines of code during a merge commit
data:image/s3,"s3://crabby-images/d4b40/d4b40325dcbf6d17be1e5cc55197fb8842634682" alt=""
VSCode:
open file
click buttons (accept current, accept incoming, accept both, compare)
add file to staging area
commit file to repository (modified)
Manually:
remove markers for head, divider
don’t add new lines of code during merge commit (evil commit)
Third Party Merge Tools
also available for handling merge conflicts
data:image/s3,"s3://crabby-images/cf18e/cf18edab0464db810eac8b1fac2fe58606112a73" alt=""
What is an evil commit?
adding new lines of code during a merge commit
in real world, sometimes not avoidable
AS MUCH AS POSSIBLE, try not to introduce new lines of code
data:image/s3,"s3://crabby-images/e52b1/e52b11cce99a764fa94a3c84a79b2daa4c0e1d20" alt=""
What are merge tools?
Hint: if using VScode or GitKraken, don’t need these tools
data:image/s3,"s3://crabby-images/f7452/f7452f638c2a31c6498d839820d0e443b8959c83" alt=""
can visually see differences
can toggle between multiple files or conflicts
save changes
add modified file to staging
commit the file
How do we abort a merge?
$git merge –abort
takes you back to the state before you start the merge
back to a clean state before starting a merge
data:image/s3,"s3://crabby-images/8d396/8d396527e21791c2967a6e7e3c71f570837d8192" alt=""
How do we undue a merge?
data:image/s3,"s3://crabby-images/310b7/310b775c4bbea57a18c4d805749b3581d3bd643d" alt=""
something happens, application breaks, merge unsuccessful
undue the merge
Two options:
remove commit (re-writing history)
okay, if only local, not shared with team
revert commit
create a new commit, cancels all changes in bad commit
How can we remove a commit?
Hint: git reset –hard HEAD~1
data:image/s3,"s3://crabby-images/8aec8/8aec8403a861326b3965ec27cec93877a682fb38" alt=""
WARNING
only do this if on local repository (haven’t pushed to a team repository)
reset the head
Resetting:
Hard
points head to new snapshot, sets working directory, staging directory to snapshot
all working environments are identical
Mixed
points head to new snapshot, staging directory to snapshot, working directory unchanged
Soft
head pointer points to new snapshot, working/staging directory not updated
data:image/s3,"s3://crabby-images/72557/72557b30794ba2a5a8e95af7dbf0d54bcde4084a" alt=""
How do we revert a merge?
Hint: git revert -m 1 HEAD
data:image/s3,"s3://crabby-images/47d51/47d51d63423240a358d2911595499a1b78714496" alt=""
Use if shared history with collaborators.
Creates a new commit reverting merge commit
doesn’t change history
Each merge has two parents:
one on master branch
one on secondary branch
tell git how to revert:
to last commit on master branch
merge commit is on master branch
merge commit should be on master branch also
$git revert -m 1 HEAD
-m 1 = first parent
HEAD = target commit (last commit)
data:image/s3,"s3://crabby-images/90d14/90d147efe90a4cb0059866e9e0cb6e0d19153ded" alt=""
What is squash merging?
Hint: git merge –squash branchName
data:image/s3,"s3://crabby-images/e3a91/e3a91e439ce5bb483d950f2005807dd1bb9eeeed" alt=""
$ git merge –squash branchName
$git add.
$git commit -m ““
creates a new commit in stagging area that combines changes from branch into staged files
apply this commit to master
Doesn’t have two parents
Not a merge commit
no reference to the sub branch
just a regular commit added on top of master
not a merge commit
Benefits?
left with a simple, clean history
When to Use?
do not want them to pollute history of master branch
small, short lived branches
bug fixes
use if branch commits are not high quality
features can implement in a few hours / day
make sure to remove target branch!
$git merge –squash branchName
$git branch -D bugfix/photo-upload
data:image/s3,"s3://crabby-images/96577/9657797712fd1df129d6f445e673e3d2ee9b9a5c" alt=""
What is rebasing?
data:image/s3,"s3://crabby-images/65877/65877c74fc028c9774653c8a24d6c7007b695c89" alt=""
can change base of our branch
can base it on latest commit on master
Why?
Creates a direct linear path (fast forward merge)
Cautious?
Rewrites history
only use for local commits
do not use for collaboration
data:image/s3,"s3://crabby-images/1b8b8/1b8b817528ba636c60690d2569200e58ff4dfff1" alt=""
What do we do if there is a conflict when rebasing?
data:image/s3,"s3://crabby-images/476ff/476ff9dca012df02c13f1d72fc7420d39ae34a07" alt=""
launch merge tool
resolve conflict
$git rebase –continue
data:image/s3,"s3://crabby-images/463f2/463f296405f1d761bc26c0b2b8fb5dbb9eaf45f3" alt=""
What is cherry picking?
data:image/s3,"s3://crabby-images/19146/191463fd99aa260d61ef6faae82e9db7017e3fcc" alt=""
$git cherry-pick commitID
new commit cherry picked from sub-branch
picking a particular commit from another branch into master
not the entire branch, just a particular commit
data:image/s3,"s3://crabby-images/4be99/4be99a63b34dce8de3017100c3b004794530892e" alt=""
How do we take a file from another branch and bring it into the current branch?
git restore –source=feature/send-email –toc.txt
restores file in working directory
from latest version in branch specified
data:image/s3,"s3://crabby-images/7295f/7295f6d8d7c99aad440aeb66be9c0aff1cc79422" alt=""
How do we work with branches in VScode?
data:image/s3,"s3://crabby-images/9d365/9d365f045573d46f467541a5f6d595a5d2f6b4d8" alt=""
can view, create, etc branches
can resolve merge conflicts, etc.
can revert commits, etc.
data:image/s3,"s3://crabby-images/8b1ad/8b1adc3559341209ca9e1e0185fe7511e04ee3dd" alt=""
What is git graph?
VS code plugin
data:image/s3,"s3://crabby-images/02130/021307ee6f301e2c19fa1b520b85cc0d201fa000" alt=""
How can we use GitKraken for branches?
data:image/s3,"s3://crabby-images/91111/91111eea7d57f3d2b45f289a089b22ef19623ff8" alt=""
Makes it incredibly easy to manage branches, resolve conflicts
What will mosh teach us about collaboration?
Super important tools, techniques we must master
Master these tools to collaborate with others
Will use every single day when working with a team
Collab workflows
Push, pull, fetch remote repos
GitHub collab tools - milestones, issues, etc.
contributing to open source projects
data:image/s3,"s3://crabby-images/d9e32/d9e321035ca560fc3fdb1bf7cbb262f9568618f6" alt=""
What is the centralized workflow?
data:image/s3,"s3://crabby-images/b9274/b927452da3633eaccabb7bc8d6ad91a7df05b7bb" alt=""
workflow used in most private teams
everyone has their local repository
central repository used to synchronize work
if central repository goes down, can work offline in remote repository
can synchronize directly with members (often complex)
Where is central respository?
private server in private network
cloud - github, bitbucket
data:image/s3,"s3://crabby-images/ea6ee/ea6eec9debb1a01afbe3b08ed9a3ddf3e24d07fd" alt=""
What is integration-manager workflow?
data:image/s3,"s3://crabby-images/4f6c2/4f6c26cee71c06738ca728d6485d2961e2d3c899" alt=""
used in open source projects
only maintainers have access to push code into central repository
Process?
fork repository
clone on our local machine
make commits
push to fork
maintainer can pull and review
if good, will push to main repository
data:image/s3,"s3://crabby-images/4f2c8/4f2c8ba5ccc83159441260427ead5ce61dc4827b" alt=""
How do we add collaborators in Github?
Settings -> manage access -> invite collaborators
data:image/s3,"s3://crabby-images/4af8d/4af8d26766d7dbc050829272ca24477fd6b7dcb4" alt=""
How do we clone a github repository onto our machine?
data:image/s3,"s3://crabby-images/07d39/07d397060736077b97d50ecb4e523b1fc868a484" alt=""
git clone http://etc
data:image/s3,"s3://crabby-images/d498a/d498abb337acff0de7f8592e0f33ec1946b82bcc" alt=""
What is origin/master?
A remote tracking branch
tells us where master branch is in origin
local repository vs. remote repository (history can change independently)
origin/master tells us where master branch is in origin repository
data:image/s3,"s3://crabby-images/cccac/cccac0ed7e335ca02275aed55f35efb56fc1354d" alt=""
Is our local repositry connected to our remote repository?
$git fetch
No,
if a change is made in remote repository, local repository is unaware
$git fetch
downloads new commit to our local repository from remote repository
git will download commit / move origin/master forward
to move changes from remote tracking back to local repositry,
merge local branch with origin/master branch
if there is a conflict, resolve like normal
data:image/s3,"s3://crabby-images/cdc54/cdc54479329891c9c8a59351e52093b03ec3cc57" alt=""
How do we bring the changes in our remote repository to our local repository?
data:image/s3,"s3://crabby-images/c6c14/c6c14b12eac24810b4a16bdfbd1ccae48b02efcf" alt=""
$git fetch
grabs the current snapshot in origin/main
$git merge origin/main
merges local repository with snapshot from origin/main
data:image/s3,"s3://crabby-images/8b8ac/8b8ace825f7a1add5fdbd7318bd9863e21b8d5f1" alt=""
How do we bring changes from our remote repository into our local repository?
Hint: pull combines two commmands
fetch + merge
$git fetch origin
$git merge origin/main
$git pull
combines fetch and merge commands
What is pulling?
data:image/s3,"s3://crabby-images/0919e/0919e4d1a1fced293247ba18739dae0794e2d7a3" alt=""
combining fetch and merge commands
$git pull
creates a three way merge commit
pollutes history
$git pull -rebase
creates a rebase action
creates a linear history
data:image/s3,"s3://crabby-images/cfb52/cfb52478fd185f31db26cb82f896c51d77afc441" alt=""
What is three way merge pull?
data:image/s3,"s3://crabby-images/2cc53/2cc53db893a815b21308c79a4ca64db848dea525" alt=""
data:image/s3,"s3://crabby-images/5cb66/5cb6638cbdfaa3f3faa39fc9b7000db54ac45117" alt=""
What is a rebase pull?
data:image/s3,"s3://crabby-images/7b06f/7b06f29febbc6573cda733be34dd3544c6e4261a" alt=""
$git pull –rebase
Creates simple, linear history
data:image/s3,"s3://crabby-images/16819/16819f71bccca1771a2f052a23671092d0708c13" alt=""
What is the push command?
data:image/s3,"s3://crabby-images/b3e97/b3e978f10d649c758f41efaac23d83f23ed943f4" alt=""
if we have local changes in our repository
push will send them to remote repository
Can be rejected, why?
someone else puts work in the remote repository
First, pull and do a merge (or rebase)
if conflicts, solve them
Then, do a push
How do we store github credentials for push requests?
keychain
git config –global credential.helper osxkeychain
data:image/s3,"s3://crabby-images/f3725/f3725c4025a727b04f4acf4ca29de61dac89b9cc" alt=""
How can we create / store tags in github?
data:image/s3,"s3://crabby-images/b934c/b934c83f35b21234c1d7a6414373644ce008d793" alt=""
very useful, allows us to download entire source file up to that point
data:image/s3,"s3://crabby-images/49021/49021c7d43f14573b4b8f69c6fd2abbf9628e9e5" alt=""
How do we delete a tag from github?
$ git push origin –delete v1.0
deletes tag from repository
$ git tag -d v1.0
deletes tag from local repository
data:image/s3,"s3://crabby-images/070ce/070ce6178676291a46fcd59b1a32df9eda2dc911" alt=""
What are releases?
data:image/s3,"s3://crabby-images/d4da0/d4da091233e0c81d0fa6a239db9c7e38cc8cd1f3" alt=""
a gitHub feature
can create release to package software, binary files (compiled version of app), release notes
data:image/s3,"s3://crabby-images/a439e/a439eec24ae741135e4a9e37c128edff79b3d57e" alt=""
How do we share branches?
Hint: git push -u origin feature/change-password
if collaborating on a branch
must explicitly push branch to github
git push -u origin branchName
-u upstream
creates a branch in github
can use it like main branch
push commits, etc.
git push -d origin feature/change-password
deletes from origin
data:image/s3,"s3://crabby-images/ab354/ab3542344fefd6fb7cc4903711e3bcea95bd63c9" alt=""
How do we work with our remote branches?
Hint: git branch -r
data:image/s3,"s3://crabby-images/d38d5/d38d53b2516d5ed3dcb94507ca39589970830218" alt=""
first, we fetch a new branch from origin
then, we create a new branch locally that points to origin/main branch
data:image/s3,"s3://crabby-images/076e8/076e85cc8a818e15a4949527eb2bd5cc22812a42" alt=""
How can we collaborate on remote branches?
data:image/s3,"s3://crabby-images/fb93c/fb93c2bcdb3fda3ab5973de2ebbe69a22ecc6872" alt=""
git pull
gets the latest commits on branch from collaborators
git switch main
return to main branch from sub branch
git merge feature/change-password
merge main with feature branch (if finished on feature)
git push
send branch changes merged into main to origin/main
git push -d origin feature/change-password
delete branch from origin repository (remote repo)
git branch -d feature/change-password
delete branch off local repository
Collaborator:
create a local branch
point origin/branch to local branch repository
make commits, push
when finished, merge with main and push
data:image/s3,"s3://crabby-images/111ed/111ed697824c8d8f0b4eacdccb3355bfc52c97c7" alt=""
How do we remove remote branches that have been deleted from our local repository?
Hint: git remote prune origin
last step after a feature branch was merged and deleted from origin
data:image/s3,"s3://crabby-images/51407/5140708dd1704a210c21112f61966b3a7f1bd603" alt=""
What is a pull request?
data:image/s3,"s3://crabby-images/bfc1c/bfc1c51ff7471a0eb9a9e50daa1765c65ee20cca" alt=""
before merging a branch to master
opening a discussion with team mates
Why?
as part of implementing a feature / fixing a bug…
we often need feedback from team
before merging a branch to master
Who closes?
Depends on your team
person who opened it, closes it
person who opened it, doesn’t close it
close + delete on github
then…
$git remote prune origin
$git branch -d feature/login
data:image/s3,"s3://crabby-images/7b9e7/7b9e73e8d6c74c6cb682ccb668b13225ed3e3315" alt=""
How can we handle conflicts in a pull request?
data:image/s3,"s3://crabby-images/ed7bf/ed7bf1e0b2bd1cf584dd875835cdab3ee199d256" alt=""
Ex. Masterbranch moves forward, feature branch is behind and conflicted
soln
start pull request
use command line
use github user interface
command line?
same as usual
- pull in latest code
- start merge process
- resolve conflicts
- add new commit
- push to github
Github?
use the UI
click on resolve conflicts
click submit
click merge
done!
data:image/s3,"s3://crabby-images/2e2e5/2e2e5b1a2e35ffdc88d3e7f6efc0dc56bbc8540d" alt=""
What is issue tracking?
data:image/s3,"s3://crabby-images/11c7f/11c7fc39f66e445edd415ba3707fea11d05bca2f" alt=""
Goes hand and hand with pull requests
can track…
bug fixes
new features
enhancements
ideas on the team
Why?
all details for working on an issue encapsulated in a single place
much better than using email
can assign to team members
can have a conversation about the issue on the github issue page
can link to a pull request
can assign to a milestone
can come in and pickup a new issue / assign to self
data:image/s3,"s3://crabby-images/80c22/80c22b465dc2a45f6522ae4a41840043e91c2acc" alt=""
What can we do in our issue tracking?
create custom lables:
refactoring, etc.
github provides a standard / custom labels
data:image/s3,"s3://crabby-images/fae13/fae136b675c1dee78378940fe2919a8ef7671075" alt=""
What are milestones?
data:image/s3,"s3://crabby-images/8cb93/8cb93cc5e22dfc3dacb31e31768381ab2e1c612e" alt=""
track progress of various issues
add issues to milestone
see progress in milestone
assign issues to a milestone
Ex. V2.0 of Software release
as issues are completed
milestone progress is updated
data:image/s3,"s3://crabby-images/cef17/cef1733484295f269f6d2a4d2fa2c0de02b1d252" alt=""
What are we going to learn in rewriting history?
great but dangerous features
re-write history
rewrite, drop, combine, split commits, etc.
data:image/s3,"s3://crabby-images/6f6bd/6f6bd128bd0d31770b1b67d0c242d66a96193dad" alt=""
Why do we need history?
Extact meaningful information from history
See what was changed
why and when it was changed
Why do we need to rewrite history?
data:image/s3,"s3://crabby-images/2f5d1/2f5d1a0fc3e452fe4a50c1b6e03f499c89b7a205" alt=""
if we have bad history,
cannot extract meaningful information
re-write to create a meaningful story
Bad history:
commit messages are not meaningful
commits are too large
commits contain unrelated changes
commits are too small or scattered all over history
data:image/s3,"s3://crabby-images/4ef07/4ef07f9a15ed3a0e881a3a90f62dbfffff159ee8" alt=""
How can we make clean, meaningful history?
Hint: re-write bad history
too small commits
squash related commits together
too large commits
split large commits into small commits with each being a logical chain
poor messages on commits
reword commits
accidental commits
drop commits
commits not representing logical steps
can modify contents of commits - add file, etc.
data:image/s3,"s3://crabby-images/e5d8e/e5d8e18826bc2e2d8f69bf295c8fbc2bdd3e3d44" alt=""
What’s the golden rule of re-writing history?
Hint: don’t re-write public history
if pushed commits to public repository (origin)
don’t re-write!
why?
if you replace a commit someone has checked out, going to get issues
will make history look noisy
only re-write private history!
It’s a good practice
clean up private history before sharing with the world
What’s a good practice?
clean up history in private
then push to world
history should tell a story
What’s wrong with this history?
data:image/s3,"s3://crabby-images/6790f/6790ffff003c5079828a9cca15bc8430c7585b91" alt=""
poor commit message
ommited the word “on”
render restaurants ON the map
Small commits
Fix a typo - this shouldn’t be a commit
change color of restaurant icons - should be combined with prev commit
WIP - this is just noise in history, drop or change message
Update terms of service and Google SDK - combining two logic steps together, split into two commits
. - mysterious commit, drop or change message
How do we undue a commit?
data:image/s3,"s3://crabby-images/7e687/7e687fd0b76a06c4b108ebee3785cd02c607e345" alt=""
Hint: git reset –hard HEAD~1 (removes history)
Locally only?
git reset –hard HEAD~1
removes from history
-soft removes commit only, working/local untouched, working can commit
-mixed removes changes in staging area
-hard discard local changes
data:image/s3,"s3://crabby-images/c8fc1/c8fc154f389074e6c3535c71b29ad867238261a7" alt=""
How can we revert last three commits?
Hint: git revert –no-commit HEAD~3..
creates a new commit that undoes last three commits
use if history is public already
$ git revert –no-commit HEAD~3
$ git revert –continue
$ git revert –abort (if bad)
data:image/s3,"s3://crabby-images/24469/24469c6576f5b1bdada25c043eac32c3df0db0b3" alt=""
How do we recover lost commits?
data:image/s3,"s3://crabby-images/5b906/5b90610e689ebe4592d908f8f15931b47c2a494b" alt=""
Hint:
$git reflog
$git reset –hard HEAD@{1}
$ git reflog
$git reset –hard HEAD@{1}
git stores commits in project
after a while, garbage collects them
data:image/s3,"s3://crabby-images/3f052/3f05222a15b1e665fb635ead47b8d3f1f1f8f71c" alt=""
How do we modify a commit?
$git commit –amend -m
data:image/s3,"s3://crabby-images/90b24/90b243fefd10847b4f1b9444681267d7a61e6e96" alt=""
commits on git are immutable
even though on surface, ammeding a commit
git is actually creating a new commit
data:image/s3,"s3://crabby-images/0e0db/0e0db770b00cbeea6b25bcc895ed6ad392a45c43" alt=""
How can we remove a commit?
$git reset –mixed HEAD~1
working directory is dirty
mixed
points head to commit before last
puts that snapshot in staging area
working directory is still dirty
Soln?
git reset –mixed HEAD~1
git clean -fd
removes untracked files from working directory
data:image/s3,"s3://crabby-images/15cdb/15cdb9a34a55a730f43ac58f60230120046a034b" alt=""
How can we ammend an earlier commit?
$ git rebase -i commitID
$ git commit –amend
data:image/s3,"s3://crabby-images/06ca9/06ca950eb570fc59d0723469651d900e87f4f1c1" alt=""
interactive rebasing
re-writes history
have to re-create other commits along the chain
How?
$ git rebase -i commitID
open in editor, change specific commit to “edit”
make modifications
$ git commit –amend
Process?
creates a separate branch
all changes made from commit will be played on top of commit
changes are carried on across subsequent commits
branch will be discarded
data:image/s3,"s3://crabby-images/3c8c3/3c8c398515163fb3360600521a76bf66c23a3e22" alt=""
How do we drop a commit?
start rebase with parent of commit (want to drop)
Problem?
if this commit introduced a new file, next commit modifies it
will be a conflict
changes from deleted commit, added to subsequent commit
data:image/s3,"s3://crabby-images/190f4/190f46daf4e8f214eb012bd16dca1abafa531c6d" alt=""
How do we reword commit messages?
Hint: git rebase -i commit~1
data:image/s3,"s3://crabby-images/fc9be/fc9becc435e4ea5eb0573141bce0167b5f49ed8c" alt=""
change message to reword
git will pull up commit, allow us to change message
all subsequent commits are re-created
re-writes history
only do with private repository!
data:image/s3,"s3://crabby-images/d57e0/d57e0c2df27d129e463f572f76dbf991b9bdf2ac" alt=""
How do we reorder commits?
Hint: git rebase -i commit
data:image/s3,"s3://crabby-images/6fbe0/6fbe049f4ccc55553ae1428c3dfeb9ed7395e916" alt=""
git rebase -i commit
move up in VScode
data:image/s3,"s3://crabby-images/1e141/1e1418d7bfdfe66bedbf09ae3c19813f0bd984eb" alt=""
How do we squash commits?
data:image/s3,"s3://crabby-images/8ad1a/8ad1ad0e28660f3318ff268d09b3c7d468660f0e" alt=""
Hint: git rebase -i
combine commits
data:image/s3,"s3://crabby-images/4c07b/4c07bd73a0c957c3af01855000943159b512ab9e" alt=""
How can we split a commit?
data:image/s3,"s3://crabby-images/2ca1e/2ca1e2991f42055037b11467d189776fb7eef1ee" alt=""
git rebase -i commitParent
opens VS code, click edit, close
brings edit snapshot to working dir / staging dir
git reset –soft HEAD~1
moves head to one step before target commit
changes in staging area, can unstage some, make separate commits
git reset –mixed HEAD^
data:image/s3,"s3://crabby-images/d85b7/d85b72b73a7cf3279fde64fbbcaf25379f317e1f" alt=""
How can we re-write history in gitkracken?
data:image/s3,"s3://crabby-images/fefb7/fefb789f5b3329a39426f31f0c8b55f32229251b" alt=""