8a7641284c
Similar to the previous patch, when git blame fails it usually means that the blamed file is not tracked by Git. I don't know of other common failure scenarios. Since git blame runs in detached process, failure handling is a bit more involved. Let's forward stderr to the debug buffer and escalate the error. I'm not sure how to get the exit code from git blame in "git blame | perl" - we can't use $PIPESTATUS. As a workaround, interpret empty output as failure. In case of failure, also hide blame; this makes the UI more consistent I think.
787 lines
31 KiB
Plaintext
787 lines
31 KiB
Plaintext
declare-option -docstring "name of the client in which documentation is to be displayed" \
|
|
str docsclient
|
|
|
|
declare-option -docstring "git diff added character" \
|
|
str git_diff_add_char "▏"
|
|
|
|
declare-option -docstring "git diff modified character" \
|
|
str git_diff_mod_char "▏"
|
|
|
|
declare-option -docstring "git diff deleted character" \
|
|
str git_diff_del_char "_"
|
|
|
|
declare-option -docstring "git diff top deleted character" \
|
|
str git_diff_top_char "‾"
|
|
|
|
hook -group git-log-highlight global WinSetOption filetype=git-log %{
|
|
add-highlighter window/git-log group
|
|
add-highlighter window/git-log/ regex '^([*|\\ /_.-])*' 0:keyword
|
|
add-highlighter window/git-log/ regex '^( ?[*|\\ /_.-])*\h{,3}(commit )?(\b[0-9a-f]{4,40}\b)' 2:keyword 3:comment
|
|
add-highlighter window/git-log/ regex '^( ?[*|\\ /_.-])*\h{,3}([a-zA-Z_-]+:) (.*?)$' 2:variable 3:value
|
|
hook -once -always window WinSetOption filetype=.* %{ remove-highlighter window/git-log }
|
|
}
|
|
|
|
hook global WinSetOption filetype=diff %{
|
|
try %{
|
|
execute-keys -draft %{/^diff --git\b<ret>}
|
|
evaluate-commands %sh{
|
|
if [ -n "$(git ls-files -- "${kak_buffile}")" ]; then
|
|
echo fail
|
|
fi
|
|
}
|
|
set-option buffer filetype git-diff
|
|
}
|
|
}
|
|
|
|
hook -group git-diff-highlight global WinSetOption filetype=(git-diff|git-log) %{
|
|
require-module diff
|
|
add-highlighter %exp{window/%val{hook_param_capture_1}-ref-diff} ref diff
|
|
hook -once -always window WinSetOption filetype=.* %exp{
|
|
remove-highlighter window/%val{hook_param_capture_1}-ref-diff
|
|
}
|
|
}
|
|
|
|
hook global WinSetOption filetype=(?:git-diff|git-log) %{
|
|
map buffer normal <ret> %exp{:git-diff-goto-source # %val{hook_param}<ret>} -docstring 'Jump to source from git diff'
|
|
hook -once -always window WinSetOption filetype=.* %exp{
|
|
unmap buffer normal <ret> %%{:git-diff-goto-source # %val{hook_param}<ret>}
|
|
}
|
|
}
|
|
|
|
hook -group git-status-highlight global WinSetOption filetype=git-status %{
|
|
add-highlighter window/git-status group
|
|
add-highlighter window/git-status/ regex '^## ' 0:comment
|
|
add-highlighter window/git-status/ regex '^## (\S*[^\s\.@])' 1:green
|
|
add-highlighter window/git-status/ regex '^## (\S*[^\s\.@])(\.\.+)(\S*[^\s\.@])' 1:green 2:comment 3:red
|
|
add-highlighter window/git-status/ regex '^(##) (No commits yet on) (\S*[^\s\.@])' 1:comment 2:Default 3:green
|
|
add-highlighter window/git-status/ regex '^## \S+ \[[^\n]*ahead (\d+)[^\n]*\]' 1:green
|
|
add-highlighter window/git-status/ regex '^## \S+ \[[^\n]*behind (\d+)[^\n]*\]' 1:red
|
|
add-highlighter window/git-status/ regex '^(?:([Aa])|([Cc])|([Dd!?])|([MUmu])|([Rr])|([Tt]))[ !\?ACDMRTUacdmrtu]\h' 1:green 2:blue 3:red 4:yellow 5:cyan 6:cyan
|
|
add-highlighter window/git-status/ regex '^[ !\?ACDMRTUacdmrtu](?:([Aa])|([Cc])|([Dd!?])|([MUmu])|([Rr])|([Tt]))\h' 1:green 2:blue 3:red 4:yellow 5:cyan 6:cyan
|
|
add-highlighter window/git-status/ regex '^R[ !\?ACDMRTUacdmrtu] [^\n]+( -> )' 1:cyan
|
|
add-highlighter window/git-status/ regex '^\h+(?:((?:both )?modified:)|(added:|new file:)|(deleted(?: by \w+)?:)|(renamed:)|(copied:))(?:.*?)$' 1:yellow 2:green 3:red 4:cyan 5:blue 6:magenta
|
|
|
|
hook -once -always window WinSetOption filetype=.* %{ remove-highlighter window/git-status }
|
|
}
|
|
|
|
hook -group git-show-branch-highlight global WinSetOption filetype=git-show-branch %{
|
|
add-highlighter window/git-show-branch group
|
|
add-highlighter window/git-show-branch/ regex '(\*)|(\+)|(!)' 1:red 2:green 3:green
|
|
add-highlighter window/git-show-branch/ regex '(!\D+\{0\}\])|(!\D+\{1\}\])|(!\D+\{2\}\])|(!\D+\{3\}\])' 1:red 2:green 3:yellow 4:blue
|
|
add-highlighter window/git-show-branch/ regex '(\B\+\D+\{0\}\])|(\B\+\D+\{1\}\])|(\B\+\D+\{2\}\])|(\B\+\D+\{3\}\])|(\B\+\D+\{1\}\^\])' 1:red 2:green 3:yellow 4:blue 5:magenta
|
|
|
|
hook -once -always window WinSetOption filetype=.* %{ remove-highlighter window/git-show-branch}
|
|
}
|
|
|
|
declare-option -hidden line-specs git_blame_flags
|
|
declare-option -hidden line-specs git_blame_index
|
|
declare-option -hidden str git_blame
|
|
declare-option -hidden str git_blob
|
|
declare-option -hidden line-specs git_diff_flags
|
|
declare-option -hidden int-list git_hunk_list
|
|
|
|
define-command -params 1.. \
|
|
-docstring %{
|
|
git [<arguments>]: git wrapping helper
|
|
All the optional arguments are forwarded to the git utility
|
|
Available commands:
|
|
add
|
|
apply - alias for "patch git apply"
|
|
blame - toggle blame annotations
|
|
blame-jump - show the commit that added the line at cursor
|
|
checkout
|
|
commit
|
|
diff
|
|
edit
|
|
grep
|
|
hide-diff
|
|
init
|
|
log
|
|
next-hunk
|
|
prev-hunk
|
|
reset
|
|
rm
|
|
show
|
|
show-branch
|
|
show-diff
|
|
status
|
|
update-diff
|
|
} -shell-script-candidates %{
|
|
if [ $kak_token_to_complete -eq 0 ]; then
|
|
printf %s\\n \
|
|
apply \
|
|
blame \
|
|
blame-jump \
|
|
checkout \
|
|
commit \
|
|
diff \
|
|
edit \
|
|
grep \
|
|
hide-diff \
|
|
init \
|
|
log \
|
|
next-hunk \
|
|
prev-hunk \
|
|
reset \
|
|
rm \
|
|
show \
|
|
show-branch \
|
|
show-diff \
|
|
status \
|
|
update-diff \
|
|
;
|
|
else
|
|
case "$1" in
|
|
commit) printf -- "--amend\n--no-edit\n--all\n--reset-author\n--fixup\n--squash\n"; git ls-files -m ;;
|
|
add) git ls-files -dmo --exclude-standard ;;
|
|
apply) printf -- "--reverse\n--cached\n--index\n--3way\n" ;;
|
|
grep|edit) git ls-files -c --recurse-submodules ;;
|
|
esac
|
|
fi
|
|
} \
|
|
git %{ evaluate-commands %sh{
|
|
cd_bufdir() {
|
|
dirname_buffer="${kak_buffile%/*}"
|
|
cd "${dirname_buffer}" 2>/dev/null || {
|
|
printf 'fail Unable to change the current working directory to: %s\n' "${dirname_buffer}"
|
|
exit 1
|
|
}
|
|
}
|
|
kakquote() {
|
|
printf "%s" "$1" | sed "s/'/''/g; 1s/^/'/; \$s/\$/'/"
|
|
}
|
|
|
|
show_git_cmd_output() {
|
|
local filetype
|
|
|
|
case "$1" in
|
|
diff) filetype=git-diff ;;
|
|
show) filetype=git-log ;;
|
|
show-branch) filetype=git-show-branch ;;
|
|
log) filetype=git-log ;;
|
|
status) filetype=git-status ;;
|
|
*) return 1 ;;
|
|
esac
|
|
output=$(mktemp -d "${TMPDIR:-/tmp}"/kak-git.XXXXXXXX)/fifo
|
|
mkfifo ${output}
|
|
( git "$@" > ${output} 2>&1 & ) > /dev/null 2>&1 < /dev/null
|
|
|
|
printf %s "evaluate-commands -try-client '$kak_opt_docsclient' '
|
|
edit! -fifo ${output} *git*
|
|
set-option buffer filetype ${filetype}
|
|
$(hide_blame)
|
|
set-option buffer git_blob %{}
|
|
hook -always -once buffer BufCloseFifo .* ''
|
|
nop %sh{ rm -r $(dirname ${output}) }
|
|
$(printf %s "${on_close_fifo}" | sed "s/'/''''/g")
|
|
''
|
|
'"
|
|
}
|
|
|
|
hide_blame() {
|
|
printf %s "
|
|
set-option buffer git_blame_flags $kak_timestamp
|
|
set-option buffer git_blame_index $kak_timestamp
|
|
set-option buffer git_blame %{}
|
|
remove-highlighter window/git-blame
|
|
unmap window normal <ret> %{:git blame-jump<ret>}
|
|
"
|
|
}
|
|
|
|
prepare_git_blame_args='
|
|
if [ -n "${kak_opt_git_blob}" ]; then {
|
|
contents_fifo=/dev/null
|
|
set -- "$@" "${kak_opt_git_blob%%:*}" -- "${kak_opt_git_blob#*:}"
|
|
} else {
|
|
contents_fifo=$(mktemp -d "${TMPDIR:-/tmp}"/kak-git.XXXXXXXX)/fifo
|
|
mkfifo ${contents_fifo}
|
|
echo >${kak_command_fifo} "evaluate-commands -save-regs | %{
|
|
set-register | %{
|
|
contents=\$(cat; printf .)
|
|
( printf %s \"\${contents%.}\" >${contents_fifo} ) >/dev/null 2>&1 &
|
|
}
|
|
execute-keys -client ${kak_client} -draft %{%<a-|><ret>}
|
|
}"
|
|
set -- "$@" --contents - -- "${kak_buffile}"
|
|
} fi
|
|
'
|
|
|
|
blame_toggle() {
|
|
echo >${kak_command_fifo} "try %{
|
|
add-highlighter window/git-blame flag-lines Information git_blame_flags
|
|
echo -to-file ${kak_response_fifo}
|
|
} catch %{
|
|
echo -to-file ${kak_response_fifo} 'hide_blame; exit'
|
|
}"
|
|
eval $(cat ${kak_response_fifo})
|
|
if [ -z "${kak_opt_git_blob}" ] && {
|
|
[ "${kak_opt_filetype}" = git-diff ] || [ "${kak_opt_filetype}" = git-log ]
|
|
} then {
|
|
echo 'remove-highlighter window/git-blame'
|
|
printf >${kak_command_fifo} %s '
|
|
evaluate-commands -client '${kak_client}' -draft %{
|
|
try %{
|
|
execute-keys <a-l><semicolon><a-?>^commit<ret><a-semicolon>
|
|
} catch %{
|
|
# Missing commit line, assume it is an uncommitted change.
|
|
execute-keys <a-l><semicolon><a-?>\A<ret><a-semicolon>
|
|
}
|
|
require-module diff
|
|
try %{
|
|
diff-parse END %{
|
|
my $line = $file_line;
|
|
if (not defined $commit) {
|
|
$commit = "HEAD";
|
|
$line = $other_file_line;
|
|
if ($diff_line_text =~ m{^\+}) {
|
|
print "echo -to-file '${kak_response_fifo}' -quoting shell "
|
|
. "%{git blame: blame from HEAD does not work on added lines}";
|
|
exit;
|
|
}
|
|
} elsif ($diff_line_text =~ m{^[-]}) {
|
|
$commit = "$commit~";
|
|
$line = $other_file_line;
|
|
}
|
|
$line = $line or 1;
|
|
printf "echo -to-file '${kak_response_fifo}' -quoting shell %s %s %d %d",
|
|
$commit, quote($file), $line, ('${kak_cursor_column}' - 1);
|
|
}
|
|
} catch %{
|
|
echo -to-file '${kak_response_fifo}' -quoting shell -- %val{error}
|
|
}
|
|
}
|
|
'
|
|
n=$#
|
|
eval set -- "$(cat ${kak_response_fifo})" "$@"
|
|
if [ $# -eq $((n+1)) ]; then
|
|
echo fail -- "$(kakquote "$1")"
|
|
exit
|
|
fi
|
|
commit=$1
|
|
file=${2#"$PWD/"}
|
|
cursor_line=$3
|
|
cursor_column=$4
|
|
shift 4
|
|
# Log commit and file name because they are only echoed briefly
|
|
# and not shown elsewhere (we don't have a :messages buffer).
|
|
message="Blaming $file as of $(git rev-parse --short $commit)"
|
|
echo "echo -debug -- $(kakquote "$message")"
|
|
on_close_fifo="
|
|
execute-keys -client ${kak_client} ${cursor_line}g<a-h>${cursor_column}lh
|
|
evaluate-commands -client ${kak_client} %{
|
|
set-option buffer git_blob $(kakquote "$commit:$file")
|
|
git blame $(for arg; do kakquote "$arg"; printf " "; done)
|
|
hook -once window NormalIdle .* %{
|
|
execute-keys vv
|
|
echo -markup -- $(kakquote "{Information}{\\}$message. Press <ret> to jump to blamed commit")
|
|
}
|
|
}
|
|
" show_git_cmd_output show "$commit:$file"
|
|
exit
|
|
} fi
|
|
eval "$prepare_git_blame_args"
|
|
echo 'map window normal <ret> %{:git blame-jump<ret>}'
|
|
echo 'echo -markup {Information}Press <ret> to jump to blamed commit'
|
|
(
|
|
cd_bufdir
|
|
printf %s "evaluate-commands -client '$kak_client' %{
|
|
set-option buffer=$kak_bufname git_blame_flags '$kak_timestamp'
|
|
set-option buffer=$kak_bufname git_blame_index '$kak_timestamp'
|
|
set-option buffer=$kak_bufname git_blame ''
|
|
}" | kak -p ${kak_session}
|
|
if ! stderr=$({ git blame --incremental "$@" <${contents_fifo} | perl -wne '
|
|
use POSIX qw(strftime);
|
|
sub quote {
|
|
my $SQ = "'\''";
|
|
my $token = shift;
|
|
$token =~ s/$SQ/$SQ$SQ/g;
|
|
return "$SQ$token$SQ";
|
|
}
|
|
sub send_flags {
|
|
my $is_last_call = shift;
|
|
if (not defined $line) {
|
|
if ($is_last_call) { exit 1; }
|
|
return;
|
|
}
|
|
my $text = substr($sha,0,7) . " " . $dates{$sha} . " " . $authors{$sha};
|
|
$text =~ s/~/~~/g;
|
|
for ( my $i = 0; $i < $count; $i++ ) {
|
|
$flags .= " %~" . ($line+$i) . "|$text~";
|
|
}
|
|
$now = time();
|
|
# Send roughly one update per second, to avoid creating too many kak processes.
|
|
if (!$is_last_call && defined $last_sent && $now - $last_sent < 1) {
|
|
return
|
|
}
|
|
open CMD, "|-", "kak -p $ENV{kak_session}";
|
|
print CMD "set-option -add buffer=$ENV{kak_bufname} git_blame_flags $flags;";
|
|
print CMD "set-option -add buffer=$ENV{kak_bufname} git_blame_index $index;";
|
|
print CMD "set-option -add buffer=$ENV{kak_bufname} git_blame " . quote $raw_blame;
|
|
close(CMD);
|
|
$flags = "";
|
|
$index = "";
|
|
$raw_blame = "";
|
|
$last_sent = $now;
|
|
}
|
|
$raw_blame .= $_;
|
|
chomp;
|
|
if (m/^([0-9a-f]+) ([0-9]+) ([0-9]+) ([0-9]+)/) {
|
|
send_flags(0);
|
|
$sha = $1;
|
|
$line = $3;
|
|
$count = $4;
|
|
for ( my $i = 0; $i < $count; $i++ ) {
|
|
$index .= " " . ($line+$i) . "|$.,$i";
|
|
}
|
|
}
|
|
if (m/^author /) {
|
|
$authors{$sha} = substr($_,7);
|
|
$authors{$sha} = "Not Committed Yet" if $authors{$sha} eq "External file (--contents)";
|
|
}
|
|
if (m/^author-time ([0-9]*)/) { $dates{$sha} = strftime("%F %T", localtime $1) }
|
|
END { send_flags(1); }'
|
|
} 2>&1); then
|
|
escape2() { printf %s "$*" | sed "s/'/''''/g"; }
|
|
echo "evaluate-commands -client ${kak_client} '
|
|
evaluate-commands -draft %{
|
|
buffer %{${kak_buffile}}
|
|
git hide-blame
|
|
}
|
|
echo -debug failed to run git blame
|
|
echo -debug git stderr: <<<
|
|
echo -debug ''$(escape2 "$stderr")>>>''
|
|
hook -once buffer NormalIdle .* %{
|
|
echo -markup %{{Error}failed to run git blame, see *debug* buffer}
|
|
}
|
|
'" | kak -p ${kak_session}
|
|
fi
|
|
if [ "$contents_fifo" != /dev/null ]; then
|
|
rm -r $(dirname $contents_fifo)
|
|
fi
|
|
) > /dev/null 2>&1 < /dev/null &
|
|
}
|
|
|
|
run_git_cmd() {
|
|
if git "${@}" > /dev/null 2>&1; then
|
|
printf %s "echo -markup '{Information}git $1 succeeded'"
|
|
else
|
|
printf 'fail git %s failed\n' "$1"
|
|
fi
|
|
}
|
|
|
|
update_diff() {
|
|
(
|
|
cd_bufdir
|
|
git --no-pager diff --no-ext-diff -U0 "$kak_buffile" | perl -e '
|
|
use utf8;
|
|
$flags = $ENV{"kak_timestamp"};
|
|
$add_char = $ENV{"kak_opt_git_diff_add_char"};
|
|
$del_char = $ENV{"kak_opt_git_diff_del_char"};
|
|
$top_char = $ENV{"kak_opt_git_diff_top_char"};
|
|
$mod_char = $ENV{"kak_opt_git_diff_mod_char"};
|
|
foreach $line (<STDIN>) {
|
|
if ($line =~ /@@ -(\d+)(?:,(\d+))? \+(\d+)(?:,(\d+))?/) {
|
|
$from_line = $1;
|
|
$from_count = ($2 eq "" ? 1 : $2);
|
|
$to_line = $3;
|
|
$to_count = ($4 eq "" ? 1 : $4);
|
|
|
|
if ($from_count == 0 and $to_count > 0) {
|
|
for $i (0..$to_count - 1) {
|
|
$line = $to_line + $i;
|
|
$flags .= " $line|\{green\}$add_char";
|
|
}
|
|
}
|
|
elsif ($from_count > 0 and $to_count == 0) {
|
|
if ($to_line == 0) {
|
|
$flags .= " 1|\{red\}$top_char";
|
|
} else {
|
|
$flags .= " $to_line|\{red\}$del_char";
|
|
}
|
|
}
|
|
elsif ($from_count > 0 and $from_count == $to_count) {
|
|
for $i (0..$to_count - 1) {
|
|
$line = $to_line + $i;
|
|
$flags .= " $line|\{blue\}$mod_char";
|
|
}
|
|
}
|
|
elsif ($from_count > 0 and $from_count < $to_count) {
|
|
for $i (0..$from_count - 1) {
|
|
$line = $to_line + $i;
|
|
$flags .= " $line|\{blue\}$mod_char";
|
|
}
|
|
for $i ($from_count..$to_count - 1) {
|
|
$line = $to_line + $i;
|
|
$flags .= " $line|\{green\}$add_char";
|
|
}
|
|
}
|
|
elsif ($to_count > 0 and $from_count > $to_count) {
|
|
for $i (0..$to_count - 2) {
|
|
$line = $to_line + $i;
|
|
$flags .= " $line|\{blue\}$mod_char";
|
|
}
|
|
$last = $to_line + $to_count - 1;
|
|
$flags .= " $last|\{blue+u\}$mod_char";
|
|
}
|
|
}
|
|
}
|
|
print "set-option buffer git_diff_flags $flags"
|
|
' )
|
|
}
|
|
|
|
jump_hunk() {
|
|
direction=$1
|
|
set -- ${kak_opt_git_diff_flags}
|
|
shift
|
|
|
|
if [ $# -lt 1 ]; then
|
|
echo "fail 'no git hunks found, try \":git show-diff\" first'"
|
|
exit
|
|
fi
|
|
|
|
# Update hunk list if required
|
|
if [ "$kak_timestamp" != "${kak_opt_git_hunk_list%% *}" ]; then
|
|
hunks=$kak_timestamp
|
|
|
|
prev_line="-1"
|
|
for line in "$@"; do
|
|
line="${line%%|*}"
|
|
if [ "$((line - prev_line))" -gt 1 ]; then
|
|
hunks="$hunks $line"
|
|
fi
|
|
prev_line="$line"
|
|
done
|
|
echo "set-option buffer git_hunk_list $hunks"
|
|
hunks=${hunks#* }
|
|
else
|
|
hunks=${kak_opt_git_hunk_list#* }
|
|
fi
|
|
|
|
prev_hunk=""
|
|
next_hunk=""
|
|
for hunk in ${hunks}; do
|
|
if [ "$hunk" -lt "$kak_cursor_line" ]; then
|
|
prev_hunk=$hunk
|
|
elif [ "$hunk" -gt "$kak_cursor_line" ]; then
|
|
next_hunk=$hunk
|
|
break
|
|
fi
|
|
done
|
|
|
|
wrapped=false
|
|
if [ "$direction" = "next" ]; then
|
|
if [ -z "$next_hunk" ]; then
|
|
next_hunk=${hunks%% *}
|
|
wrapped=true
|
|
fi
|
|
if [ -n "$next_hunk" ]; then
|
|
echo "select $next_hunk.1,$next_hunk.1"
|
|
fi
|
|
elif [ "$direction" = "prev" ]; then
|
|
if [ -z "$prev_hunk" ]; then
|
|
wrapped=true
|
|
prev_hunk=${hunks##* }
|
|
fi
|
|
if [ -n "$prev_hunk" ]; then
|
|
echo "select $prev_hunk.1,$prev_hunk.1"
|
|
fi
|
|
fi
|
|
|
|
if [ "$wrapped" = true ]; then
|
|
echo "echo -markup '{Information}git hunk search wrapped around buffer'"
|
|
fi
|
|
}
|
|
|
|
commit() {
|
|
# Handle case where message needs not to be edited
|
|
if grep -E -q -e "-m|-F|-C|--message=.*|--file=.*|--reuse-message=.*|--no-edit|--fixup.*|--squash.*"; then
|
|
if git commit "$@" > /dev/null 2>&1; then
|
|
echo 'echo -markup "{Information}Commit succeeded"'
|
|
else
|
|
echo 'fail Commit failed'
|
|
fi
|
|
exit
|
|
fi <<-EOF
|
|
$@
|
|
EOF
|
|
|
|
# fails, and generate COMMIT_EDITMSG
|
|
GIT_EDITOR='' EDITOR='' git commit "$@" > /dev/null 2>&1
|
|
msgfile="$(git rev-parse --git-dir)/COMMIT_EDITMSG"
|
|
printf %s "edit '$msgfile'
|
|
hook buffer BufWritePost '.*\Q$msgfile\E' %{ evaluate-commands %sh{
|
|
if git commit -F '$msgfile' --cleanup=strip $* > /dev/null; then
|
|
printf %s 'evaluate-commands -client $kak_client echo -markup %{{Information}Commit succeeded}; delete-buffer'
|
|
else
|
|
printf 'evaluate-commands -client %s fail Commit failed\n' "$kak_client"
|
|
fi
|
|
} }"
|
|
}
|
|
|
|
blame_jump() {
|
|
echo >${kak_command_fifo} "echo -to-file ${kak_response_fifo} -- %opt{git_blame}"
|
|
blame_info=$(cat < ${kak_response_fifo})
|
|
blame_index=
|
|
cursor_column=${kak_cursor_column}
|
|
cursor_line=${kak_cursor_line}
|
|
if [ -n "$blame_info" ]; then {
|
|
echo >${kak_command_fifo} "
|
|
update-option buffer git_blame_index
|
|
echo -to-file ${kak_response_fifo} -- %opt{git_blame_index}
|
|
"
|
|
blame_index=$(cat < ${kak_response_fifo})
|
|
} elif [ "${kak_opt_filetype}" = git-diff ] || [ "${kak_opt_filetype}" = git-log ]; then {
|
|
printf >${kak_command_fifo} %s '
|
|
evaluate-commands -draft %{
|
|
try %{
|
|
execute-keys <a-l><semicolon><a-?>^commit<ret><a-semicolon>
|
|
} catch %{
|
|
# Missing commit line, assume it is an uncommitted change.
|
|
execute-keys <a-l><semicolon><a-?>\A<ret><a-semicolon>
|
|
}
|
|
require-module diff
|
|
try %{
|
|
diff-parse BEGIN %{
|
|
$version = "-";
|
|
} END %{
|
|
if ($diff_line_text !~ m{^[ -]}) {
|
|
print "set-register e fail git blame-jump: recursive blame only works on context or deleted lines";
|
|
} else {
|
|
if (not defined $commit) {
|
|
$commit = "HEAD";
|
|
} else {
|
|
$commit = "$commit~" if $diff_line_text =~ m{^[- ]};
|
|
}
|
|
printf "echo -to-file '${kak_response_fifo}' -quoting shell %s %s %d %d",
|
|
$commit, quote($file), $file_line, ('$cursor_column' - 1);
|
|
}
|
|
}
|
|
} catch %{
|
|
echo -to-file '${kak_response_fifo}' -quoting shell -- %val{error}
|
|
}
|
|
}
|
|
'
|
|
eval set -- "$(cat ${kak_response_fifo})"
|
|
if [ $# -eq 1 ]; then
|
|
echo fail -- "$(kakquote "$1")"
|
|
exit
|
|
fi
|
|
starting_commit=$1
|
|
file=$2
|
|
cursor_line=$3
|
|
cursor_column=$4
|
|
blame_info=$(git blame --porcelain "$starting_commit" -L"$cursor_line,$cursor_line" -- "$file")
|
|
if [ $? -ne 0 ]; then
|
|
echo 'echo -markup %{{Error}failed to run git blame, see *debug* buffer}'
|
|
exit
|
|
fi
|
|
} else {
|
|
set --
|
|
eval "$prepare_git_blame_args"
|
|
blame_info=$(git blame --porcelain -L"$cursor_line,$cursor_line" "$@" <${contents_fifo})
|
|
status=$?
|
|
if [ "$contents_fifo" != /dev/null ]; then
|
|
rm -r $(dirname $contents_fifo)
|
|
fi
|
|
if [ $status -ne 0 ]; then
|
|
echo 'echo -markup %{{Error}failed to run git blame, see *debug* buffer}'
|
|
exit
|
|
fi
|
|
} fi
|
|
eval "$(printf '%s\n---\n%s' "$blame_index" "$blame_info" |
|
|
client=${kak_opt_docsclient:-$kak_client} \
|
|
cursor_line=$cursor_line cursor_column=$cursor_column \
|
|
perl -wne '
|
|
BEGIN {
|
|
use POSIX qw(strftime);
|
|
our $SQ = "'\''";
|
|
sub escape {
|
|
return shift =~ s/$SQ/$SQ$SQ/gr
|
|
}
|
|
sub quote {
|
|
my $token = escape shift;
|
|
return "$SQ$token$SQ";
|
|
}
|
|
sub shellquote {
|
|
my $token = shift;
|
|
$token =~ s/$SQ/$SQ\\$SQ$SQ/g;
|
|
return "$SQ$token$SQ";
|
|
}
|
|
sub perlquote {
|
|
my $token = shift;
|
|
$token =~ s/\\/\\\\/g;
|
|
$token =~ s/$SQ/\\$SQ/g;
|
|
return "$SQ$token$SQ";
|
|
}
|
|
$target = $ENV{"cursor_line"};
|
|
$state = "index";
|
|
}
|
|
chomp;
|
|
if ($state eq "index") {
|
|
if ($_ eq "---") {
|
|
$state = "blame";
|
|
next;
|
|
}
|
|
@blame_index = split;
|
|
next unless @blame_index;
|
|
shift @blame_index;
|
|
foreach (@blame_index) {
|
|
$_ =~ m{(\d+)\|(\d+),(\d+)} or die "bad blame index flag: $_";
|
|
my $buffer_line = $1;
|
|
if ($buffer_line == $target) {
|
|
$target_in_blame = $2;
|
|
$target_offset = $3;
|
|
last;
|
|
}
|
|
}
|
|
defined $target_in_blame and next, or last;
|
|
}
|
|
if (m/^([0-9a-f]+) ([0-9]+) ([0-9]+) ([0-9]+)/) {
|
|
if ($done) {
|
|
last;
|
|
}
|
|
$sha = $1;
|
|
$old_line = $2;
|
|
$new_line = $3;
|
|
$count = $4;
|
|
if (defined $target_in_blame) {
|
|
if ($target_in_blame == $. - 2) {
|
|
$old_line += $target_offset;
|
|
$done = 1;
|
|
}
|
|
} else {
|
|
if ($new_line <= $target and $target < $new_line + $count) {
|
|
$old_line += $target - $new_line;
|
|
$done = 1;
|
|
}
|
|
}
|
|
}
|
|
if (m/^filename /) { $old_filenames{$sha} = substr($_,9) }
|
|
if (m/^author /) { $authors{$sha} = substr($_,7) }
|
|
if (m/^author-time ([0-9]*)/) { $dates{$sha} = strftime("%F", localtime $1) }
|
|
if (m/^summary /) { $summaries{$sha} = substr($_,8) }
|
|
END {
|
|
if (@blame_index and not defined $target_in_blame) {
|
|
print "echo fail git blame-jump: line has no blame information;";
|
|
exit;
|
|
}
|
|
if (not defined $sha) {
|
|
print "echo fail git blame-jump: missing blame info";
|
|
exit;
|
|
}
|
|
if (not $done) {
|
|
print "echo \"fail git blame-jump: line not found in annotations (blame still loading?)\"";
|
|
exit;
|
|
}
|
|
$info = "{Information}{\\}";
|
|
if ($sha =~ m{^0+$}) {
|
|
$old_filename = $ENV{"kak_buffile"};
|
|
$old_filename = substr $old_filename, length($ENV{"PWD"}) + 1;
|
|
$show_diff = "diff HEAD";
|
|
$info .= "Not committed yet";
|
|
} else {
|
|
$old_filename = $old_filenames{$sha};
|
|
$author = $authors{$sha};
|
|
$date = $dates{$sha};
|
|
$summary = $summaries{$sha};
|
|
$show_diff = "show $sha";
|
|
$info .= "$date $author \"$summary\"";
|
|
}
|
|
$on_close_fifo = "
|
|
evaluate-commands -draft $SQ
|
|
execute-keys <percent>
|
|
require-module diff
|
|
diff-parse BEGIN %{
|
|
\$in_file = " . escape(perlquote($old_filename)) . ";
|
|
\$in_file_line = $old_line;
|
|
} END $SQ$SQ
|
|
print \"execute-keys -client $ENV{client} \${diff_line}g<a-h>$ENV{cursor_column}l;\";
|
|
printf \"evaluate-commands -client $ENV{client} $SQ$SQ$SQ$SQ
|
|
hook -once window NormalIdle .* $SQ$SQ$SQ$SQ$SQ$SQ$SQ$SQ
|
|
execute-keys vv
|
|
echo -markup -- %s
|
|
$SQ$SQ$SQ$SQ$SQ$SQ$SQ$SQ
|
|
$SQ$SQ$SQ$SQ ;\"," . escape(escape(perlquote(escape(escape(quote($info)))))) . ";
|
|
$SQ$SQ
|
|
$SQ
|
|
";
|
|
printf "on_close_fifo=%s show_git_cmd_output %s",
|
|
shellquote($on_close_fifo), $show_diff;
|
|
}
|
|
')"
|
|
}
|
|
|
|
case "$1" in
|
|
apply)
|
|
shift
|
|
enquoted="$(printf '"%s" ' "$@")"
|
|
echo "require-module patch"
|
|
echo "patch git apply $enquoted"
|
|
;;
|
|
show|show-branch|log|diff|status)
|
|
show_git_cmd_output "$@"
|
|
;;
|
|
blame)
|
|
shift
|
|
blame_toggle "$@"
|
|
;;
|
|
blame-jump)
|
|
blame_jump
|
|
;;
|
|
hide-blame)
|
|
hide_blame
|
|
;;
|
|
show-diff)
|
|
echo 'try %{ add-highlighter window/git-diff flag-lines Default git_diff_flags }'
|
|
update_diff
|
|
;;
|
|
hide-diff)
|
|
echo 'try %{ remove-highlighter window/git-diff }'
|
|
;;
|
|
update-diff) update_diff ;;
|
|
next-hunk) jump_hunk next ;;
|
|
prev-hunk) jump_hunk prev ;;
|
|
commit)
|
|
shift
|
|
commit "$@"
|
|
;;
|
|
init)
|
|
shift
|
|
git init "$@" > /dev/null 2>&1
|
|
;;
|
|
add|rm)
|
|
cmd="$1"
|
|
shift
|
|
run_git_cmd $cmd "${@:-"${kak_buffile}"}"
|
|
;;
|
|
reset|checkout)
|
|
run_git_cmd "$@"
|
|
;;
|
|
grep)
|
|
shift
|
|
enquoted="$(printf '"%s" ' "$@")"
|
|
printf %s "try %{
|
|
set-option current grepcmd 'git grep -n --column'
|
|
grep $enquoted
|
|
set-option current grepcmd '$kak_opt_grepcmd'
|
|
}"
|
|
;;
|
|
edit)
|
|
shift
|
|
enquoted="$(printf '"%s" ' "$@")"
|
|
printf %s "edit -existing -- $enquoted"
|
|
;;
|
|
*)
|
|
printf "fail unknown git command '%s'\n" "$1"
|
|
exit
|
|
;;
|
|
esac
|
|
}}
|
|
|
|
# Works within :git diff and :git show
|
|
define-command git-diff-goto-source \
|
|
-docstring 'Navigate to source by pressing the enter key in hunks when git diff is displayed. Works within :git diff and :git show' %{
|
|
require-module diff
|
|
diff-jump %sh{ git rev-parse --show-toplevel }
|
|
}
|