home/rc/base/lint.kak

141 lines
5.6 KiB
Plaintext
Raw Normal View History

declare-option -docstring %{shell command to which the path of a copy of the current buffer will be passed
The output returned by this command is expected to comply with the following format:
{filename}:{line}:{column}: {kind}: {message}} \
str lintcmd
2016-10-11 18:34:11 +02:00
declare-option -hidden line-specs lint_flags
declare-option -hidden range-specs lint_errors
declare-option -hidden int lint_error_count
declare-option -hidden int lint_warning_count
2016-10-11 18:34:11 +02:00
define-command lint -docstring 'Parse the current buffer with a linter' %{
evaluate-commands %sh{
dir=$(mktemp -d "${TMPDIR:-/tmp}"/kak-lint.XXXXXXXX)
2016-10-11 18:34:11 +02:00
mkfifo "$dir"/fifo
printf '%s\n' "evaluate-commands -no-hooks write -sync $dir/buf"
2016-10-11 18:34:11 +02:00
printf '%s\n' "evaluate-commands -draft %{
edit! -fifo $dir/fifo -debug *lint-output*
set-option buffer filetype make
set-option buffer make_current_error_line 0
hook -always -group fifo buffer BufCloseFifo .* %{
2016-10-11 18:34:11 +02:00
nop %sh{ rm -r '$dir' }
2017-01-04 01:07:45 +01:00
remove-hooks buffer fifo
2016-10-11 18:34:11 +02:00
}
}"
{ # do the parsing in the background and when ready send to the session
eval "$kak_opt_lintcmd '$dir'/buf" | sort -t: -k2,2 -n > "$dir"/stderr
2016-10-11 18:34:11 +02:00
# Flags for the gutter:
# stamp l3|{red}█ l11|{yellow}█
2016-10-11 18:34:11 +02:00
# Contextual error messages:
# stamp 'l1.c1,l1.c1|kind:message' 'l2.c2,l2.c2|kind:message'
awk -F: -v file="$kak_buffile" -v stamp="$kak_timestamp" -v client="$kak_client" '
BEGIN {
error_count = 0
warning_count = 0
}
2016-10-11 18:34:11 +02:00
/:[0-9]+:[0-9]+: ([Ff]atal )?[Ee]rror/ {
flags = flags " " $2 "|{red}█"
error_count++
2016-10-11 18:34:11 +02:00
}
/:[0-9]+:[0-9]+:/ {
if ($4 !~ /[Ee]rror/) {
flags = flags " " $2 "|{yellow}█"
warning_count++
2016-10-11 18:34:11 +02:00
}
}
/:[0-9]+:[0-9]+:/ {
2018-05-24 19:36:32 +02:00
kind = substr($4, 2)
error = $2 "." $3 "," $2 "." $3 "|" kind
2018-05-24 19:36:32 +02:00
# fix case where $5 is not the last field because of extra colons in the message
for (i=5; i<=NF; i++) error = error "\\:" $i
error = error " (col " $3 ")"
gsub("'\''", "'"''"'", error)
errors = errors " '\''" error "'\''"
2016-10-11 18:34:11 +02:00
}
END {
print "set-option \"buffer=" file "\" lint_flags " stamp flags
gsub("~", "\\~", errors)
print "set-option \"buffer=" file "\" lint_errors " stamp errors
print "set-option \"buffer=" file "\" lint_error_count " error_count
print "set-option \"buffer=" file "\" lint_warning_count " warning_count
print "evaluate-commands -client " client " lint-show-counters"
2016-10-11 18:34:11 +02:00
}
' "$dir"/stderr | kak -p "$kak_session"
cut -d: -f2- "$dir"/stderr | sed "s@^@$kak_bufname:@" > "$dir"/fifo
} >/dev/null 2>&1 </dev/null &
}
}
define-command -hidden lint-show %{
update-option buffer lint_errors
evaluate-commands %sh{
desc=$(printf '%s\n' "$kak_opt_lint_errors" | sed -e 's/\([^\\]\):/\1\n/g' | tail -n +2 |
sed -ne "/^$kak_cursor_line\.[^|]\+|.*/ { s/^[^|]\+|//g; s/'/\\\\'/g; s/\\\\:/:/g; p; }")
if [ -n "$desc" ]; then
printf '%s\n' "info -anchor $kak_cursor_line.$kak_cursor_column '$desc'"
fi
} }
2016-10-11 18:34:11 +02:00
define-command -hidden lint-show-counters %{
echo -markup linting results:{red} %opt{lint_error_count} error(s){yellow} %opt{lint_warning_count} warning(s)
}
define-command lint-enable -docstring "Activate automatic diagnostics of the code" %{
2018-07-07 01:51:18 +02:00
add-highlighter window/lint flag_lines default lint_flags
2016-10-11 18:34:11 +02:00
hook window -group lint-diagnostics NormalIdle .* %{ lint-show }
hook window -group lint-diagnostics WinSetOption lint_flags=.* %{ info; lint-show }
2016-10-11 18:34:11 +02:00
}
define-command lint-disable -docstring "Disable automatic diagnostics of the code" %{
2018-07-07 01:51:18 +02:00
remove-highlighter window/lint
2017-01-04 01:07:45 +01:00
remove-hooks window lint-diagnostics
2016-10-11 18:34:11 +02:00
}
define-command lint-next-error -docstring "Jump to the next line that contains an error" %{
update-option buffer lint_errors
evaluate-commands %sh{
printf '%s\n' "$kak_opt_lint_errors" | sed -e 's/\([^\\]\):/\1\n/g' | tail -n +2 | {
while IFS='|' read -r candidate rest
do
first_range=${first_range-$candidate}
if [ "${candidate%%.*}" -gt "$kak_cursor_line" ]; then
range=$candidate
break
fi
done
range=${range-$first_range}
if [ -n "$range" ]; then
printf '%s\n' "select $range"
else
printf 'echo -markup "{Error}no lint diagnostics"\n'
2016-10-11 18:34:11 +02:00
fi
}
}}
2016-11-02 13:42:35 +01:00
define-command lint-previous-error -docstring "Jump to the previous line that contains an error" %{
update-option buffer lint_errors
evaluate-commands %sh{
printf '%s\n' "$kak_opt_lint_errors" | sed -e 's/\([^\\]\):/\1\n/g' | tail -n +2 | sort -t. -k1,1 -rn | {
while IFS='|' read -r candidate rest
do
first_range=${first_range-$candidate}
if [ "${candidate%%.*}" -lt "$kak_cursor_line" ]; then
range=$candidate
break
fi
done
range=${range-$first_range}
if [ -n "$range" ]; then
printf '%s\n' "select $range"
else
printf 'echo -markup "{Error}no lint diagnostics"\n'
2016-11-02 13:42:35 +01:00
fi
}
}}