2017-06-15 04:30:34 -04:00
|
|
|
" Author: w0rp <devw0rp@gmail.com>, David Alexander <opensource@thelonelyghost.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 13:51:29 -05:00
|
|
|
" Description: Primary code path for the plugin
|
|
|
|
" Manages execution of linters when requested by autocommands
|
|
|
|
|
2018-05-28 19:19:20 +01:00
|
|
|
" Strings used for severity in the echoed message
|
|
|
|
let g:ale_echo_msg_error_str = get(g:, 'ale_echo_msg_error_str', 'Error')
|
|
|
|
let g:ale_echo_msg_info_str = get(g:, 'ale_echo_msg_info_str', 'Info')
|
|
|
|
let g:ale_echo_msg_warning_str = get(g:, 'ale_echo_msg_warning_str', 'Warning')
|
2018-06-24 21:16:45 +01:00
|
|
|
" Ignoring linters, for disabling some, or ignoring LSP diagnostics.
|
|
|
|
let g:ale_linters_ignore = get(g:, 'ale_linters_ignore', {})
|
2019-03-30 13:53:18 +09:00
|
|
|
let g:ale_disable_lsp = get(g:, 'ale_disable_lsp', 0)
|
2018-05-28 19:19:20 +01:00
|
|
|
|
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 13:51:29 -05:00
|
|
|
let s:lint_timer = -1
|
2018-10-29 11:29:12 +00:00
|
|
|
let s:getcmdwintype_exists = exists('*getcmdwintype')
|
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 13:51:29 -05:00
|
|
|
|
2017-05-26 21:21:15 +01:00
|
|
|
" Return 1 if a file is too large for ALE to handle.
|
2018-05-28 19:19:20 +01:00
|
|
|
function! ale#FileTooLarge(buffer) abort
|
|
|
|
let l:max = getbufvar(a:buffer, 'ale_maximum_file_size', get(g:, 'ale_maximum_file_size', 0))
|
2017-05-26 21:21:15 +01:00
|
|
|
|
|
|
|
return l:max > 0 ? (line2byte(line('$') + 1) > l:max) : 0
|
|
|
|
endfunction
|
|
|
|
|
2017-02-14 21:02:49 +00:00
|
|
|
" A function for checking various conditions whereby ALE just shouldn't
|
|
|
|
" attempt to do anything, say if particular buffer types are open in Vim.
|
2017-08-01 00:03:24 +01:00
|
|
|
function! ale#ShouldDoNothing(buffer) abort
|
2017-10-14 16:51:12 +01:00
|
|
|
" The checks are split into separate if statements to make it possible to
|
|
|
|
" profile each check individually with Vim's profiling tools.
|
2018-09-04 16:51:18 +01:00
|
|
|
"
|
2018-04-12 20:31:33 +01:00
|
|
|
" Do nothing if ALE is disabled.
|
|
|
|
if !getbufvar(a:buffer, 'ale_enabled', get(g:, 'ale_enabled', 0))
|
|
|
|
return 1
|
|
|
|
endif
|
|
|
|
|
2017-10-14 19:22:19 +01:00
|
|
|
" Don't perform any checks when newer NeoVim versions are exiting.
|
|
|
|
if get(v:, 'exiting', v:null) isnot v:null
|
|
|
|
return 1
|
|
|
|
endif
|
|
|
|
|
2018-05-28 17:38:14 +01:00
|
|
|
let l:filetype = getbufvar(a:buffer, '&filetype')
|
|
|
|
|
|
|
|
" Do nothing when there's no filetype.
|
|
|
|
if l:filetype is# ''
|
|
|
|
return 1
|
|
|
|
endif
|
|
|
|
|
|
|
|
" Do nothing for blacklisted files.
|
|
|
|
if index(get(g:, 'ale_filetype_blacklist', []), l:filetype) >= 0
|
2017-10-14 16:51:12 +01:00
|
|
|
return 1
|
|
|
|
endif
|
|
|
|
|
2018-05-28 17:38:14 +01:00
|
|
|
" Do nothing if running from command mode.
|
2017-10-14 17:31:58 +01:00
|
|
|
if s:getcmdwintype_exists && !empty(getcmdwintype())
|
2017-10-14 16:51:12 +01:00
|
|
|
return 1
|
|
|
|
endif
|
|
|
|
|
2017-10-26 20:29:23 +01:00
|
|
|
let l:filename = fnamemodify(bufname(a:buffer), ':t')
|
|
|
|
|
2018-05-28 17:38:14 +01:00
|
|
|
" Do nothing for directories.
|
2017-10-26 20:29:23 +01:00
|
|
|
if l:filename is# '.'
|
|
|
|
return 1
|
|
|
|
endif
|
|
|
|
|
2018-09-03 13:59:56 +01:00
|
|
|
" Don't start linting and so on when an operator is pending.
|
|
|
|
if ale#util#Mode(1) is# 'no'
|
|
|
|
return 1
|
|
|
|
endif
|
|
|
|
|
2018-05-28 17:38:14 +01:00
|
|
|
" Do nothing if running in the sandbox.
|
2017-10-14 16:51:12 +01:00
|
|
|
if ale#util#InSandbox()
|
|
|
|
return 1
|
|
|
|
endif
|
|
|
|
|
|
|
|
" Do nothing if the file is too large.
|
2018-05-28 19:19:20 +01:00
|
|
|
if ale#FileTooLarge(a:buffer)
|
2017-10-14 16:51:12 +01:00
|
|
|
return 1
|
|
|
|
endif
|
|
|
|
|
|
|
|
" Do nothing from CtrlP buffers with CtrlP-funky.
|
2017-10-14 17:11:30 +01:00
|
|
|
if exists(':CtrlPFunky') is 2
|
|
|
|
\&& getbufvar(a:buffer, '&l:statusline') =~# 'CtrlPMode.*funky'
|
2017-10-14 16:51:12 +01:00
|
|
|
return 1
|
|
|
|
endif
|
|
|
|
|
|
|
|
return 0
|
2017-02-14 21:02:49 +00:00
|
|
|
endfunction
|
|
|
|
|
2018-10-29 11:29:12 +00:00
|
|
|
function! s:Lint(buffer, should_lint_file, timer_id) abort
|
|
|
|
" Use the filetype from the buffer
|
|
|
|
let l:filetype = getbufvar(a:buffer, '&filetype')
|
|
|
|
let l:linters = ale#linter#Get(l:filetype)
|
|
|
|
|
|
|
|
" Apply ignore lists for linters only if needed.
|
|
|
|
let l:ignore_config = ale#Var(a:buffer, 'linters_ignore')
|
2019-03-30 13:53:18 +09:00
|
|
|
let l:disable_lsp = ale#Var(a:buffer, 'disable_lsp')
|
2018-10-29 11:29:12 +00:00
|
|
|
let l:linters = !empty(l:ignore_config)
|
2019-03-30 13:53:18 +09:00
|
|
|
\ ? ale#engine#ignore#Exclude(l:filetype, l:linters, l:ignore_config, l:disable_lsp)
|
2018-10-29 11:29:12 +00:00
|
|
|
\ : l:linters
|
|
|
|
|
2018-10-29 18:28:28 +00:00
|
|
|
" Tell other sources that they can start checking the buffer now.
|
|
|
|
let g:ale_want_results_buffer = a:buffer
|
|
|
|
silent doautocmd <nomodeline> User ALEWantResults
|
|
|
|
unlet! g:ale_want_results_buffer
|
|
|
|
|
2018-10-29 11:29:12 +00:00
|
|
|
" Don't set up buffer data and so on if there are no linters to run.
|
|
|
|
if !has_key(g:ale_buffer_info, a:buffer) && empty(l:linters)
|
|
|
|
return
|
|
|
|
endif
|
|
|
|
|
|
|
|
" Clear lint_file linters, or only run them if the file exists.
|
|
|
|
let l:lint_file = empty(l:linters)
|
|
|
|
\ || (a:should_lint_file && filereadable(expand('#' . a:buffer . ':p')))
|
|
|
|
|
|
|
|
call ale#engine#RunLinters(a:buffer, l:linters, l:lint_file)
|
|
|
|
endfunction
|
|
|
|
|
2017-08-01 00:03:24 +01:00
|
|
|
" (delay, [linting_flag, buffer_number])
|
2017-03-14 23:51:57 +00:00
|
|
|
function! ale#Queue(delay, ...) abort
|
2017-08-01 00:03:24 +01:00
|
|
|
if a:0 > 2
|
2017-03-14 23:51:57 +00:00
|
|
|
throw 'too many arguments!'
|
|
|
|
endif
|
|
|
|
|
2018-10-29 11:29:12 +00:00
|
|
|
let l:buffer = get(a:000, 1, v:null)
|
2017-08-09 00:05:55 +01:00
|
|
|
|
2018-10-29 11:29:12 +00:00
|
|
|
if l:buffer is v:null
|
|
|
|
let l:buffer = bufnr('')
|
2017-03-14 23:51:57 +00:00
|
|
|
endif
|
|
|
|
|
2018-07-25 01:27:28 +01:00
|
|
|
if type(l:buffer) isnot v:t_number
|
2017-08-01 00:03:24 +01:00
|
|
|
throw 'buffer_number must be a Number'
|
|
|
|
endif
|
|
|
|
|
2018-06-20 08:34:42 +01:00
|
|
|
if ale#ShouldDoNothing(l:buffer)
|
2016-10-31 14:47:08 +00:00
|
|
|
return
|
|
|
|
endif
|
|
|
|
|
2018-10-29 11:29:12 +00:00
|
|
|
" Default linting_flag to ''
|
|
|
|
let l:should_lint_file = get(a:000, 0) is# 'lint_file'
|
2017-03-14 23:51:57 +00:00
|
|
|
|
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 13:51:29 -05:00
|
|
|
if s:lint_timer != -1
|
|
|
|
call timer_stop(s:lint_timer)
|
|
|
|
let s:lint_timer = -1
|
|
|
|
endif
|
|
|
|
|
|
|
|
if a:delay > 0
|
2018-10-29 11:29:12 +00:00
|
|
|
let s:lint_timer = timer_start(
|
|
|
|
\ a:delay,
|
|
|
|
\ function('s:Lint', [l:buffer, l:should_lint_file])
|
|
|
|
\)
|
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 13:51:29 -05:00
|
|
|
else
|
2018-10-29 11:29:12 +00:00
|
|
|
call s:Lint(l:buffer, l:should_lint_file, 0)
|
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 13:51:29 -05:00
|
|
|
endif
|
|
|
|
endfunction
|
|
|
|
|
2019-04-07 15:34:39 +01:00
|
|
|
let s:current_ale_version = [2, 4, 0]
|
2017-05-05 23:03:19 +01:00
|
|
|
|
2019-04-07 15:34:39 +01:00
|
|
|
" A function used to check for ALE features in files outside of the project.
|
2017-05-05 23:03:19 +01:00
|
|
|
function! ale#Has(feature) abort
|
2019-04-07 15:34:39 +01:00
|
|
|
let l:match = matchlist(a:feature, '\c\v^ale-(\d+)\.(\d+)(\.(\d+))?$')
|
|
|
|
|
|
|
|
if !empty(l:match)
|
|
|
|
let l:version = [l:match[1] + 0, l:match[2] + 0, l:match[4] + 0]
|
|
|
|
|
|
|
|
return ale#semver#GTE(s:current_ale_version, l:version)
|
|
|
|
endif
|
|
|
|
|
|
|
|
return 0
|
2017-05-05 23:03:19 +01:00
|
|
|
endfunction
|
|
|
|
|
2017-04-16 00:16:48 +01:00
|
|
|
" Given a buffer number and a variable name, look for that variable in the
|
|
|
|
" buffer scope, then in global scope. If the name does not exist in the global
|
|
|
|
" scope, an exception will be thrown.
|
|
|
|
"
|
|
|
|
" Every variable name will be prefixed with 'ale_'.
|
|
|
|
function! ale#Var(buffer, variable_name) abort
|
|
|
|
let l:full_name = 'ale_' . a:variable_name
|
2018-07-22 22:31:46 +01:00
|
|
|
let l:vars = getbufvar(str2nr(a:buffer), '', {})
|
2017-05-30 22:15:24 +01:00
|
|
|
|
|
|
|
return get(l:vars, l:full_name, g:[l:full_name])
|
2017-04-16 00:16:48 +01:00
|
|
|
endfunction
|
2017-05-12 09:20:16 +01:00
|
|
|
|
2017-05-20 23:32:41 +01:00
|
|
|
" Initialize a variable with a default value, if it isn't already set.
|
|
|
|
"
|
|
|
|
" Every variable name will be prefixed with 'ale_'.
|
|
|
|
function! ale#Set(variable_name, default) abort
|
|
|
|
let l:full_name = 'ale_' . a:variable_name
|
|
|
|
|
2018-07-05 00:40:15 +01:00
|
|
|
if !has_key(g:, l:full_name)
|
|
|
|
let g:[l:full_name] = a:default
|
|
|
|
endif
|
2017-05-20 23:32:41 +01:00
|
|
|
endfunction
|
|
|
|
|
2018-07-15 18:24:53 +01:00
|
|
|
" Given a string for adding to a command, return the string padded with a
|
|
|
|
" space on the left if it is not empty. Otherwise return an empty string.
|
|
|
|
"
|
|
|
|
" This can be used for making command strings cleaner and easier to test.
|
|
|
|
function! ale#Pad(string) abort
|
|
|
|
return !empty(a:string) ? ' ' . a:string : ''
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
" Given a environment variable name and a value, produce part of a command for
|
|
|
|
" setting an environment variable before running a command. The syntax will be
|
|
|
|
" valid for cmd on Windows, or most shells on Unix.
|
|
|
|
function! ale#Env(variable_name, value) abort
|
|
|
|
if has('win32')
|
|
|
|
return 'set ' . a:variable_name . '=' . ale#Escape(a:value) . ' && '
|
|
|
|
endif
|
|
|
|
|
|
|
|
return a:variable_name . '=' . ale#Escape(a:value) . ' '
|
|
|
|
endfunction
|
|
|
|
|
2017-05-12 09:20:16 +01:00
|
|
|
" Escape a string suitably for each platform.
|
2017-05-15 20:21:18 +01:00
|
|
|
" shellescape does not work on Windows.
|
2017-05-12 09:20:16 +01:00
|
|
|
function! ale#Escape(str) abort
|
2017-08-08 08:39:13 +01:00
|
|
|
if fnamemodify(&shell, ':t') is? 'cmd.exe'
|
2017-06-14 11:05:49 +01:00
|
|
|
" If the string contains spaces, it will be surrounded by quotes.
|
|
|
|
" Otherwise, special characters will be escaped with carets (^).
|
|
|
|
return substitute(
|
|
|
|
\ a:str =~# ' '
|
|
|
|
\ ? '"' . substitute(a:str, '"', '""', 'g') . '"'
|
|
|
|
\ : substitute(a:str, '\v([&|<>^])', '^\1', 'g'),
|
|
|
|
\ '%',
|
|
|
|
\ '%%',
|
|
|
|
\ 'g',
|
|
|
|
\)
|
2017-05-12 09:20:16 +01:00
|
|
|
endif
|
2017-05-26 00:06:16 +01:00
|
|
|
|
|
|
|
return shellescape (a:str)
|
2017-05-12 09:20:16 +01:00
|
|
|
endfunction
|
2017-11-14 10:28:36 +00:00
|
|
|
|
|
|
|
" Get the loclist item message according to a given format string.
|
|
|
|
"
|
|
|
|
" See `:help g:ale_loclist_msg_format` and `:help g:ale_echo_msg_format`
|
|
|
|
function! ale#GetLocItemMessage(item, format_string) abort
|
|
|
|
let l:msg = a:format_string
|
|
|
|
let l:severity = g:ale_echo_msg_warning_str
|
|
|
|
let l:code = get(a:item, 'code', '')
|
|
|
|
let l:type = get(a:item, 'type', 'E')
|
|
|
|
let l:linter_name = get(a:item, 'linter_name', '')
|
|
|
|
let l:code_repl = !empty(l:code) ? '\=submatch(1) . l:code . submatch(2)' : ''
|
|
|
|
|
|
|
|
if l:type is# 'E'
|
|
|
|
let l:severity = g:ale_echo_msg_error_str
|
|
|
|
elseif l:type is# 'I'
|
|
|
|
let l:severity = g:ale_echo_msg_info_str
|
|
|
|
endif
|
|
|
|
|
|
|
|
" Replace special markers with certain information.
|
|
|
|
" \=l:variable is used to avoid escaping issues.
|
|
|
|
let l:msg = substitute(l:msg, '\V%severity%', '\=l:severity', 'g')
|
|
|
|
let l:msg = substitute(l:msg, '\V%linter%', '\=l:linter_name', 'g')
|
|
|
|
let l:msg = substitute(l:msg, '\v\%([^\%]*)code([^\%]*)\%', l:code_repl, 'g')
|
|
|
|
" Replace %s with the text.
|
|
|
|
let l:msg = substitute(l:msg, '\V%s', '\=a:item.text', 'g')
|
|
|
|
|
|
|
|
return l:msg
|
|
|
|
endfunction
|