2018-03-31 10:55:20 -04:00
|
|
|
" Author: Zach Perrault -- @zperrault
|
|
|
|
" Author: Florian Beeres <yuuki@protonmail.com>
|
|
|
|
" Description: FlowType checking for JavaScript files
|
|
|
|
|
|
|
|
call ale#Set('javascript_flow_executable', 'flow')
|
|
|
|
call ale#Set('javascript_flow_use_home_config', 0)
|
2018-06-14 06:31:12 -04:00
|
|
|
call ale#Set('javascript_flow_use_global', get(g:, 'ale_use_global_executables', 0))
|
|
|
|
call ale#Set('javascript_flow_use_respect_pragma', 1)
|
2018-03-31 10:55:20 -04:00
|
|
|
|
|
|
|
function! ale_linters#javascript#flow#GetExecutable(buffer) abort
|
|
|
|
let l:flow_config = ale#path#FindNearestFile(a:buffer, '.flowconfig')
|
|
|
|
|
|
|
|
if empty(l:flow_config)
|
|
|
|
" Don't run Flow if we can't find a .flowconfig file.
|
|
|
|
return ''
|
|
|
|
endif
|
|
|
|
|
|
|
|
" Don't run Flow with a configuration file from the home directory by
|
|
|
|
" default, which can eat all of your RAM.
|
|
|
|
if fnamemodify(l:flow_config, ':h') is? $HOME
|
|
|
|
\&& !ale#Var(a:buffer, 'javascript_flow_use_home_config')
|
|
|
|
return ''
|
|
|
|
endif
|
|
|
|
|
2021-05-31 16:30:26 -04:00
|
|
|
return ale#path#FindExecutable(a:buffer, 'javascript_flow', [
|
2018-03-31 10:55:20 -04:00
|
|
|
\ 'node_modules/.bin/flow',
|
|
|
|
\])
|
|
|
|
endfunction
|
|
|
|
|
2019-05-17 10:09:13 -04:00
|
|
|
function! ale_linters#javascript#flow#GetCommand(buffer, version) abort
|
2018-03-31 10:55:20 -04:00
|
|
|
" If we can parse the version number, then only use --respect-pragma
|
|
|
|
" if the version is >= 0.36.0, which added the argument.
|
2018-06-14 06:31:12 -04:00
|
|
|
let l:use_respect_pragma = ale#Var(a:buffer, 'javascript_flow_use_respect_pragma')
|
2019-05-17 10:09:13 -04:00
|
|
|
\ && (empty(a:version) || ale#semver#GTE(a:version, [0, 36]))
|
2018-03-31 10:55:20 -04:00
|
|
|
|
2019-05-17 10:09:13 -04:00
|
|
|
return '%e check-contents'
|
2018-03-31 10:55:20 -04:00
|
|
|
\ . (l:use_respect_pragma ? ' --respect-pragma': '')
|
2019-03-08 06:04:56 -05:00
|
|
|
\ . ' --json --from ale %s < %t'
|
|
|
|
\ . (!has('win32') ? '; echo' : '')
|
2018-03-31 10:55:20 -04:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
" Filter lines of flow output until we find the first line where the JSON
|
|
|
|
" output starts.
|
|
|
|
function! s:GetJSONLines(lines) abort
|
|
|
|
let l:start_index = 0
|
|
|
|
|
|
|
|
for l:line in a:lines
|
|
|
|
if l:line[:0] is# '{'
|
|
|
|
break
|
|
|
|
endif
|
|
|
|
|
|
|
|
let l:start_index += 1
|
|
|
|
endfor
|
|
|
|
|
|
|
|
return a:lines[l:start_index :]
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:ExtraErrorMsg(current, new) abort
|
|
|
|
let l:newMsg = ''
|
|
|
|
|
|
|
|
if a:current is# ''
|
|
|
|
" extra messages appear to already have a :
|
|
|
|
let l:newMsg = a:new
|
|
|
|
else
|
|
|
|
let l:newMsg = a:current . ' ' . a:new
|
|
|
|
endif
|
|
|
|
|
|
|
|
return l:newMsg
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:GetDetails(error) abort
|
|
|
|
let l:detail = ''
|
|
|
|
|
|
|
|
for l:extra_error in a:error.extra
|
|
|
|
if has_key(l:extra_error, 'message')
|
|
|
|
for l:extra_message in l:extra_error.message
|
|
|
|
let l:detail = s:ExtraErrorMsg(l:detail, l:extra_message.descr)
|
|
|
|
endfor
|
|
|
|
endif
|
|
|
|
|
|
|
|
if has_key(l:extra_error, 'children')
|
|
|
|
for l:child in l:extra_error.children
|
|
|
|
for l:child_message in l:child.message
|
|
|
|
let l:detail = l:detail . ' ' . l:child_message.descr
|
|
|
|
endfor
|
|
|
|
endfor
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
|
|
|
|
return l:detail
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! ale_linters#javascript#flow#Handle(buffer, lines) abort
|
|
|
|
let l:str = join(s:GetJSONLines(a:lines), '')
|
|
|
|
|
|
|
|
if empty(l:str)
|
|
|
|
return []
|
|
|
|
endif
|
|
|
|
|
|
|
|
let l:flow_output = json_decode(l:str)
|
|
|
|
let l:output = []
|
|
|
|
|
|
|
|
for l:error in get(l:flow_output, 'errors', [])
|
|
|
|
" Each error is broken up into parts
|
|
|
|
let l:text = ''
|
|
|
|
let l:line = 0
|
|
|
|
let l:col = 0
|
|
|
|
|
|
|
|
for l:message in l:error.message
|
|
|
|
" Comments have no line of column information, so we skip them.
|
|
|
|
" In certain cases, `l:message.loc.source` points to a different path
|
|
|
|
" than the buffer one, thus we skip this loc information too.
|
|
|
|
if has_key(l:message, 'loc')
|
|
|
|
\&& l:line is# 0
|
|
|
|
\&& ale#path#IsBufferPath(a:buffer, l:message.loc.source)
|
|
|
|
let l:line = l:message.loc.start.line + 0
|
|
|
|
let l:col = l:message.loc.start.column + 0
|
|
|
|
endif
|
|
|
|
|
|
|
|
if l:text is# ''
|
|
|
|
let l:text = l:message.descr . ':'
|
|
|
|
else
|
|
|
|
let l:text = l:text . ' ' . l:message.descr
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
|
|
|
|
if has_key(l:error, 'operation')
|
|
|
|
let l:text = l:text . ' See also: ' . l:error.operation.descr
|
|
|
|
endif
|
|
|
|
|
|
|
|
let l:errorToAdd = {
|
|
|
|
\ 'lnum': l:line,
|
|
|
|
\ 'col': l:col,
|
|
|
|
\ 'text': l:text,
|
|
|
|
\ 'type': has_key(l:error, 'level') && l:error.level is# 'error' ? 'E' : 'W',
|
|
|
|
\}
|
|
|
|
|
|
|
|
if has_key(l:error, 'extra')
|
2019-03-08 06:04:56 -05:00
|
|
|
let l:errorToAdd.detail = l:errorToAdd.text
|
|
|
|
\ . "\n" . s:GetDetails(l:error)
|
2018-03-31 10:55:20 -04:00
|
|
|
endif
|
|
|
|
|
|
|
|
call add(l:output, l:errorToAdd)
|
|
|
|
endfor
|
|
|
|
|
|
|
|
return l:output
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
call ale#linter#Define('javascript', {
|
|
|
|
\ 'name': 'flow',
|
2019-03-08 06:04:56 -05:00
|
|
|
\ 'executable': function('ale_linters#javascript#flow#GetExecutable'),
|
2019-05-17 10:09:13 -04:00
|
|
|
\ 'command': {buffer -> ale#semver#RunWithVersionCheck(
|
|
|
|
\ buffer,
|
|
|
|
\ ale_linters#javascript#flow#GetExecutable(buffer),
|
|
|
|
\ '%e --version',
|
|
|
|
\ function('ale_linters#javascript#flow#GetCommand'),
|
|
|
|
\ )},
|
2018-03-31 10:55:20 -04:00
|
|
|
\ 'callback': 'ale_linters#javascript#flow#Handle',
|
2019-03-08 06:04:56 -05:00
|
|
|
\ 'read_buffer': 0,
|
2018-03-31 10:55:20 -04:00
|
|
|
\})
|