348 lines
9.2 KiB
VimL
348 lines
9.2 KiB
VimL
" Author: w0rp <devw0rp@gmail.com>
|
|
" Description: Main entry point for this plugin
|
|
" Loads linters and manages lint jobs
|
|
|
|
if exists('g:loaded_ale_zmain')
|
|
finish
|
|
endif
|
|
|
|
let g:loaded_ale_zmain = 1
|
|
|
|
let s:lint_timer = -1
|
|
let s:linters = {}
|
|
|
|
if !exists('g:ale_linters')
|
|
let g:ale_linters = {}
|
|
endif
|
|
|
|
" Stores information for each job including:
|
|
"
|
|
" linter: The linter dictionary for the job.
|
|
" buffer: The buffer number for the job.
|
|
" output: The array of lines for the ouptut of the job.
|
|
let s:job_info_map = {}
|
|
|
|
let s:job_linter_map = {}
|
|
let s:job_buffer_map = {}
|
|
let s:job_output_map = {}
|
|
|
|
" Globals which each part of the plugin should use.
|
|
let g:ale_buffer_loclist_map = {}
|
|
let g:ale_buffer_should_reset_map = {}
|
|
|
|
function! s:GetFunction(string_or_ref)
|
|
if type(a:string_or_ref) == type('')
|
|
return function(a:string_or_ref)
|
|
endif
|
|
|
|
return a:string_or_ref
|
|
endfunction
|
|
|
|
function! s:ClearJob(job)
|
|
let linter = s:job_info_map[a:job].linter
|
|
|
|
if has('nvim')
|
|
call jobstop(a:job)
|
|
else
|
|
" We must close the channel for reading the buffer if it is open
|
|
" when stopping a job. Otherwise, we will get errors in the status line.
|
|
if ch_status(job_getchannel(a:job)) ==# 'open'
|
|
call ch_close_in(job_getchannel(a:job))
|
|
endif
|
|
|
|
call job_stop(a:job)
|
|
endif
|
|
|
|
call remove(s:job_info_map, a:job)
|
|
call remove(linter, 'job')
|
|
endfunction
|
|
|
|
function! s:GatherOutput(job, data)
|
|
if !has_key(s:job_info_map, a:job)
|
|
return
|
|
endif
|
|
|
|
call extend(s:job_info_map[a:job].output, a:data)
|
|
endfunction
|
|
|
|
function! s:GatherOutputNeoVim(job, data, event)
|
|
call s:GatherOutput(a:job, a:data)
|
|
endfunction
|
|
|
|
function! s:GatherOutputVim(channel, data)
|
|
call s:GatherOutput(ch_getjob(a:channel), [a:data])
|
|
endfunction
|
|
|
|
function! s:LocItemCompare(left, right)
|
|
if a:left['lnum'] < a:right['lnum']
|
|
return -1
|
|
endif
|
|
|
|
if a:left['lnum'] > a:right['lnum']
|
|
return 1
|
|
endif
|
|
|
|
if a:left['col'] < a:right['col']
|
|
return -1
|
|
endif
|
|
|
|
if a:left['col'] > a:right['col']
|
|
return 1
|
|
endif
|
|
|
|
return 0
|
|
endfunction
|
|
|
|
function! s:HandleExit(job)
|
|
if !has_key(s:job_info_map, a:job)
|
|
return
|
|
endif
|
|
|
|
let job_info = s:job_info_map[a:job]
|
|
|
|
call s:ClearJob(a:job)
|
|
|
|
let linter = job_info.linter
|
|
let output = job_info.output
|
|
let buffer = job_info.buffer
|
|
|
|
let linter_loclist = s:GetFunction(linter.callback)(buffer, output)
|
|
|
|
if g:ale_buffer_should_reset_map[buffer]
|
|
let g:ale_buffer_should_reset_map[buffer] = 0
|
|
let g:ale_buffer_loclist_map[buffer] = []
|
|
endif
|
|
|
|
" Add the loclist items from the linter.
|
|
call extend(g:ale_buffer_loclist_map[buffer], linter_loclist)
|
|
|
|
" Sort the loclist again.
|
|
" We need a sorted list so we can run a binary search against it
|
|
" for efficient lookup of the messages in the cursor handler.
|
|
call sort(g:ale_buffer_loclist_map[buffer], 's:LocItemCompare')
|
|
|
|
if g:ale_set_loclist
|
|
call setloclist(0, g:ale_buffer_loclist_map[buffer])
|
|
endif
|
|
|
|
if g:ale_set_signs
|
|
call ale#sign#SetSigns(buffer, g:ale_buffer_loclist_map[buffer])
|
|
endif
|
|
|
|
" Mark line 200, column 17 with a squiggly line or something
|
|
" matchadd('ALEError', '\%200l\%17v')
|
|
endfunction
|
|
|
|
function! s:HandleExitNeoVim(job, data, event)
|
|
call s:HandleExit(a:job)
|
|
endfunction
|
|
|
|
function! s:HandleExitVim(channel)
|
|
call s:HandleExit(ch_getjob(a:channel))
|
|
endfunction
|
|
|
|
function! s:ApplyLinter(buffer, linter)
|
|
if has_key(a:linter, 'job')
|
|
" Stop previous jobs for the same linter.
|
|
call s:ClearJob(a:linter.job)
|
|
endif
|
|
|
|
if has_key(a:linter, 'command_callback')
|
|
" If there is a callback for generating a command, call that instead.
|
|
let command = s:GetFunction(a:linter.command_callback)(a:buffer)
|
|
else
|
|
let command = a:linter.command
|
|
endif
|
|
|
|
if command =~# '%s'
|
|
" If there is a '%s' in the command string, replace it with the name
|
|
" of the file.
|
|
let command = printf(command, shellescape(fnamemodify(bufname(a:buffer), ':p')))
|
|
endif
|
|
|
|
if has('nvim')
|
|
if a:linter.output_stream ==# 'stderr'
|
|
" Read from stderr instead of stdout.
|
|
let a:linter.job = jobstart(command, {
|
|
\ 'on_stderr': 's:GatherOutputNeoVim',
|
|
\ 'on_exit': 's:HandleExitNeoVim',
|
|
\})
|
|
else
|
|
let a:linter.job = jobstart(command, {
|
|
\ 'on_stdout': 's:GatherOutputNeoVim',
|
|
\ 'on_exit': 's:HandleExitNeoVim',
|
|
\})
|
|
endif
|
|
else
|
|
let job_options = {
|
|
\ 'out_mode': 'nl',
|
|
\ 'err_mode': 'nl',
|
|
\ 'close_cb': function('s:HandleExitVim'),
|
|
\ 'in_io': 'buffer',
|
|
\ 'in_buf': a:buffer,
|
|
\}
|
|
|
|
if a:linter.output_stream ==# 'stderr'
|
|
" Read from stderr instead of stdout.
|
|
let job_options.err_cb = function('s:GatherOutputVim')
|
|
else
|
|
let job_options.out_cb = function('s:GatherOutputVim')
|
|
endif
|
|
|
|
" Vim 8 will read the stdin from the file's buffer.
|
|
let a:linter.job = job_start(l:command, l:job_options)
|
|
endif
|
|
|
|
let s:job_info_map[a:linter.job] = {
|
|
\ 'linter': a:linter,
|
|
\ 'buffer': a:buffer,
|
|
\ 'output': [],
|
|
\}
|
|
|
|
if has('nvim')
|
|
" For NeoVim, we have to send the text in the buffer to the command.
|
|
call jobsend(a:linter.job, join(getline(1, '$'), "\n") . "\n")
|
|
call jobclose(a:linter.job, 'stdin')
|
|
endif
|
|
endfunction
|
|
|
|
function! s:TimerHandler(...)
|
|
let filetype = &filetype
|
|
let linters = ALEGetLinters(filetype)
|
|
|
|
let buffer = bufnr('%')
|
|
|
|
" Set a variable telling us to clear the loclist later.
|
|
let g:ale_buffer_should_reset_map[buffer] = 1
|
|
|
|
for linter in linters
|
|
" Check if a given linter has a program which can be executed.
|
|
if has_key(linter, 'executable_callback')
|
|
let l:executable = s:GetFunction(linter.executable_callback)(buffer)
|
|
else
|
|
let l:executable = linter.executable
|
|
endif
|
|
|
|
if !executable(l:executable)
|
|
" The linter's program cannot be executed, so skip it.
|
|
continue
|
|
endif
|
|
|
|
call s:ApplyLinter(buffer, linter)
|
|
endfor
|
|
endfunction
|
|
|
|
function s:BufferCleanup(buffer)
|
|
if has_key(g:ale_buffer_should_reset_map, a:buffer)
|
|
call remove(g:ale_buffer_should_reset_map, a:buffer)
|
|
endif
|
|
|
|
if has_key(g:ale_buffer_loclist_map, a:buffer)
|
|
call remove(g:ale_buffer_loclist_map, a:buffer)
|
|
endif
|
|
endfunction
|
|
|
|
function! ALEAddLinter(filetype, linter)
|
|
if !has_key(s:linters, a:filetype)
|
|
let s:linters[a:filetype] = []
|
|
endif
|
|
|
|
let new_linter = {
|
|
\ 'name': a:linter.name,
|
|
\ 'callback': a:linter.callback,
|
|
\}
|
|
|
|
if has_key(a:linter, 'executable_callback')
|
|
let new_linter.executable_callback = a:linter.executable_callback
|
|
else
|
|
let new_linter.executable = a:linter.executable
|
|
endif
|
|
|
|
if has_key(a:linter, 'command_callback')
|
|
let new_linter.command_callback = a:linter.command_callback
|
|
else
|
|
let new_linter.command = a:linter.command
|
|
endif
|
|
|
|
if has_key(a:linter, 'output_stream')
|
|
let new_linter.output_stream = a:linter.output_stream
|
|
else
|
|
let new_linter.output_stream = 'stdout'
|
|
endif
|
|
|
|
" TODO: Assert the value of the output_stream to be something sensible.
|
|
|
|
call add(s:linters[a:filetype], new_linter)
|
|
endfunction
|
|
|
|
function! ALEGetLinters(filetype)
|
|
if !has_key(s:linters, a:filetype)
|
|
return []
|
|
endif
|
|
|
|
if has_key(g:ale_linters, a:filetype)
|
|
let linters = []
|
|
" Filter loaded linters according to list of linters specified in option
|
|
for linter in s:linters[a:filetype]
|
|
if index(g:ale_linters[a:filetype], linter.name) != -1
|
|
call add(linters, linter)
|
|
endif
|
|
endfor
|
|
return linters
|
|
endif
|
|
|
|
return s:linters[a:filetype]
|
|
endfunction
|
|
|
|
function! ALELint(delay)
|
|
let filetype = &filetype
|
|
let linters = ALEGetLinters(filetype)
|
|
|
|
if s:lint_timer != -1
|
|
call timer_stop(s:lint_timer)
|
|
let s:lint_timer = -1
|
|
endif
|
|
|
|
if len(linters) == 0
|
|
" There are no linters to lint with, so stop here.
|
|
return
|
|
endif
|
|
|
|
if a:delay > 0
|
|
let s:lint_timer = timer_start(a:delay, function('s:TimerHandler'))
|
|
else
|
|
" Show empty gutter if g:ale_sign_column_always = 1
|
|
call ale#sign#InsertDummy(0)
|
|
call s:TimerHandler()
|
|
endif
|
|
endfunction
|
|
|
|
" Load all of the linters for each filetype.
|
|
runtime! ale_linters/*/*.vim
|
|
|
|
if !has('nvim') && !(has('timers') && has('job') && has('channel'))
|
|
echoerr 'ALE requires NeoVim or Vim 8 with +timers +job +channel'
|
|
echoerr 'ALE will not be run automatically'
|
|
finish
|
|
endif
|
|
|
|
if g:ale_lint_on_text_changed
|
|
augroup ALERunOnTextChangedGroup
|
|
autocmd!
|
|
autocmd TextChanged,TextChangedI * call ALELint(g:ale_lint_delay)
|
|
augroup END
|
|
endif
|
|
|
|
if g:ale_lint_on_enter
|
|
augroup ALERunOnEnterGroup
|
|
autocmd!
|
|
autocmd BufEnter * call ALELint(0)
|
|
augroup END
|
|
endif
|
|
|
|
" Clean up buffers automatically when they are unloaded.
|
|
augroup ALEBuffferCleanup
|
|
autocmd!
|
|
autocmd BufUnload * call s:BufferCleanup('<abuf>')
|
|
augroup END
|