#!/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