2018-06-14 06:31:12 -04:00
|
|
|
" Author: buffalocoder - https://github.com/buffalocoder, soywod - https://github.com/soywod, hecrj - https://github.com/hecrj
|
2018-03-31 10:55:20 -04:00
|
|
|
" Description: Elm linting in Ale. Closely follows the Syntastic checker in https://github.com/ElmCast/elm-vim.
|
|
|
|
|
2018-06-14 06:31:12 -04:00
|
|
|
call ale#Set('elm_make_executable', 'elm')
|
|
|
|
call ale#Set('elm_make_use_global', get(g:, 'ale_use_global_executables', 0))
|
2018-03-31 10:55:20 -04:00
|
|
|
|
|
|
|
function! ale_linters#elm#make#GetExecutable(buffer) abort
|
|
|
|
return ale#node#FindExecutable(a:buffer, 'elm_make', [
|
2018-06-14 06:31:12 -04:00
|
|
|
\ 'node_modules/.bin/elm',
|
2018-03-31 10:55:20 -04:00
|
|
|
\])
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! ale_linters#elm#make#Handle(buffer, lines) abort
|
|
|
|
let l:output = []
|
|
|
|
let l:unparsed_lines = []
|
|
|
|
|
2018-06-14 06:31:12 -04:00
|
|
|
for l:line in a:lines
|
|
|
|
if l:line[0] is# '{'
|
|
|
|
" Elm 0.19
|
|
|
|
call ale_linters#elm#make#HandleElm019Line(l:line, l:output)
|
|
|
|
elseif l:line[0] is# '['
|
|
|
|
" Elm 0.18
|
|
|
|
call ale_linters#elm#make#HandleElm018Line(l:line, l:output)
|
2018-03-31 10:55:20 -04:00
|
|
|
elseif l:line isnot# 'Successfully generated /dev/null'
|
|
|
|
call add(l:unparsed_lines, l:line)
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
|
|
|
|
if len(l:unparsed_lines) > 0
|
|
|
|
call add(l:output, {
|
|
|
|
\ 'lnum': 1,
|
|
|
|
\ 'type': 'E',
|
|
|
|
\ 'text': l:unparsed_lines[0],
|
|
|
|
\ 'detail': join(l:unparsed_lines, "\n")
|
|
|
|
\})
|
|
|
|
endif
|
|
|
|
|
|
|
|
return l:output
|
|
|
|
endfunction
|
|
|
|
|
2018-06-14 06:31:12 -04:00
|
|
|
function! ale_linters#elm#make#HandleElm019Line(line, output) abort
|
|
|
|
let l:report = json_decode(a:line)
|
|
|
|
|
|
|
|
if l:report.type is? 'error'
|
|
|
|
" General problem
|
|
|
|
let l:details = ale_linters#elm#make#ParseMessage(l:report.message)
|
|
|
|
|
|
|
|
if empty(l:report.path)
|
|
|
|
let l:report.path = 'Elm'
|
|
|
|
endif
|
|
|
|
|
|
|
|
if ale_linters#elm#make#FileIsBuffer(l:report.path)
|
|
|
|
call add(a:output, {
|
|
|
|
\ 'lnum': 1,
|
|
|
|
\ 'type': 'E',
|
|
|
|
\ 'text': l:details,
|
|
|
|
\})
|
|
|
|
else
|
|
|
|
call add(a:output, {
|
|
|
|
\ 'lnum': 1,
|
|
|
|
\ 'type': 'E',
|
|
|
|
\ 'text': l:report.path .' - '. l:details,
|
|
|
|
\ 'detail': l:report.path ." ----------\n\n". l:details,
|
|
|
|
\})
|
|
|
|
endif
|
|
|
|
else
|
|
|
|
" Compilation errors
|
|
|
|
for l:error in l:report.errors
|
|
|
|
let l:file_is_buffer = ale_linters#elm#make#FileIsBuffer(l:error.path)
|
|
|
|
|
|
|
|
for l:problem in l:error.problems
|
|
|
|
let l:details = ale_linters#elm#make#ParseMessage(l:problem.message)
|
|
|
|
|
|
|
|
if l:file_is_buffer
|
|
|
|
" Buffer module has problems
|
|
|
|
call add(a:output, {
|
|
|
|
\ 'lnum': l:problem.region.start.line,
|
|
|
|
\ 'col': l:problem.region.start.column,
|
|
|
|
\ 'end_lnum': l:problem.region.end.line,
|
|
|
|
\ 'end_col': l:problem.region.end.column,
|
|
|
|
\ 'type': 'E',
|
|
|
|
\ 'text': l:details,
|
|
|
|
\})
|
|
|
|
else
|
|
|
|
" Imported module has problems
|
|
|
|
let l:location = l:error.path .':'. l:problem.region.start.line
|
|
|
|
call add(a:output, {
|
|
|
|
\ 'lnum': 1,
|
|
|
|
\ 'type': 'E',
|
|
|
|
\ 'text': l:location .' - '. l:details,
|
|
|
|
\ 'detail': l:location ." ----------\n\n". l:details,
|
|
|
|
\})
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
endfor
|
|
|
|
endif
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! ale_linters#elm#make#HandleElm018Line(line, output) abort
|
|
|
|
let l:errors = json_decode(a:line)
|
|
|
|
|
|
|
|
for l:error in l:errors
|
|
|
|
let l:file_is_buffer = ale_linters#elm#make#FileIsBuffer(l:error.file)
|
|
|
|
|
|
|
|
if l:file_is_buffer
|
|
|
|
" Current buffer has problems
|
|
|
|
call add(a:output, {
|
|
|
|
\ 'lnum': l:error.region.start.line,
|
|
|
|
\ 'col': l:error.region.start.column,
|
|
|
|
\ 'end_lnum': l:error.region.end.line,
|
|
|
|
\ 'end_col': l:error.region.end.column,
|
|
|
|
\ 'type': (l:error.type is? 'error') ? 'E' : 'W',
|
|
|
|
\ 'text': l:error.overview,
|
|
|
|
\ 'detail': l:error.overview . "\n\n" . l:error.details
|
|
|
|
\})
|
|
|
|
elseif l:error.type is? 'error'
|
|
|
|
" Imported module has errors
|
|
|
|
let l:location = l:error.file .':'. l:error.region.start.line
|
|
|
|
|
|
|
|
call add(a:output, {
|
|
|
|
\ 'lnum': 1,
|
|
|
|
\ 'type': 'E',
|
|
|
|
\ 'text': l:location .' - '. l:error.overview,
|
|
|
|
\ 'detail': l:location ." ----------\n\n". l:error.overview . "\n\n" . l:error.details
|
|
|
|
\})
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! ale_linters#elm#make#FileIsBuffer(path) abort
|
2018-07-19 08:52:53 -04:00
|
|
|
return ale#path#IsTempName(a:path)
|
2018-06-14 06:31:12 -04:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! ale_linters#elm#make#ParseMessage(message) abort
|
|
|
|
return join(map(copy(a:message), 'ale_linters#elm#make#ParseMessageItem(v:val)'), '')
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! ale_linters#elm#make#ParseMessageItem(item) abort
|
2018-07-30 17:18:16 -04:00
|
|
|
if type(a:item) is v:t_string
|
2018-06-14 06:31:12 -04:00
|
|
|
return a:item
|
|
|
|
else
|
|
|
|
return a:item.string
|
|
|
|
endif
|
|
|
|
endfunction
|
|
|
|
|
2018-03-31 10:55:20 -04:00
|
|
|
" Return the command to execute the linter in the projects directory.
|
|
|
|
" If it doesn't, then this will fail when imports are needed.
|
|
|
|
function! ale_linters#elm#make#GetCommand(buffer) abort
|
2018-06-14 06:31:12 -04:00
|
|
|
let l:elm_json = ale#path#FindNearestFile(a:buffer, 'elm.json')
|
2018-03-31 10:55:20 -04:00
|
|
|
let l:elm_exe = ale_linters#elm#make#GetExecutable(a:buffer)
|
2018-06-14 06:31:12 -04:00
|
|
|
|
|
|
|
if empty(l:elm_json)
|
|
|
|
" Fallback to Elm 0.18
|
|
|
|
let l:elm_json = ale#path#FindNearestFile(a:buffer, 'elm-package.json')
|
|
|
|
endif
|
|
|
|
|
|
|
|
if empty(l:elm_json)
|
2018-03-31 10:55:20 -04:00
|
|
|
let l:dir_set_cmd = ''
|
|
|
|
else
|
2018-06-14 06:31:12 -04:00
|
|
|
let l:root_dir = fnamemodify(l:elm_json, ':p:h')
|
2018-03-31 10:55:20 -04:00
|
|
|
let l:dir_set_cmd = 'cd ' . ale#Escape(l:root_dir) . ' && '
|
|
|
|
endif
|
|
|
|
|
2018-06-14 06:31:12 -04:00
|
|
|
" The elm compiler, at the time of this writing, uses '/dev/null' as
|
2018-03-31 10:55:20 -04:00
|
|
|
" a sort of flag to tell the compiler not to generate an output file,
|
2018-06-14 06:31:12 -04:00
|
|
|
" which is why this is hard coded here.
|
|
|
|
" Source: https://github.com/elm-lang/elm-compiler/blob/19d5a769b30ec0b2fc4475985abb4cd94cd1d6c3/builder/src/Generate/Output.hs#L253
|
2018-03-31 10:55:20 -04:00
|
|
|
let l:elm_cmd = ale#Escape(l:elm_exe)
|
2018-06-14 06:31:12 -04:00
|
|
|
\ . ' make'
|
2018-03-31 10:55:20 -04:00
|
|
|
\ . ' --report=json'
|
|
|
|
\ . ' --output=/dev/null'
|
|
|
|
|
|
|
|
return l:dir_set_cmd . ' ' . l:elm_cmd . ' %t'
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
call ale#linter#Define('elm', {
|
|
|
|
\ 'name': 'make',
|
|
|
|
\ 'executable_callback': 'ale_linters#elm#make#GetExecutable',
|
|
|
|
\ 'output_stream': 'both',
|
|
|
|
\ 'command_callback': 'ale_linters#elm#make#GetCommand',
|
|
|
|
\ 'callback': 'ale_linters#elm#make#Handle'
|
|
|
|
\})
|