2016-10-03 18:41:02 +00:00
|
|
|
" Author: w0rp <devw0rp@gmail.com>
|
First pass at optimizing ale to autoload (#80)
* First pass at optimizing ale to autoload
First off, the structure/function names should be revised a bit,
but I will wait for @w0rp's input before unifying the naming style.
Second off, the docs probably need some more work, I just did some
simple find-and-replace work.
With that said, this pull brings major performance gains for ale. On my
slowest system, fully loading ale and all its code takes around 150ms.
I have moved all of ale's autoload-able code to autoload/, and in
addition, implemented lazy-loading of linters. This brings load time on
that same system down to 5ms.
The only downside of lazy loading is that `g:ale_linters` cannot be
changed at runtime; however, it also speeds up performance at runtime by
simplfying the logic greatly.
Please let me know what you think!
Closes #59
* Address Travis/Vint errors
For some reason, ale isn't running vint for me...
* Incorporate feedback, make fixes
Lazy-loading logic is much improved.
* Add header comments; remove incorrect workaround
* Remove unneeded plugin guards
* Fix lazy-loading linter logic
Set the wrong variable....
* Fix capitialization
2016-10-10 18:51:29 +00:00
|
|
|
" Description: Backend execution and job management
|
|
|
|
" Executes linters in the background, using NeoVim or Vim 8 jobs
|
2016-09-17 11:06:53 +00:00
|
|
|
|
2016-09-14 10:47:52 +00:00
|
|
|
" Stores information for each job including:
|
|
|
|
"
|
|
|
|
" linter: The linter dictionary for the job.
|
|
|
|
" buffer: The buffer number for the job.
|
2016-10-08 16:27:59 +00:00
|
|
|
" output: The array of lines for the output of the job.
|
2017-05-21 15:16:06 +00:00
|
|
|
if !has_key(s:, 'job_info_map')
|
|
|
|
let s:job_info_map = {}
|
|
|
|
endif
|
|
|
|
|
2017-04-29 10:58:43 +00:00
|
|
|
let s:executable_cache_map = {}
|
|
|
|
|
|
|
|
" Check if files are executable, and if they are, remember that they are
|
|
|
|
" for subsequent calls. We'll keep checking until programs can be executed.
|
|
|
|
function! s:IsExecutable(executable) abort
|
|
|
|
if has_key(s:executable_cache_map, a:executable)
|
|
|
|
return 1
|
|
|
|
endif
|
|
|
|
|
|
|
|
if executable(a:executable)
|
|
|
|
let s:executable_cache_map[a:executable] = 1
|
|
|
|
|
|
|
|
return 1
|
|
|
|
endif
|
|
|
|
|
|
|
|
return 0
|
|
|
|
endfunction
|
2016-09-14 10:47:52 +00:00
|
|
|
|
2016-10-23 21:41:00 +00:00
|
|
|
function! ale#engine#InitBufferInfo(buffer) abort
|
|
|
|
if !has_key(g:ale_buffer_info, a:buffer)
|
2016-10-31 13:45:22 +00:00
|
|
|
" job_list will hold the list of jobs
|
|
|
|
" loclist holds the loclist items after all jobs have completed.
|
2017-02-11 15:16:08 +00:00
|
|
|
" temporary_file_list holds temporary files to be cleaned up
|
|
|
|
" temporary_directory_list holds temporary directories to be cleaned up
|
2017-02-14 23:44:37 +00:00
|
|
|
" history holds a list of previously run commands for this buffer
|
2016-10-23 21:41:00 +00:00
|
|
|
let g:ale_buffer_info[a:buffer] = {
|
|
|
|
\ 'job_list': [],
|
2016-10-24 19:21:32 +00:00
|
|
|
\ 'loclist': [],
|
2017-02-11 15:16:08 +00:00
|
|
|
\ 'temporary_file_list': [],
|
|
|
|
\ 'temporary_directory_list': [],
|
2017-02-14 23:44:37 +00:00
|
|
|
\ 'history': [],
|
2016-10-23 21:41:00 +00:00
|
|
|
\}
|
|
|
|
endif
|
|
|
|
endfunction
|
|
|
|
|
2017-07-07 22:47:41 +00:00
|
|
|
" Return 1 if ALE is busy checking a given buffer
|
|
|
|
function! ale#engine#IsCheckingBuffer(buffer) abort
|
|
|
|
let l:info = get(g:ale_buffer_info, a:buffer, {})
|
|
|
|
|
|
|
|
return get(l:info, 'waiting_for_tsserver') == 1
|
|
|
|
\|| !empty(get(l:info, 'job_list'))
|
|
|
|
endfunction
|
|
|
|
|
2017-02-11 15:16:08 +00:00
|
|
|
" Register a temporary file to be managed with the ALE engine for
|
|
|
|
" a current job run.
|
|
|
|
function! ale#engine#ManageFile(buffer, filename) abort
|
|
|
|
call add(g:ale_buffer_info[a:buffer].temporary_file_list, a:filename)
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
" Same as the above, but manage an entire directory.
|
|
|
|
function! ale#engine#ManageDirectory(buffer, directory) abort
|
|
|
|
call add(g:ale_buffer_info[a:buffer].temporary_directory_list, a:directory)
|
|
|
|
endfunction
|
|
|
|
|
2017-03-31 19:14:53 +00:00
|
|
|
" Create a new temporary directory and manage it in one go.
|
|
|
|
function! ale#engine#CreateDirectory(buffer) abort
|
|
|
|
let l:temporary_directory = tempname()
|
|
|
|
" Create the temporary directory for the file, unreadable by 'other'
|
|
|
|
" users.
|
|
|
|
call mkdir(l:temporary_directory, '', 0750)
|
|
|
|
call ale#engine#ManageDirectory(a:buffer, l:temporary_directory)
|
|
|
|
|
|
|
|
return l:temporary_directory
|
|
|
|
endfunction
|
|
|
|
|
2017-02-11 15:16:08 +00:00
|
|
|
function! ale#engine#RemoveManagedFiles(buffer) abort
|
2017-07-07 22:47:41 +00:00
|
|
|
let l:info = get(g:ale_buffer_info, a:buffer)
|
2017-02-11 15:16:08 +00:00
|
|
|
|
2017-02-14 21:02:49 +00:00
|
|
|
" We can't delete anything in a sandbox, so wait until we escape from
|
|
|
|
" it to delete temporary files and directories.
|
|
|
|
if ale#util#InSandbox()
|
|
|
|
return
|
|
|
|
endif
|
|
|
|
|
2017-02-11 15:16:08 +00:00
|
|
|
" Delete files with a call akin to a plan `rm` command.
|
2017-07-07 22:47:41 +00:00
|
|
|
if has_key(l:info, 'temporary_file_list')
|
|
|
|
for l:filename in l:info.temporary_file_list
|
|
|
|
call delete(l:filename)
|
|
|
|
endfor
|
2017-02-11 15:16:08 +00:00
|
|
|
|
2017-07-07 22:47:41 +00:00
|
|
|
let l:info.temporary_file_list = []
|
|
|
|
endif
|
2017-02-11 15:16:08 +00:00
|
|
|
|
|
|
|
" Delete directories like `rm -rf`.
|
|
|
|
" Directories are handled differently from files, so paths that are
|
|
|
|
" intended to be single files can be set up for automatic deletion without
|
|
|
|
" accidentally deleting entire directories.
|
2017-07-07 22:47:41 +00:00
|
|
|
if has_key(l:info, 'temporary_directory_list')
|
|
|
|
for l:directory in l:info.temporary_directory_list
|
|
|
|
call delete(l:directory, 'rf')
|
|
|
|
endfor
|
2017-02-11 15:16:08 +00:00
|
|
|
|
2017-07-07 22:47:41 +00:00
|
|
|
let l:info.temporary_directory_list = []
|
|
|
|
endif
|
2017-02-11 15:16:08 +00:00
|
|
|
endfunction
|
|
|
|
|
2017-05-12 20:16:15 +00:00
|
|
|
function! s:GatherOutput(job_id, line) abort
|
|
|
|
if has_key(s:job_info_map, a:job_id)
|
|
|
|
call add(s:job_info_map[a:job_id].output, a:line)
|
2016-10-08 20:04:42 +00:00
|
|
|
endif
|
2017-05-12 20:16:15 +00:00
|
|
|
endfunction
|
2016-10-08 20:04:42 +00:00
|
|
|
|
2017-06-13 16:53:47 +00:00
|
|
|
function! s:HandleLoclist(linter_name, buffer, loclist) abort
|
2017-06-08 16:28:38 +00:00
|
|
|
" Make some adjustments to the loclists to fix common problems, and also
|
|
|
|
" to set default values for loclist items.
|
2017-06-13 16:53:47 +00:00
|
|
|
let l:linter_loclist = ale#engine#FixLocList(a:buffer, a:linter_name, a:loclist)
|
2017-06-08 16:28:38 +00:00
|
|
|
|
|
|
|
" Remove previous items for this linter.
|
2017-06-13 16:53:47 +00:00
|
|
|
call filter(g:ale_buffer_info[a:buffer].loclist, 'v:val.linter_name !=# a:linter_name')
|
2017-06-08 16:28:38 +00:00
|
|
|
" Add the new items.
|
|
|
|
call extend(g:ale_buffer_info[a:buffer].loclist, l: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_info[a:buffer].loclist, 'ale#util#LocItemCompare')
|
|
|
|
|
2017-07-07 22:47:41 +00:00
|
|
|
if ale#ShouldDoNothing()
|
|
|
|
return
|
2017-06-08 16:28:38 +00:00
|
|
|
endif
|
|
|
|
|
|
|
|
call ale#engine#SetResults(a:buffer, g:ale_buffer_info[a:buffer].loclist)
|
|
|
|
endfunction
|
|
|
|
|
2017-05-12 20:16:15 +00:00
|
|
|
function! s:HandleExit(job_id, exit_code) abort
|
|
|
|
if !has_key(s:job_info_map, a:job_id)
|
2016-09-08 23:23:26 +00:00
|
|
|
return
|
|
|
|
endif
|
|
|
|
|
2017-05-12 20:16:15 +00:00
|
|
|
let l:job_info = s:job_info_map[a:job_id]
|
2016-10-10 23:00:09 +00:00
|
|
|
let l:linter = l:job_info.linter
|
|
|
|
let l:output = l:job_info.output
|
|
|
|
let l:buffer = l:job_info.buffer
|
2016-10-25 19:25:23 +00:00
|
|
|
let l:next_chain_index = l:job_info.next_chain_index
|
2016-09-08 23:23:26 +00:00
|
|
|
|
2017-05-12 20:16:15 +00:00
|
|
|
if g:ale_history_enabled
|
|
|
|
call ale#history#SetExitCode(l:buffer, a:job_id, a:exit_code)
|
|
|
|
endif
|
|
|
|
|
2017-06-06 15:44:01 +00:00
|
|
|
" Remove this job from the list.
|
|
|
|
call ale#job#Stop(a:job_id)
|
|
|
|
call remove(s:job_info_map, a:job_id)
|
|
|
|
call filter(g:ale_buffer_info[l:buffer].job_list, 'v:val !=# a:job_id')
|
2016-10-13 14:13:11 +00:00
|
|
|
|
2017-02-14 21:02:49 +00:00
|
|
|
" Stop here if we land in the handle for a job completing if we're in
|
|
|
|
" a sandbox.
|
|
|
|
if ale#util#InSandbox()
|
|
|
|
return
|
|
|
|
endif
|
|
|
|
|
2017-05-12 19:38:52 +00:00
|
|
|
if has('nvim') && !empty(l:output) && empty(l:output[-1])
|
|
|
|
call remove(l:output, -1)
|
|
|
|
endif
|
|
|
|
|
2016-10-25 19:25:23 +00:00
|
|
|
if l:next_chain_index < len(get(l:linter, 'command_chain', []))
|
|
|
|
call s:InvokeChain(l:buffer, l:linter, l:next_chain_index, l:output)
|
|
|
|
return
|
|
|
|
endif
|
|
|
|
|
2017-02-16 23:18:57 +00:00
|
|
|
" Log the output of the command for ALEInfo if we should.
|
|
|
|
if g:ale_history_enabled && g:ale_history_log_output
|
2017-05-12 20:16:15 +00:00
|
|
|
call ale#history#RememberOutput(l:buffer, a:job_id, l:output[:])
|
2017-02-16 23:18:57 +00:00
|
|
|
endif
|
|
|
|
|
2017-06-08 16:28:38 +00:00
|
|
|
let l:loclist = ale#util#GetFunction(l:linter.callback)(l:buffer, l:output)
|
2016-10-10 18:56:05 +00:00
|
|
|
|
2017-06-13 16:53:47 +00:00
|
|
|
call s:HandleLoclist(l:linter.name, l:buffer, l:loclist)
|
2017-06-08 16:28:38 +00:00
|
|
|
endfunction
|
2017-03-21 14:52:02 +00:00
|
|
|
|
2017-06-13 16:53:47 +00:00
|
|
|
function! s:HandleLSPResponse(response) abort
|
|
|
|
let l:is_diag_response = get(a:response, 'type', '') ==# 'event'
|
|
|
|
\ && get(a:response, 'event', '') ==# 'semanticDiag'
|
2017-06-07 22:12:45 +00:00
|
|
|
|
2017-06-13 16:53:47 +00:00
|
|
|
if !l:is_diag_response
|
2017-06-08 16:28:38 +00:00
|
|
|
return
|
|
|
|
endif
|
2017-06-07 22:12:45 +00:00
|
|
|
|
2017-06-13 16:53:47 +00:00
|
|
|
let l:buffer = bufnr(a:response.body.file)
|
|
|
|
|
|
|
|
let l:info = get(g:ale_buffer_info, l:buffer, {})
|
|
|
|
|
|
|
|
if empty(l:info)
|
|
|
|
return
|
|
|
|
endif
|
2016-10-31 13:45:22 +00:00
|
|
|
|
2017-06-13 16:53:47 +00:00
|
|
|
let l:info.waiting_for_tsserver = 0
|
2017-06-07 22:12:45 +00:00
|
|
|
|
2017-06-08 16:28:38 +00:00
|
|
|
let l:loclist = ale#lsp#response#ReadTSServerDiagnostics(a:response)
|
2017-06-07 22:12:45 +00:00
|
|
|
|
2017-06-13 16:53:47 +00:00
|
|
|
call s:HandleLoclist('tsserver', l:buffer, l:loclist)
|
2017-02-09 18:47:14 +00:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! ale#engine#SetResults(buffer, loclist) abort
|
2017-07-07 22:47:41 +00:00
|
|
|
let l:linting_is_done = !ale#engine#IsCheckingBuffer(a:buffer)
|
2017-06-07 22:12:45 +00:00
|
|
|
|
2017-03-14 23:04:25 +00:00
|
|
|
" Set signs first. This could potentially fix some line numbers.
|
2017-03-21 14:52:02 +00:00
|
|
|
" The List could be sorted again here by SetSigns.
|
2016-09-08 23:23:26 +00:00
|
|
|
if g:ale_set_signs
|
2017-02-09 18:47:14 +00:00
|
|
|
call ale#sign#SetSigns(a:buffer, a:loclist)
|
2017-06-07 22:12:45 +00:00
|
|
|
|
|
|
|
if l:linting_is_done
|
|
|
|
call ale#sign#RemoveDummySignIfNeeded(a:buffer)
|
|
|
|
endif
|
2016-09-08 23:23:26 +00:00
|
|
|
endif
|
|
|
|
|
2017-03-14 23:04:25 +00:00
|
|
|
if g:ale_set_quickfix || g:ale_set_loclist
|
|
|
|
call ale#list#SetLists(a:buffer, a:loclist)
|
2017-06-07 22:12:45 +00:00
|
|
|
|
|
|
|
if l:linting_is_done
|
|
|
|
call ale#list#CloseWindowIfNeeded(a:buffer)
|
|
|
|
endif
|
2017-03-14 23:04:25 +00:00
|
|
|
endif
|
|
|
|
|
Implement a more efficient statusbar
The statusbar now keeps its state in a separate variable, in order to
avoid excess iterations. The engine now updates said variable on run,
and a new function is made available for external statusbars to call (to
avoid dependencies on internal implementation details of ale).
To keep things light, the status bar code is not loaded unless invoked
by the user or an external plugin. On the first load it will update
itself from the global loclist, after that, the engine will handle all
updates.
The external integration function, `ale#statusline#Count()`, will return
a tuple in the format [E, W] (where E is errors, W is warnings), unless
no data exists (ie, the plugin doesn't have a linter for a file or has
not run yet), in which case it returns 0/false.
2016-10-11 21:51:01 +00:00
|
|
|
if exists('*ale#statusline#Update')
|
|
|
|
" Don't load/run if not already loaded.
|
2017-02-09 18:47:14 +00:00
|
|
|
call ale#statusline#Update(a:buffer, a:loclist)
|
Implement a more efficient statusbar
The statusbar now keeps its state in a separate variable, in order to
avoid excess iterations. The engine now updates said variable on run,
and a new function is made available for external statusbars to call (to
avoid dependencies on internal implementation details of ale).
To keep things light, the status bar code is not loaded unless invoked
by the user or an external plugin. On the first load it will update
itself from the global loclist, after that, the engine will handle all
updates.
The external integration function, `ale#statusline#Count()`, will return
a tuple in the format [E, W] (where E is errors, W is warnings), unless
no data exists (ie, the plugin doesn't have a linter for a file or has
not run yet), in which case it returns 0/false.
2016-10-11 21:51:01 +00:00
|
|
|
endif
|
2017-02-13 00:18:51 +00:00
|
|
|
|
|
|
|
if g:ale_set_highlights
|
|
|
|
call ale#highlight#SetHighlights(a:buffer, a:loclist)
|
|
|
|
endif
|
2017-03-02 23:36:31 +00:00
|
|
|
|
|
|
|
if g:ale_echo_cursor
|
|
|
|
" Try and echo the warning now.
|
|
|
|
" This will only do something meaningful if we're in normal mode.
|
|
|
|
call ale#cursor#EchoCursorWarning()
|
|
|
|
endif
|
2017-07-07 22:47:41 +00:00
|
|
|
|
|
|
|
if l:linting_is_done
|
|
|
|
" Automatically remove all managed temporary files and directories
|
|
|
|
" now that all jobs have completed.
|
|
|
|
call ale#engine#RemoveManagedFiles(a:buffer)
|
|
|
|
|
|
|
|
" Call user autocommands. This allows users to hook into ALE's lint cycle.
|
|
|
|
silent doautocmd User ALELint
|
|
|
|
endif
|
2016-09-08 23:23:26 +00:00
|
|
|
endfunction
|
|
|
|
|
2017-06-14 16:59:13 +00:00
|
|
|
function! s:RemapItemTypes(type_map, loclist) abort
|
|
|
|
for l:item in a:loclist
|
|
|
|
let l:key = l:item.type
|
|
|
|
\ . (get(l:item, 'sub_type', '') ==# 'style' ? 'S' : '')
|
|
|
|
let l:new_key = get(a:type_map, l:key, '')
|
|
|
|
|
|
|
|
if l:new_key ==# 'E'
|
|
|
|
\|| l:new_key ==# 'ES'
|
|
|
|
\|| l:new_key ==# 'W'
|
|
|
|
\|| l:new_key ==# 'WS'
|
|
|
|
\|| l:new_key ==# 'I'
|
|
|
|
let l:item.type = l:new_key[0]
|
|
|
|
|
|
|
|
if l:new_key ==# 'ES' || l:new_key ==# 'WS'
|
|
|
|
let l:item.sub_type = 'style'
|
|
|
|
elseif has_key(l:item, 'sub_type')
|
|
|
|
call remove(l:item, 'sub_type')
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
endfunction
|
|
|
|
|
2017-06-13 16:53:47 +00:00
|
|
|
function! ale#engine#FixLocList(buffer, linter_name, loclist) abort
|
2017-02-26 14:51:22 +00:00
|
|
|
let l:new_loclist = []
|
|
|
|
|
First pass at optimizing ale to autoload (#80)
* First pass at optimizing ale to autoload
First off, the structure/function names should be revised a bit,
but I will wait for @w0rp's input before unifying the naming style.
Second off, the docs probably need some more work, I just did some
simple find-and-replace work.
With that said, this pull brings major performance gains for ale. On my
slowest system, fully loading ale and all its code takes around 150ms.
I have moved all of ale's autoload-able code to autoload/, and in
addition, implemented lazy-loading of linters. This brings load time on
that same system down to 5ms.
The only downside of lazy loading is that `g:ale_linters` cannot be
changed at runtime; however, it also speeds up performance at runtime by
simplfying the logic greatly.
Please let me know what you think!
Closes #59
* Address Travis/Vint errors
For some reason, ale isn't running vint for me...
* Incorporate feedback, make fixes
Lazy-loading logic is much improved.
* Add header comments; remove incorrect workaround
* Remove unneeded plugin guards
* Fix lazy-loading linter logic
Set the wrong variable....
* Fix capitialization
2016-10-10 18:51:29 +00:00
|
|
|
" Some errors have line numbers beyond the end of the file,
|
|
|
|
" so we need to adjust them so they set the error at the last line
|
|
|
|
" of the file instead.
|
2016-10-10 23:00:09 +00:00
|
|
|
let l:last_line_number = ale#util#GetLineCount(a:buffer)
|
First pass at optimizing ale to autoload (#80)
* First pass at optimizing ale to autoload
First off, the structure/function names should be revised a bit,
but I will wait for @w0rp's input before unifying the naming style.
Second off, the docs probably need some more work, I just did some
simple find-and-replace work.
With that said, this pull brings major performance gains for ale. On my
slowest system, fully loading ale and all its code takes around 150ms.
I have moved all of ale's autoload-able code to autoload/, and in
addition, implemented lazy-loading of linters. This brings load time on
that same system down to 5ms.
The only downside of lazy loading is that `g:ale_linters` cannot be
changed at runtime; however, it also speeds up performance at runtime by
simplfying the logic greatly.
Please let me know what you think!
Closes #59
* Address Travis/Vint errors
For some reason, ale isn't running vint for me...
* Incorporate feedback, make fixes
Lazy-loading logic is much improved.
* Add header comments; remove incorrect workaround
* Remove unneeded plugin guards
* Fix lazy-loading linter logic
Set the wrong variable....
* Fix capitialization
2016-10-10 18:51:29 +00:00
|
|
|
|
2017-02-26 14:51:22 +00:00
|
|
|
for l:old_item in a:loclist
|
|
|
|
" Copy the loclist item with some default values and corrections.
|
|
|
|
"
|
|
|
|
" line and column numbers will be converted to numbers.
|
|
|
|
" The buffer will default to the buffer being checked.
|
|
|
|
" The vcol setting will default to 0, a byte index.
|
|
|
|
" The error type will default to 'E' for errors.
|
|
|
|
" The error number will default to -1.
|
|
|
|
"
|
|
|
|
" The line number and text are the only required keys.
|
|
|
|
"
|
|
|
|
" The linter_name will be set on the errors so it can be used in
|
|
|
|
" output, filtering, etc..
|
|
|
|
let l:item = {
|
|
|
|
\ 'text': l:old_item.text,
|
|
|
|
\ 'lnum': str2nr(l:old_item.lnum),
|
|
|
|
\ 'col': str2nr(get(l:old_item, 'col', 0)),
|
|
|
|
\ 'bufnr': get(l:old_item, 'bufnr', a:buffer),
|
|
|
|
\ 'vcol': get(l:old_item, 'vcol', 0),
|
|
|
|
\ 'type': get(l:old_item, 'type', 'E'),
|
|
|
|
\ 'nr': get(l:old_item, 'nr', -1),
|
2017-06-13 16:53:47 +00:00
|
|
|
\ 'linter_name': a:linter_name,
|
2017-02-26 14:51:22 +00:00
|
|
|
\}
|
|
|
|
|
2017-03-02 07:14:30 +00:00
|
|
|
if has_key(l:old_item, 'detail')
|
|
|
|
let l:item.detail = l:old_item.detail
|
|
|
|
endif
|
|
|
|
|
2017-05-31 12:14:39 +00:00
|
|
|
" Pass on a end_col key if set, used for highlights.
|
2017-05-16 17:12:49 +00:00
|
|
|
if has_key(l:old_item, 'end_col')
|
|
|
|
let l:item.end_col = str2nr(l:old_item.end_col)
|
|
|
|
endif
|
|
|
|
|
2017-05-31 12:14:39 +00:00
|
|
|
if has_key(l:old_item, 'end_lnum')
|
|
|
|
let l:item.end_lnum = str2nr(l:old_item.end_lnum)
|
|
|
|
endif
|
|
|
|
|
2017-05-20 22:32:41 +00:00
|
|
|
if has_key(l:old_item, 'sub_type')
|
|
|
|
let l:item.sub_type = l:old_item.sub_type
|
|
|
|
endif
|
|
|
|
|
2017-06-24 11:35:01 +00:00
|
|
|
if l:item.lnum < 1
|
|
|
|
" When errors appear before line 1, put them at line 1.
|
2016-10-10 23:00:09 +00:00
|
|
|
let l:item.lnum = 1
|
|
|
|
elseif l:item.lnum > l:last_line_number
|
2017-02-26 14:51:22 +00:00
|
|
|
" When errors go beyond the end of the file, put them at the end.
|
2016-10-10 23:00:09 +00:00
|
|
|
let l:item.lnum = l:last_line_number
|
First pass at optimizing ale to autoload (#80)
* First pass at optimizing ale to autoload
First off, the structure/function names should be revised a bit,
but I will wait for @w0rp's input before unifying the naming style.
Second off, the docs probably need some more work, I just did some
simple find-and-replace work.
With that said, this pull brings major performance gains for ale. On my
slowest system, fully loading ale and all its code takes around 150ms.
I have moved all of ale's autoload-able code to autoload/, and in
addition, implemented lazy-loading of linters. This brings load time on
that same system down to 5ms.
The only downside of lazy loading is that `g:ale_linters` cannot be
changed at runtime; however, it also speeds up performance at runtime by
simplfying the logic greatly.
Please let me know what you think!
Closes #59
* Address Travis/Vint errors
For some reason, ale isn't running vint for me...
* Incorporate feedback, make fixes
Lazy-loading logic is much improved.
* Add header comments; remove incorrect workaround
* Remove unneeded plugin guards
* Fix lazy-loading linter logic
Set the wrong variable....
* Fix capitialization
2016-10-10 18:51:29 +00:00
|
|
|
endif
|
2017-02-26 14:51:22 +00:00
|
|
|
|
|
|
|
call add(l:new_loclist, l:item)
|
First pass at optimizing ale to autoload (#80)
* First pass at optimizing ale to autoload
First off, the structure/function names should be revised a bit,
but I will wait for @w0rp's input before unifying the naming style.
Second off, the docs probably need some more work, I just did some
simple find-and-replace work.
With that said, this pull brings major performance gains for ale. On my
slowest system, fully loading ale and all its code takes around 150ms.
I have moved all of ale's autoload-able code to autoload/, and in
addition, implemented lazy-loading of linters. This brings load time on
that same system down to 5ms.
The only downside of lazy loading is that `g:ale_linters` cannot be
changed at runtime; however, it also speeds up performance at runtime by
simplfying the logic greatly.
Please let me know what you think!
Closes #59
* Address Travis/Vint errors
For some reason, ale isn't running vint for me...
* Incorporate feedback, make fixes
Lazy-loading logic is much improved.
* Add header comments; remove incorrect workaround
* Remove unneeded plugin guards
* Fix lazy-loading linter logic
Set the wrong variable....
* Fix capitialization
2016-10-10 18:51:29 +00:00
|
|
|
endfor
|
2017-02-26 14:51:22 +00:00
|
|
|
|
2017-06-14 16:59:13 +00:00
|
|
|
let l:type_map = get(ale#Var(a:buffer, 'type_map'), a:linter_name, {})
|
|
|
|
|
|
|
|
if !empty(l:type_map)
|
|
|
|
call s:RemapItemTypes(l:type_map, l:new_loclist)
|
|
|
|
endif
|
|
|
|
|
2017-02-26 14:51:22 +00:00
|
|
|
return l:new_loclist
|
First pass at optimizing ale to autoload (#80)
* First pass at optimizing ale to autoload
First off, the structure/function names should be revised a bit,
but I will wait for @w0rp's input before unifying the naming style.
Second off, the docs probably need some more work, I just did some
simple find-and-replace work.
With that said, this pull brings major performance gains for ale. On my
slowest system, fully loading ale and all its code takes around 150ms.
I have moved all of ale's autoload-able code to autoload/, and in
addition, implemented lazy-loading of linters. This brings load time on
that same system down to 5ms.
The only downside of lazy loading is that `g:ale_linters` cannot be
changed at runtime; however, it also speeds up performance at runtime by
simplfying the logic greatly.
Please let me know what you think!
Closes #59
* Address Travis/Vint errors
For some reason, ale isn't running vint for me...
* Incorporate feedback, make fixes
Lazy-loading logic is much improved.
* Add header comments; remove incorrect workaround
* Remove unneeded plugin guards
* Fix lazy-loading linter logic
Set the wrong variable....
* Fix capitialization
2016-10-10 18:51:29 +00:00
|
|
|
endfunction
|
|
|
|
|
2017-02-11 18:14:18 +00:00
|
|
|
" Given part of a command, replace any % with %%, so that no characters in
|
|
|
|
" the string will be replaced with filenames, etc.
|
|
|
|
function! ale#engine#EscapeCommandPart(command_part) abort
|
|
|
|
return substitute(a:command_part, '%', '%%', 'g')
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:CreateTemporaryFileForJob(buffer, temporary_file) abort
|
|
|
|
if empty(a:temporary_file)
|
|
|
|
" There is no file, so we didn't create anything.
|
|
|
|
return 0
|
|
|
|
endif
|
|
|
|
|
|
|
|
let l:temporary_directory = fnamemodify(a:temporary_file, ':h')
|
|
|
|
" Create the temporary directory for the file, unreadable by 'other'
|
|
|
|
" users.
|
|
|
|
call mkdir(l:temporary_directory, '', 0750)
|
|
|
|
" Automatically delete the directory later.
|
|
|
|
call ale#engine#ManageDirectory(a:buffer, l:temporary_directory)
|
|
|
|
" Write the buffer out to a file.
|
2017-02-11 18:19:01 +00:00
|
|
|
call writefile(getbufline(a:buffer, 1, '$'), a:temporary_file)
|
2017-02-11 18:14:18 +00:00
|
|
|
|
|
|
|
return 1
|
|
|
|
endfunction
|
|
|
|
|
2017-07-07 22:47:41 +00:00
|
|
|
" Run a job.
|
|
|
|
"
|
|
|
|
" Returns 1 when the job was started successfully.
|
2017-02-09 23:32:57 +00:00
|
|
|
function! s:RunJob(options) abort
|
|
|
|
let l:command = a:options.command
|
|
|
|
let l:buffer = a:options.buffer
|
|
|
|
let l:linter = a:options.linter
|
|
|
|
let l:output_stream = a:options.output_stream
|
|
|
|
let l:next_chain_index = a:options.next_chain_index
|
|
|
|
let l:read_buffer = a:options.read_buffer
|
2016-09-11 15:49:55 +00:00
|
|
|
|
2017-07-07 22:47:41 +00:00
|
|
|
if empty(l:command)
|
|
|
|
return 0
|
|
|
|
endif
|
|
|
|
|
2017-05-17 10:17:49 +00:00
|
|
|
let [l:temporary_file, l:command] = ale#command#FormatCommand(l:buffer, l:command, l:read_buffer)
|
2017-02-11 22:43:13 +00:00
|
|
|
|
2017-02-11 18:14:18 +00:00
|
|
|
if s:CreateTemporaryFileForJob(l:buffer, l:temporary_file)
|
|
|
|
" If a temporary filename has been formatted in to the command, then
|
|
|
|
" we do not need to send the Vim buffer to the command.
|
|
|
|
let l:read_buffer = 0
|
2016-10-04 12:50:44 +00:00
|
|
|
endif
|
|
|
|
|
2017-07-07 09:47:09 +00:00
|
|
|
" Add a newline to commands which need it.
|
|
|
|
" This is only used for Flow for now, and is not documented.
|
|
|
|
if l:linter.add_newline
|
|
|
|
if has('win32')
|
|
|
|
let l:command = l:command . '; echo.'
|
|
|
|
else
|
|
|
|
let l:command = l:command . '; echo'
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
|
2017-05-12 20:43:34 +00:00
|
|
|
let l:command = ale#job#PrepareCommand(l:command)
|
2017-05-12 20:16:15 +00:00
|
|
|
let l:job_options = {
|
|
|
|
\ 'mode': 'nl',
|
|
|
|
\ 'exit_cb': function('s:HandleExit'),
|
|
|
|
\}
|
|
|
|
|
|
|
|
if l:output_stream ==# 'stderr'
|
|
|
|
let l:job_options.err_cb = function('s:GatherOutput')
|
|
|
|
elseif l:output_stream ==# 'both'
|
|
|
|
let l:job_options.out_cb = function('s:GatherOutput')
|
|
|
|
let l:job_options.err_cb = function('s:GatherOutput')
|
|
|
|
else
|
|
|
|
let l:job_options.out_cb = function('s:GatherOutput')
|
2017-04-03 21:24:30 +00:00
|
|
|
endif
|
|
|
|
|
|
|
|
if get(g:, 'ale_run_synchronously') == 1
|
|
|
|
" Find a unique Job value to use, which will be the same as the ID for
|
|
|
|
" running commands synchronously. This is only for test code.
|
2017-05-12 20:16:15 +00:00
|
|
|
let l:job_id = len(s:job_info_map) + 1
|
2017-04-03 21:24:30 +00:00
|
|
|
|
2017-05-12 20:16:15 +00:00
|
|
|
while has_key(s:job_info_map, l:job_id)
|
|
|
|
let l:job_id += 1
|
2017-04-03 21:24:30 +00:00
|
|
|
endwhile
|
2016-09-10 00:37:40 +00:00
|
|
|
else
|
2017-05-12 20:16:15 +00:00
|
|
|
let l:job_id = ale#job#Start(l:command, l:job_options)
|
2016-09-10 00:37:40 +00:00
|
|
|
endif
|
2016-09-08 23:23:26 +00:00
|
|
|
|
2017-02-14 23:44:37 +00:00
|
|
|
let l:status = 'failed'
|
|
|
|
|
2016-10-08 20:52:41 +00:00
|
|
|
" Only proceed if the job is being run.
|
2017-05-12 20:16:15 +00:00
|
|
|
if l:job_id
|
2016-10-23 21:41:00 +00:00
|
|
|
" Add the job to the list of jobs, so we can track them.
|
2017-05-12 20:16:15 +00:00
|
|
|
call add(g:ale_buffer_info[l:buffer].job_list, l:job_id)
|
2016-09-08 23:23:26 +00:00
|
|
|
|
2017-02-16 21:18:03 +00:00
|
|
|
let l:status = 'started'
|
2016-10-08 20:04:42 +00:00
|
|
|
" Store the ID for the job in the map to read back again.
|
2017-02-14 23:44:37 +00:00
|
|
|
let s:job_info_map[l:job_id] = {
|
2016-10-25 19:25:23 +00:00
|
|
|
\ 'linter': l:linter,
|
|
|
|
\ 'buffer': l:buffer,
|
2016-10-08 20:04:42 +00:00
|
|
|
\ 'output': [],
|
2016-10-25 19:25:23 +00:00
|
|
|
\ 'next_chain_index': l:next_chain_index,
|
2016-10-08 20:04:42 +00:00
|
|
|
\}
|
2016-09-10 00:37:40 +00:00
|
|
|
endif
|
2017-02-14 23:44:37 +00:00
|
|
|
|
2017-02-16 21:33:44 +00:00
|
|
|
if g:ale_history_enabled
|
|
|
|
call ale#history#Add(l:buffer, l:status, l:job_id, l:command)
|
|
|
|
else
|
|
|
|
let g:ale_buffer_info[l:buffer].history = []
|
|
|
|
endif
|
2017-04-03 21:24:30 +00:00
|
|
|
|
|
|
|
if get(g:, 'ale_run_synchronously') == 1
|
|
|
|
" Run a command synchronously if this test option is set.
|
|
|
|
let s:job_info_map[l:job_id].output = systemlist(
|
|
|
|
\ type(l:command) == type([])
|
2017-05-12 08:20:16 +00:00
|
|
|
\ ? join(l:command[0:1]) . ' ' . ale#Escape(l:command[2])
|
2017-04-03 21:24:30 +00:00
|
|
|
\ : l:command
|
|
|
|
\)
|
2017-05-12 20:16:15 +00:00
|
|
|
|
2017-05-18 12:21:14 +00:00
|
|
|
call l:job_options.exit_cb(l:job_id, v:shell_error)
|
2017-04-03 21:24:30 +00:00
|
|
|
endif
|
2017-07-07 22:47:41 +00:00
|
|
|
|
|
|
|
return l:job_id != 0
|
2016-09-08 23:23:26 +00:00
|
|
|
endfunction
|
2016-10-17 22:26:19 +00:00
|
|
|
|
2017-02-09 23:32:57 +00:00
|
|
|
" Determine which commands to run for a link in a command chain, or
|
|
|
|
" just a regular command.
|
|
|
|
function! ale#engine#ProcessChain(buffer, linter, chain_index, input) abort
|
2016-10-25 19:25:23 +00:00
|
|
|
let l:output_stream = get(a:linter, 'output_stream', 'stdout')
|
2017-02-09 23:32:57 +00:00
|
|
|
let l:read_buffer = a:linter.read_buffer
|
2017-02-04 18:30:30 +00:00
|
|
|
let l:chain_index = a:chain_index
|
|
|
|
let l:input = a:input
|
2016-10-25 19:25:23 +00:00
|
|
|
|
|
|
|
if has_key(a:linter, 'command_chain')
|
2017-02-04 18:30:30 +00:00
|
|
|
while l:chain_index < len(a:linter.command_chain)
|
|
|
|
" Run a chain of commands, one asychronous command after the other,
|
|
|
|
" so that many programs can be run in a sequence.
|
|
|
|
let l:chain_item = a:linter.command_chain[l:chain_index]
|
|
|
|
|
|
|
|
if l:chain_index == 0
|
|
|
|
" The first callback in the chain takes only a buffer number.
|
|
|
|
let l:command = ale#util#GetFunction(l:chain_item.callback)(
|
|
|
|
\ a:buffer
|
|
|
|
\)
|
|
|
|
else
|
|
|
|
" The second callback in the chain takes some input too.
|
|
|
|
let l:command = ale#util#GetFunction(l:chain_item.callback)(
|
|
|
|
\ a:buffer,
|
|
|
|
\ l:input
|
|
|
|
\)
|
|
|
|
endif
|
2016-10-25 19:25:23 +00:00
|
|
|
|
2017-02-04 18:30:30 +00:00
|
|
|
if !empty(l:command)
|
|
|
|
" We hit a command to run, so we'll execute that
|
2017-02-09 23:32:57 +00:00
|
|
|
|
|
|
|
" The chain item can override the output_stream option.
|
|
|
|
if has_key(l:chain_item, 'output_stream')
|
|
|
|
let l:output_stream = l:chain_item.output_stream
|
|
|
|
endif
|
|
|
|
|
|
|
|
" The chain item can override the read_buffer option.
|
|
|
|
if has_key(l:chain_item, 'read_buffer')
|
|
|
|
let l:read_buffer = l:chain_item.read_buffer
|
|
|
|
elseif l:chain_index != len(a:linter.command_chain) - 1
|
|
|
|
" Don't read the buffer for commands besides the last one
|
|
|
|
" in the chain by default.
|
|
|
|
let l:read_buffer = 0
|
|
|
|
endif
|
|
|
|
|
2017-02-04 18:30:30 +00:00
|
|
|
break
|
|
|
|
endif
|
2016-10-25 19:25:23 +00:00
|
|
|
|
2017-02-04 18:30:30 +00:00
|
|
|
" Command chain items can return an empty string to indicate that
|
|
|
|
" a command should be skipped, so we should try the next item
|
|
|
|
" with no input.
|
|
|
|
let l:input = []
|
|
|
|
let l:chain_index += 1
|
|
|
|
endwhile
|
2016-10-25 19:25:23 +00:00
|
|
|
else
|
2017-07-02 12:17:24 +00:00
|
|
|
let l:command = ale#linter#GetCommand(a:buffer, a:linter)
|
2016-10-25 19:25:23 +00:00
|
|
|
endif
|
|
|
|
|
2017-02-09 23:32:57 +00:00
|
|
|
return {
|
|
|
|
\ 'command': l:command,
|
2016-10-25 19:25:23 +00:00
|
|
|
\ 'buffer': a:buffer,
|
|
|
|
\ 'linter': a:linter,
|
|
|
|
\ 'output_stream': l:output_stream,
|
2017-02-04 18:30:30 +00:00
|
|
|
\ 'next_chain_index': l:chain_index + 1,
|
2017-02-09 23:32:57 +00:00
|
|
|
\ 'read_buffer': l:read_buffer,
|
|
|
|
\}
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:InvokeChain(buffer, linter, chain_index, input) abort
|
|
|
|
let l:options = ale#engine#ProcessChain(a:buffer, a:linter, a:chain_index, a:input)
|
|
|
|
|
2017-07-07 22:47:41 +00:00
|
|
|
return s:RunJob(l:options)
|
2016-10-25 19:25:23 +00:00
|
|
|
endfunction
|
|
|
|
|
2017-07-07 22:47:41 +00:00
|
|
|
function! s:StopCurrentJobs(buffer, include_lint_file_jobs) abort
|
2017-06-06 15:44:01 +00:00
|
|
|
let l:info = get(g:ale_buffer_info, a:buffer, {})
|
|
|
|
let l:new_job_list = []
|
|
|
|
|
|
|
|
for l:job_id in get(l:info, 'job_list', [])
|
|
|
|
let l:job_info = get(s:job_info_map, l:job_id, {})
|
2017-03-14 23:51:57 +00:00
|
|
|
|
2017-06-06 15:44:01 +00:00
|
|
|
if !empty(l:job_info)
|
|
|
|
if a:include_lint_file_jobs || !l:job_info.linter.lint_file
|
|
|
|
call ale#job#Stop(l:job_id)
|
|
|
|
call remove(s:job_info_map, l:job_id)
|
|
|
|
else
|
|
|
|
call add(l:new_job_list, l:job_id)
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
|
|
|
|
" Update the List, so it includes only the jobs we still need.
|
|
|
|
let l:info.job_list = l:new_job_list
|
|
|
|
endfunction
|
|
|
|
|
2017-06-08 16:28:38 +00:00
|
|
|
function! s:CheckWithTSServer(buffer, linter, executable) abort
|
|
|
|
let l:info = g:ale_buffer_info[a:buffer]
|
|
|
|
|
2017-07-02 12:17:24 +00:00
|
|
|
let l:command = ale#job#PrepareCommand(
|
|
|
|
\ ale#linter#GetCommand(a:buffer, a:linter),
|
|
|
|
\)
|
2017-07-01 13:30:34 +00:00
|
|
|
let l:id = ale#lsp#StartProgram(
|
|
|
|
\ a:executable,
|
|
|
|
\ l:command,
|
|
|
|
\ function('s:HandleLSPResponse'),
|
|
|
|
\)
|
2017-06-14 15:53:21 +00:00
|
|
|
|
2017-07-01 13:30:34 +00:00
|
|
|
if !l:id
|
2017-06-14 15:53:21 +00:00
|
|
|
if g:ale_history_enabled
|
2017-07-01 13:30:34 +00:00
|
|
|
call ale#history#Add(a:buffer, 'failed', l:id, l:command)
|
2017-06-14 15:53:21 +00:00
|
|
|
endif
|
|
|
|
|
2017-07-07 22:47:41 +00:00
|
|
|
return 0
|
2017-06-14 15:53:21 +00:00
|
|
|
endif
|
2017-06-13 16:53:47 +00:00
|
|
|
|
2017-07-01 13:30:34 +00:00
|
|
|
if ale#lsp#OpenTSServerDocumentIfNeeded(l:id, a:buffer)
|
2017-06-14 15:53:21 +00:00
|
|
|
if g:ale_history_enabled
|
2017-07-01 13:30:34 +00:00
|
|
|
call ale#history#Add(a:buffer, 'started', l:id, l:command)
|
2017-06-14 15:53:21 +00:00
|
|
|
endif
|
2017-06-08 16:28:38 +00:00
|
|
|
endif
|
|
|
|
|
2017-07-01 13:30:34 +00:00
|
|
|
call ale#lsp#Send(l:id, ale#lsp#tsserver_message#Change(a:buffer))
|
2017-06-08 16:28:38 +00:00
|
|
|
|
2017-07-01 13:30:34 +00:00
|
|
|
let l:request_id = ale#lsp#Send(
|
|
|
|
\ l:id,
|
2017-06-13 16:53:47 +00:00
|
|
|
\ ale#lsp#tsserver_message#Geterr(a:buffer),
|
2017-06-08 16:28:38 +00:00
|
|
|
\)
|
|
|
|
|
2017-06-13 16:53:47 +00:00
|
|
|
if l:request_id != 0
|
|
|
|
let l:info.waiting_for_tsserver = 1
|
2017-06-08 16:28:38 +00:00
|
|
|
endif
|
2017-07-07 22:47:41 +00:00
|
|
|
|
|
|
|
return l:request_id != 0
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:RemoveProblemsForDisabledLinters(buffer, linters) abort
|
|
|
|
" Figure out which linters are still enabled, and remove
|
|
|
|
" problems for linters which are no longer enabled.
|
|
|
|
let l:name_map = {}
|
|
|
|
|
|
|
|
for l:linter in a:linters
|
|
|
|
let l:name_map[l:linter.name] = 1
|
|
|
|
endfor
|
|
|
|
|
|
|
|
call filter(
|
|
|
|
\ get(g:ale_buffer_info[a:buffer], 'loclist', []),
|
|
|
|
\ 'get(l:name_map, v:val.linter_name)',
|
|
|
|
\)
|
2017-06-08 16:28:38 +00:00
|
|
|
endfunction
|
2017-03-14 23:51:57 +00:00
|
|
|
|
2017-07-07 22:47:41 +00:00
|
|
|
" Run a linter for a buffer.
|
|
|
|
"
|
|
|
|
" Returns 1 if the linter was successfully run.
|
|
|
|
function! s:RunLinter(buffer, linter) abort
|
2017-06-08 16:28:38 +00:00
|
|
|
if empty(a:linter.lsp) || a:linter.lsp ==# 'tsserver'
|
2017-07-02 12:17:24 +00:00
|
|
|
let l:executable = ale#linter#GetExecutable(a:buffer, a:linter)
|
2017-06-08 16:28:38 +00:00
|
|
|
|
|
|
|
" Run this program if it can be executed.
|
|
|
|
if s:IsExecutable(l:executable)
|
|
|
|
if a:linter.lsp ==# 'tsserver'
|
2017-07-07 22:47:41 +00:00
|
|
|
return s:CheckWithTSServer(a:buffer, a:linter, l:executable)
|
2017-06-08 16:28:38 +00:00
|
|
|
endif
|
2017-07-07 22:47:41 +00:00
|
|
|
|
|
|
|
return s:InvokeChain(a:buffer, a:linter, 0, [])
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
|
|
|
|
return 0
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! ale#engine#RunLinters(buffer, linters, should_lint_file) abort
|
|
|
|
" Initialise the buffer information if needed.
|
|
|
|
call ale#engine#InitBufferInfo(a:buffer)
|
|
|
|
call s:StopCurrentJobs(a:buffer, a:should_lint_file)
|
|
|
|
call s:RemoveProblemsForDisabledLinters(a:buffer, a:linters)
|
|
|
|
|
|
|
|
let l:any_linter_ran = 0
|
|
|
|
|
|
|
|
for l:linter in a:linters
|
|
|
|
" Skip linters for checking files if we shouldn't check the file.
|
|
|
|
if l:linter.lint_file && !a:should_lint_file
|
|
|
|
continue
|
|
|
|
endif
|
|
|
|
|
|
|
|
if s:RunLinter(a:buffer, l:linter)
|
|
|
|
let l:any_linter_ran = 1
|
2017-06-08 16:28:38 +00:00
|
|
|
endif
|
2017-07-07 22:47:41 +00:00
|
|
|
endfor
|
|
|
|
|
|
|
|
" If we didn't manage to start checking the buffer with anything,
|
|
|
|
" and there's nothing running currently for the buffer, then clear the
|
|
|
|
" results.
|
|
|
|
"
|
|
|
|
" We need to use both checks, as we run some tests synchronously.
|
|
|
|
if !l:any_linter_ran && !ale#engine#IsCheckingBuffer(a:buffer)
|
|
|
|
call ale#engine#SetResults(a:buffer, [])
|
|
|
|
endif
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
" Clean up a buffer.
|
|
|
|
"
|
|
|
|
" This function will stop all current jobs for the buffer,
|
|
|
|
" clear the state of everything, and remove the Dictionary for managing
|
|
|
|
" the buffer.
|
|
|
|
function! ale#engine#Cleanup(buffer) abort
|
|
|
|
call ale#engine#RunLinters(a:buffer, [], 1)
|
|
|
|
|
|
|
|
if g:ale_set_highlights
|
|
|
|
call ale#highlight#UnqueueHighlights(a:buffer)
|
2017-07-09 23:02:49 +00:00
|
|
|
call ale#highlight#RemoveHighlights()
|
2017-03-14 23:51:57 +00:00
|
|
|
endif
|
2017-07-07 22:47:41 +00:00
|
|
|
|
|
|
|
call remove(g:ale_buffer_info, a:buffer)
|
2016-10-25 19:25:23 +00:00
|
|
|
endfunction
|
|
|
|
|
2016-10-24 19:21:32 +00:00
|
|
|
" Given a buffer number, return the warnings and errors for a given buffer.
|
|
|
|
function! ale#engine#GetLoclist(buffer) abort
|
|
|
|
if !has_key(g:ale_buffer_info, a:buffer)
|
|
|
|
return []
|
|
|
|
endif
|
|
|
|
|
|
|
|
return g:ale_buffer_info[a:buffer].loclist
|
|
|
|
endfunction
|
|
|
|
|
2016-10-17 22:26:19 +00:00
|
|
|
" This function can be called with a timeout to wait for all jobs to finish.
|
|
|
|
" If the jobs to not finish in the given number of milliseconds,
|
|
|
|
" an exception will be thrown.
|
|
|
|
"
|
|
|
|
" The time taken will be a very rough approximation, and more time may be
|
|
|
|
" permitted than is specified.
|
|
|
|
function! ale#engine#WaitForJobs(deadline) abort
|
2017-03-09 20:22:02 +00:00
|
|
|
let l:start_time = ale#util#ClockMilliseconds()
|
2016-10-17 22:43:31 +00:00
|
|
|
|
|
|
|
if l:start_time == 0
|
|
|
|
throw 'Failed to read milliseconds from the clock!'
|
|
|
|
endif
|
|
|
|
|
2016-10-17 22:26:19 +00:00
|
|
|
let l:job_list = []
|
|
|
|
|
2016-10-24 08:58:45 +00:00
|
|
|
" Gather all of the jobs from every buffer.
|
2016-10-23 21:41:00 +00:00
|
|
|
for l:info in values(g:ale_buffer_info)
|
|
|
|
call extend(l:job_list, l:info.job_list)
|
2016-10-17 22:26:19 +00:00
|
|
|
endfor
|
|
|
|
|
2017-05-12 19:38:52 +00:00
|
|
|
" NeoVim has a built-in API for this, so use that.
|
|
|
|
if has('nvim')
|
|
|
|
let l:nvim_code_list = jobwait(l:job_list, a:deadline)
|
|
|
|
|
|
|
|
if index(l:nvim_code_list, -1) >= 0
|
|
|
|
throw 'Jobs did not complete on time!'
|
|
|
|
endif
|
|
|
|
|
|
|
|
return
|
|
|
|
endif
|
|
|
|
|
2016-10-17 22:26:19 +00:00
|
|
|
let l:should_wait_more = 1
|
|
|
|
|
|
|
|
while l:should_wait_more
|
|
|
|
let l:should_wait_more = 0
|
|
|
|
|
2017-05-12 20:16:15 +00:00
|
|
|
for l:job_id in l:job_list
|
|
|
|
if ale#job#IsRunning(l:job_id)
|
2017-03-09 20:22:02 +00:00
|
|
|
let l:now = ale#util#ClockMilliseconds()
|
2016-10-17 22:43:31 +00:00
|
|
|
|
|
|
|
if l:now - l:start_time > a:deadline
|
2016-10-17 22:26:19 +00:00
|
|
|
" Stop waiting after a timeout, so we don't wait forever.
|
|
|
|
throw 'Jobs did not complete on time!'
|
|
|
|
endif
|
|
|
|
|
|
|
|
" Wait another 10 milliseconds
|
|
|
|
let l:should_wait_more = 1
|
|
|
|
sleep 10ms
|
|
|
|
break
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
endwhile
|
2016-10-17 22:43:31 +00:00
|
|
|
|
|
|
|
" Sleep for a small amount of time after all jobs finish.
|
|
|
|
" This seems to be enough to let handlers after jobs end run, and
|
|
|
|
" prevents the occasional failure where this function exits after jobs
|
|
|
|
" end, but before handlers are run.
|
|
|
|
sleep 10ms
|
2017-02-04 18:30:30 +00:00
|
|
|
|
|
|
|
" We must check the buffer data again to see if new jobs started
|
|
|
|
" for command_chain linters.
|
|
|
|
let l:has_new_jobs = 0
|
|
|
|
|
2017-03-21 14:52:02 +00:00
|
|
|
" Check again to see if any jobs are running.
|
2017-02-04 18:30:30 +00:00
|
|
|
for l:info in values(g:ale_buffer_info)
|
2017-05-12 20:16:15 +00:00
|
|
|
for l:job_id in l:info.job_list
|
|
|
|
if ale#job#IsRunning(l:job_id)
|
2017-03-21 14:52:02 +00:00
|
|
|
let l:has_new_jobs = 1
|
|
|
|
break
|
|
|
|
endif
|
|
|
|
endfor
|
2017-02-04 18:30:30 +00:00
|
|
|
endfor
|
|
|
|
|
|
|
|
if l:has_new_jobs
|
|
|
|
" We have to wait more. Offset the timeout by the time taken so far.
|
2017-03-09 20:22:02 +00:00
|
|
|
let l:now = ale#util#ClockMilliseconds()
|
2017-02-04 18:30:30 +00:00
|
|
|
let l:new_deadline = a:deadline - (l:now - l:start_time)
|
|
|
|
|
|
|
|
if l:new_deadline <= 0
|
|
|
|
" Enough time passed already, so stop immediately.
|
|
|
|
throw 'Jobs did not complete on time!'
|
|
|
|
endif
|
|
|
|
|
|
|
|
call ale#engine#WaitForJobs(l:new_deadline)
|
|
|
|
endif
|
2016-10-17 22:26:19 +00:00
|
|
|
endfunction
|