350 lines
8.9 KiB
350 lines
8.9 KiB
8 years ago
|
#!/usr/bin/env bash
|
||
|
|
||
|
|
||
|
# usage info
|
||
|
usage() {
|
||
|
cat <<EOF
|
||
|
Usage: git recall [options]
|
||
|
Options:
|
||
|
-d, --date Show logs for last n days.
|
||
|
-a, --author Author name.
|
||
|
-f, --fetch fetch commits.
|
||
|
-h, --help This message.
|
||
|
-v, --version Show version.
|
||
|
-- End of options.
|
||
|
EOF
|
||
|
}
|
||
|
|
||
|
|
||
|
# Global variables
|
||
|
SINCE="1 days ago" # show logs for last day by default
|
||
|
AUTHOR=""
|
||
|
FETCH=false
|
||
|
GIT_FORMAT=""
|
||
|
GIT_LOG=""
|
||
|
COMMITS=""
|
||
|
COMMITS_UNCOL=() # commits without colors
|
||
|
LESSKEY=false
|
||
|
SED_BIN="" # Sed option to use according OS.
|
||
|
VERSION="1.1.10"
|
||
|
|
||
|
# Are we in a git repo?
|
||
|
if [[ ! -d ".git" ]] && ! git rev-parse --git-dir &>/dev/null; then
|
||
|
echo "abort: not a git repository." 1>&2
|
||
|
exit 1
|
||
|
fi
|
||
|
|
||
|
# Parse options
|
||
|
while [[ "$1" =~ ^- && ! "$1" == "--" ]]; do
|
||
|
case $1 in
|
||
|
-v | --version )
|
||
|
echo "$version"
|
||
|
exit
|
||
|
;;
|
||
|
-d | --date )
|
||
|
SINCE="$2 days ago"
|
||
|
shift;
|
||
|
;;
|
||
|
-a | --author )
|
||
|
AUTHOR="$2"
|
||
|
shift
|
||
|
;;
|
||
|
-f | --fetch )
|
||
|
FETCH=true
|
||
|
;;
|
||
|
-h | --help )
|
||
|
usage
|
||
|
exit
|
||
|
;;
|
||
|
* )
|
||
|
echo "abort: unknown argument" 1>&2
|
||
|
exit 1
|
||
|
esac
|
||
|
shift
|
||
|
done
|
||
|
if [[ "$1" == "--" ]]; then shift; fi
|
||
|
|
||
|
|
||
|
# Colored output.
|
||
|
function colored() {
|
||
|
GREEN=$(tput setaf 4)
|
||
|
YELLOW=$(tput setaf 3)
|
||
|
NORMAL=$(tput sgr0)
|
||
|
REVERSE=$(tput rev)
|
||
|
}
|
||
|
|
||
|
# Uncolored output.
|
||
|
function uncolored() {
|
||
|
GREEN=""
|
||
|
YELLOW=""
|
||
|
NORMAL=""
|
||
|
REVERSE=""
|
||
|
}
|
||
|
|
||
|
# Enable colors if supported by terminal.
|
||
|
if [[ -t 1 ]] && [[ -n "$TERM" ]] && which tput &>/dev/null && tput colors &>/dev/null; then
|
||
|
ncolors=$(tput colors)
|
||
|
if [[ -n "$ncolors" ]] && [[ "$ncolors" -ge 8 ]] ; then
|
||
|
colored
|
||
|
else
|
||
|
uncolored
|
||
|
fi
|
||
|
else
|
||
|
uncolored
|
||
|
fi
|
||
|
|
||
|
# Check if lesskey is installed.
|
||
|
if command -v lesskey &> /dev/null; then
|
||
|
LESSKEY=true
|
||
|
fi
|
||
|
|
||
|
# Set AUTHOR to current user if no param passed or display for all users if param equal to "all".
|
||
|
if [[ ! -n $AUTHOR ]]; then
|
||
|
AUTHOR=$(git config user.name 2>/dev/null)
|
||
|
elif [[ $AUTHOR = "all" ]]; then
|
||
|
AUTHOR=".*"
|
||
|
fi
|
||
|
|
||
|
# Fetch changes before.
|
||
|
if [[ $FETCH == true ]]; then
|
||
|
echo "${GREEN}Fetching changes...${NORMAL}"
|
||
|
git fetch --all &> /dev/null
|
||
|
tput cuu1
|
||
|
tput ed # clear screen
|
||
|
fi
|
||
|
|
||
|
# Log template.
|
||
|
GIT_FORMAT="%Cred%h%Creset -%C(yellow)%d%Creset %s %Cgreen(%cr) %C(bold blue)<%an>%Creset"
|
||
|
|
||
|
# Log command.
|
||
|
GIT_LOG="git log --pretty=format:'${GIT_FORMAT}'
|
||
|
--author \"$AUTHOR\"
|
||
|
--since \"$SINCE\" --abbrev-commit"
|
||
|
|
||
|
# Change temporary the IFS to store GIT_LOG's output into an array.
|
||
|
IFS=$'\n'
|
||
|
COMMITS=($(eval ${GIT_LOG} 2>/dev/null))
|
||
|
unset IFS
|
||
|
|
||
|
NI=${#COMMITS[@]} # Total number of items.
|
||
|
SN=$(( `tput lines` - 1 )) # Screen's number of lines.
|
||
|
CN=$(tput cols) # Screen's number of columns.
|
||
|
TN=$(( $NI < $((SN -1)) ? $NI : $((SN -1)))) # Number of lines that we will display.
|
||
|
OFFSET=0 #Incremented by one each time a commit's length is higher than teminal width.
|
||
|
|
||
|
# If there is no items, exit.
|
||
|
if [[ $NI = 0 ]]; then
|
||
|
if [[ $AUTHOR = ".*" ]]; then
|
||
|
echo "${YELLOW}All contributors did nothing during this period.${NORMAL}" && exit 0
|
||
|
else
|
||
|
echo "${YELLOW}The contributor \"${AUTHOR}\" did nothing during this period.${NORMAL}" && exit 0
|
||
|
fi
|
||
|
fi
|
||
|
|
||
|
# Set correct sed option according OS's type
|
||
|
case "$OSTYPE" in
|
||
|
darwin*) SED_BIN="sed -E" ;;
|
||
|
*) SED_BIN="sed -r" ;;
|
||
|
esac
|
||
|
|
||
|
# Create array with uncolred commits (removing escape sequences using sed)
|
||
|
for elt in "${COMMITS[@]}"
|
||
|
do
|
||
|
ELT="$(echo "$elt" | $SED_BIN "s/\x1B\[([0-9]{1,2}(;[0-9]{1,2})?)?[m|K]//g")" # remove colors escape codes
|
||
|
COMMITS_UNCOL+=("$ELT")
|
||
|
done
|
||
|
# Add +1 to OFFSET if a commit's length is bigger than the current terminal session's width. (This is to fix a redraw issue)
|
||
|
for C in "${COMMITS_UNCOL[@]}"
|
||
|
do
|
||
|
if [[ ${#C} -gt $CN ]]; then
|
||
|
OFFSET=$(( OFFSET + 1 ))
|
||
|
fi
|
||
|
done
|
||
|
|
||
|
# Set keys.
|
||
|
au="`echo -e '\e[A'`" # arrow up
|
||
|
au_1="k" # arrow up
|
||
|
ad="`echo -e '\e[B'`" # arrow down
|
||
|
ad_1="j" # arrow down
|
||
|
ec="`echo -e '\e'`" # escape
|
||
|
nl="`echo -e '\n'`" # newline
|
||
|
nl_1="e" # expand
|
||
|
co="c" # checkout
|
||
|
|
||
|
# Create a temporary lesskey file to change the keybindings so the user can use the TAB key to quit less. (more convenient)
|
||
|
if [[ $LESSKEY = true ]]; then
|
||
|
echo "\t quit" | lesskey -o $HOME/.lsh_less_keys_tmp -- - &> /dev/null
|
||
|
fi
|
||
|
|
||
|
## Get commit's diff.
|
||
|
function get_diff() {
|
||
|
ELT="$(echo "${COMMITS_UNCOL[$CP-1]}")"
|
||
|
DIFF_TIP=${ELT:0:7}
|
||
|
DIFF_CMD="git show $DIFF_TIP --color=always"
|
||
|
DIFF=$(eval ${DIFF_CMD} 2>/dev/null)
|
||
|
tmp_diff="$(echo "$DIFF" | $SED_BIN "s/\x1B\[([0-9]{1,2}(;[0-9]{1,2})?)?[m|K]//g")" # remove colors escape codes
|
||
|
off=$(echo "$tmp_diff" | grep -c ".\{$CN\}") #Number of lines in the diff that are longer than terminal width.
|
||
|
DIFF_LINES_NUMBER="$(echo "$DIFF" | wc -l)"
|
||
|
DIFF_LINES_NUMBER=$(( DIFF_LINES_NUMBER + off ))
|
||
|
}
|
||
|
|
||
|
## This function will print the diff according the commit's tip. If the diff is too long, the result will be displayed using 'less'.
|
||
|
function print_diff() {
|
||
|
get_diff # get commit's diff
|
||
|
if [[ $(( TN + DIFF_LINES_NUMBER + OFFSET )) -ge $(( `tput lines` - 1 )) ]]; then
|
||
|
if [[ $LESSKEY = true ]]; then
|
||
|
echo "$DIFF" | less -r -k $HOME/.lsh_less_keys_tmp
|
||
|
else
|
||
|
echo "$DIFF" | less -r
|
||
|
fi
|
||
|
tput ed # Clear screen
|
||
|
else
|
||
|
stop=false
|
||
|
tput ed
|
||
|
for i in `seq 1 $TN`
|
||
|
do
|
||
|
echo -n "$NORMAL"
|
||
|
[[ $CP == "$i" ]] && echo -n "$REVERSE"
|
||
|
echo "${COMMITS[$i - 1]}"
|
||
|
[[ $CP == "$i" ]] && echo "$DIFF"
|
||
|
done
|
||
|
# Wait for user action.
|
||
|
while ! $stop
|
||
|
do
|
||
|
read -sn 1 key
|
||
|
case "$key" in
|
||
|
"$nl" | "$nl_1")
|
||
|
stop=true
|
||
|
;;
|
||
|
"q")
|
||
|
stop=true
|
||
|
END=true
|
||
|
;;
|
||
|
esac
|
||
|
done
|
||
|
[[ $END = false ]] && tput cuu $(( TN + DIFF_LINES_NUMBER + OFFSET )) && tput ed
|
||
|
[[ $END = true ]] && tput cuu 1
|
||
|
fi
|
||
|
}
|
||
|
|
||
|
function do_checkout(){
|
||
|
ELT="$(echo "${COMMITS_UNCOL[$CP-1]}")"
|
||
|
DIFF_TIP=${ELT:0:7}
|
||
|
eval "git checkout $DIFF_TIP 2> /dev/null"
|
||
|
}
|
||
|
|
||
|
# Calculate OFFSET to avoid bad redraw.
|
||
|
function calculate_offset {
|
||
|
tmp=1
|
||
|
index=$(( SI -1 ))
|
||
|
while [[ $tmp -lt $SN ]]
|
||
|
do
|
||
|
el=${COMMITS_UNCOL[$index]}
|
||
|
if [[ ${#el} -gt $CN ]] && [[ $CP -lt $((SN -1)) ]]; then
|
||
|
OFFSET_2=$(( OFFSET_2 + 1 ))
|
||
|
tmp=$(( tmp + 1 ))
|
||
|
fi
|
||
|
tmp=$(( tmp + 1 ))
|
||
|
index=$(( index + 1 ))
|
||
|
done
|
||
|
}
|
||
|
|
||
|
{ # capture stdout to stderr
|
||
|
|
||
|
tput civis # hide cursor.
|
||
|
CP=1 # current position
|
||
|
SI=1 # index
|
||
|
END=false # end while loop
|
||
|
EXT=0 # Used to extend the number of lines to display.
|
||
|
|
||
|
## Loops, reads inputs and prints commits until user presses 'q' to exit or TAB to show the diff.
|
||
|
while ! $END
|
||
|
do
|
||
|
END_INDEX=0 # Index for the last item to display
|
||
|
# When the number of item is higher than screen's number of lines, OFFSET_2 is recalculated each time we select a new item
|
||
|
# Set last index to print. (based on OFFSET)
|
||
|
if [[ $TN == $NI ]]; then
|
||
|
END_INDEX=$TN
|
||
|
OFFSET_2=$OFFSET
|
||
|
elif [[ $TN == $(( SN - 1 )) ]]; then
|
||
|
# Calculate new OFFSET.
|
||
|
if [[ $OFFSET != 0 ]]; then
|
||
|
[[ $CP -lt $((SN -1)) ]] && OFFSET_2=0
|
||
|
EXT=1
|
||
|
calculate_offset
|
||
|
fi
|
||
|
END_INDEX=$(( TN + SI -1 + EXT - OFFSET_2 ))
|
||
|
fi
|
||
|
|
||
|
# Loop and echo commits
|
||
|
for i in `seq $SI $END_INDEX`
|
||
|
do
|
||
|
echo -n "$NORMAL"
|
||
|
[[ $CP == $i ]] && echo -n "$REVERSE"
|
||
|
echo "${COMMITS[$i - 1]}"
|
||
|
done
|
||
|
|
||
|
read -sn 1 key
|
||
|
[[ "$key" == "$ec" ]] &&
|
||
|
{
|
||
|
read -sn 2 k2
|
||
|
key="$key$k2"
|
||
|
}
|
||
|
|
||
|
case "$key" in
|
||
|
|
||
|
"$au" | "$au_1")
|
||
|
CP=$(( CP - 1 ))
|
||
|
[[ $CP == 0 ]] && [[ $SI == 1 ]] && [[ $TN == $(( SN - 1 )) ]] && CP=$NI && SI=$(( NI - SN + 2 + OFFSET_2 ))
|
||
|
[[ $CP == 0 ]] && [[ $SI == 1 ]] && [[ $TN == $NI ]] && CP=$TN
|
||
|
[[ $CP == $(( SI - 1 )) ]] && [[ $SI != 1 ]] && SI=$(( SI - 1 ))
|
||
|
|
||
|
[[ $TN != $(( SN - 1 )) ]] && tput cuu $(( TN + OFFSET_2 ))
|
||
|
[[ $TN == $(( SN - 1 )) ]] && tput cuu $(( TN + EXT ))
|
||
|
[[ $SI != 1 ]] && tput ed # clear screen
|
||
|
;;
|
||
|
|
||
|
"$ad" | "$ad_1")
|
||
|
CP=$(( CP + 1 ))
|
||
|
[[ $CP == $(( NI + 1 )) ]] && CP=1 && SI=1
|
||
|
[[ $CP == $(( SN + SI - 1 + EXT - OFFSET_2 )) ]] && [[ $TN == $(( SN - 1 )) ]] && SI=$(( SI + 1 ))
|
||
|
|
||
|
[[ $TN != $(( SN - 1 )) ]] && tput cuu $(( TN + OFFSET_2 ))
|
||
|
[[ $TN == $(( SN - 1 )) ]] && tput cuu $(( TN + EXT ))
|
||
|
[[ $SI != 1 ]] && tput ed # clear screen
|
||
|
[[ $SI = 1 ]] && [[ $CP = 1 ]] && tput ed # clear screen
|
||
|
;;
|
||
|
|
||
|
"$nl" | "$nl_1")
|
||
|
[[ $TN == $NI ]] && tput cuu $(( TN + OFFSET_2 ))
|
||
|
[[ $TN != $NI ]] && tput cuu $(( TN + EXT ))
|
||
|
print_diff
|
||
|
;;
|
||
|
"$co")
|
||
|
si=false
|
||
|
END=true
|
||
|
do_checkout
|
||
|
tput cuu 1 #move cursor up one line. (remove extra line)
|
||
|
;;
|
||
|
|
||
|
"q")
|
||
|
si=false
|
||
|
END=true
|
||
|
tput cuu 1 #move cursor up one line. (remove extra line)
|
||
|
;;
|
||
|
|
||
|
* )
|
||
|
tput cuu $(( TN + OFFSET_2 ))
|
||
|
esac
|
||
|
|
||
|
done
|
||
|
|
||
|
# remove temporary less keybindings
|
||
|
[[ $LESSKEY = true ]] && rm $HOME/.lsh_less_keys_tmp
|
||
|
|
||
|
tput cnorm # unhide cursor
|
||
|
echo "$NORMAL" # normal colors
|
||
|
|
||
|
} >&2 # END capture
|
||
|
|