mirror of
1
0
Fork 0
ultimate-vim/sources_non_forked/vim-go/autoload/go/lint.vim

358 lines
10 KiB
VimL
Raw Normal View History

2015-12-08 08:20:04 -05:00
if !exists("g:go_metalinter_command")
2016-06-26 07:12:36 -04:00
let g:go_metalinter_command = ""
2015-12-08 08:20:04 -05:00
endif
if !exists("g:go_metalinter_autosave_enabled")
2016-06-26 07:12:36 -04:00
let g:go_metalinter_autosave_enabled = ['vet', 'golint']
2015-12-08 08:20:04 -05:00
endif
if !exists("g:go_metalinter_enabled")
2016-06-26 07:12:36 -04:00
let g:go_metalinter_enabled = ['vet', 'golint', 'errcheck']
2015-12-08 08:20:04 -05:00
endif
2018-02-04 06:35:08 -05:00
if !exists("g:go_metalinter_disabled")
let g:go_metalinter_disabled = []
2017-05-02 08:42:08 -04:00
endif
2014-10-31 17:30:24 -04:00
if !exists("g:go_golint_bin")
2016-06-26 07:12:36 -04:00
let g:go_golint_bin = "golint"
2014-10-31 17:30:24 -04:00
endif
2015-12-08 08:20:04 -05:00
if !exists("g:go_errcheck_bin")
2016-06-26 07:12:36 -04:00
let g:go_errcheck_bin = "errcheck"
2015-12-08 08:20:04 -05:00
endif
function! go#lint#Gometa(autosave, ...) abort
2016-06-26 07:12:36 -04:00
if a:0 == 0
2018-02-04 06:35:08 -05:00
let goargs = [expand('%:p:h')]
2016-06-26 07:12:36 -04:00
else
2018-02-04 06:35:08 -05:00
let goargs = a:000
2016-06-26 07:12:36 -04:00
endif
2016-12-27 09:46:49 -05:00
let bin_path = go#path#CheckBinPath("gometalinter")
if empty(bin_path)
return
endif
2015-12-08 08:20:04 -05:00
2016-12-27 09:46:49 -05:00
let cmd = [bin_path]
let cmd += ["--disable-all"]
2015-12-08 08:20:04 -05:00
2016-12-27 09:46:49 -05:00
if a:autosave || empty(g:go_metalinter_command)
2016-06-26 07:12:36 -04:00
" linters
let linters = a:autosave ? g:go_metalinter_autosave_enabled : g:go_metalinter_enabled
for linter in linters
2016-12-27 09:46:49 -05:00
let cmd += ["--enable=".linter]
2016-06-26 07:12:36 -04:00
endfor
2018-02-04 06:35:08 -05:00
for linter in g:go_metalinter_disabled
let cmd += ["--disable=".linter]
2017-05-02 08:42:08 -04:00
endfor
" gometalinter has a --tests flag to tell its linters whether to run
" against tests. While not all of its linters respect this flag, for those
" that do, it means if we don't pass --tests, the linter won't run against
" test files. One example of a linter that will not run against tests if
" we do not specify this flag is errcheck.
let cmd += ["--tests"]
2016-06-26 07:12:36 -04:00
else
" the user wants something else, let us use it.
2017-02-11 08:01:38 -05:00
let cmd += split(g:go_metalinter_command, " ")
2016-12-27 09:46:49 -05:00
endif
2018-02-04 06:35:08 -05:00
if a:autosave
" redraw so that any messages that were displayed while writing the file
" will be cleared
redraw
" Include only messages for the active buffer for autosave.
let cmd += [printf('--include=^%s:.*$', fnamemodify(expand('%:p'), ":."))]
endif
2017-02-11 08:01:38 -05:00
" gometalinter has a default deadline of 5 seconds.
"
" For async mode (s:lint_job), we want to override the default deadline only
" if we have a deadline configured.
"
" For sync mode (go#util#System), always explicitly pass the 5 seconds
2017-02-11 08:01:38 -05:00
" deadline if there is no other deadline configured. If a deadline is
" configured, then use it.
" Call gometalinter asynchronously.
2016-12-27 09:46:49 -05:00
if go#util#has_job() && has('lambda')
2017-02-11 08:01:38 -05:00
let deadline = get(g:, 'go_metalinter_deadline', 0)
if deadline != 0
let cmd += ["--deadline=" . deadline]
endif
2018-02-04 06:35:08 -05:00
let cmd += goargs
call s:lint_job({'cmd': cmd}, a:autosave)
2016-12-27 09:46:49 -05:00
return
endif
2017-02-11 08:01:38 -05:00
" We're calling gometalinter synchronously.
let cmd += ["--deadline=" . get(g:, 'go_metalinter_deadline', "5s")]
2018-02-04 06:35:08 -05:00
let cmd += goargs
2016-06-26 07:12:36 -04:00
2018-02-04 06:35:08 -05:00
let [l:out, l:err] = go#util#Exec(cmd)
2016-12-27 09:46:49 -05:00
2018-02-04 06:35:08 -05:00
if a:autosave
let l:listtype = go#list#Type("GoMetaLinterAutoSave")
else
let l:listtype = go#list#Type("GoMetaLinter")
endif
2016-06-26 07:12:36 -04:00
2018-02-04 06:35:08 -05:00
if l:err == 0
2016-06-26 07:12:36 -04:00
call go#list#Clean(l:listtype)
echon "vim-go: " | echohl Function | echon "[metalinter] PASS" | echohl None
else
" GoMetaLinter can output one of the two, so we look for both:
" <file>:<line>:[<column>]: <message> (<linter>)
" <file>:<line>:: <message> (<linter>)
" This can be defined by the following errorformat:
let errformat = "%f:%l:%c:%t%*[^:]:\ %m,%f:%l::%t%*[^:]:\ %m"
" Parse and populate our location list
2016-12-27 09:46:49 -05:00
call go#list#ParseFormat(l:listtype, errformat, split(out, "\n"), 'GoMetaLinter')
2016-06-26 07:12:36 -04:00
let errors = go#list#Get(l:listtype)
call go#list#Window(l:listtype, len(errors))
if !a:autosave
call go#list#JumpToFirst(l:listtype)
2015-12-08 08:20:04 -05:00
endif
2016-06-26 07:12:36 -04:00
endif
2015-12-08 08:20:04 -05:00
endfunction
" Golint calls 'golint' on the current directory. Any warnings are populated in
" the location list
function! go#lint#Golint(...) abort
2017-03-07 12:04:28 -05:00
let bin_path = go#path#CheckBinPath(g:go_golint_bin)
if empty(bin_path)
return
2016-06-26 07:12:36 -04:00
endif
2017-09-02 06:43:18 -04:00
let bin_path = go#util#Shellescape(bin_path)
2016-06-26 07:12:36 -04:00
if a:0 == 0
2017-09-02 06:43:18 -04:00
let out = go#util#System(bin_path . " " . go#util#Shellescape(go#package#ImportPath()))
2016-06-26 07:12:36 -04:00
else
2017-05-26 05:30:32 -04:00
let out = go#util#System(bin_path . " " . go#util#Shelljoin(a:000))
2016-06-26 07:12:36 -04:00
endif
if empty(out)
echon "vim-go: " | echohl Function | echon "[lint] PASS" | echohl None
return
endif
let l:listtype = go#list#Type("GoLint")
2018-03-31 10:56:26 -04:00
call go#list#Parse(l:listtype, out, "GoLint")
2016-06-26 07:12:36 -04:00
let errors = go#list#Get(l:listtype)
call go#list#Window(l:listtype, len(errors))
call go#list#JumpToFirst(l:listtype)
2015-12-08 08:20:04 -05:00
endfunction
" Vet calls 'go vet' on the current directory. Any warnings are populated in
" the location list
2016-12-27 09:46:49 -05:00
function! go#lint#Vet(bang, ...) abort
2016-06-26 07:12:36 -04:00
call go#cmd#autowrite()
echon "vim-go: " | echohl Identifier | echon "calling vet..." | echohl None
if a:0 == 0
2017-09-02 06:43:18 -04:00
let out = go#util#System('go vet ' . go#util#Shellescape(go#package#ImportPath()))
2016-06-26 07:12:36 -04:00
else
2017-09-02 06:43:18 -04:00
let out = go#util#System('go tool vet ' . go#util#Shelljoin(a:000))
2016-06-26 07:12:36 -04:00
endif
let l:listtype = go#list#Type("GoVet")
2016-06-26 07:12:36 -04:00
if go#util#ShellError() != 0
2018-03-31 10:56:26 -04:00
let errorformat="%-Gexit status %\\d%\\+," . &errorformat
call go#list#ParseFormat(l:listtype, l:errorformat, out, "GoVet")
let errors = go#list#Get(l:listtype)
2016-06-26 07:12:36 -04:00
call go#list#Window(l:listtype, len(errors))
if !empty(errors) && !a:bang
call go#list#JumpToFirst(l:listtype)
2015-12-08 08:20:04 -05:00
endif
2016-06-26 07:12:36 -04:00
echon "vim-go: " | echohl ErrorMsg | echon "[vet] FAIL" | echohl None
else
call go#list#Clean(l:listtype)
redraw | echon "vim-go: " | echohl Function | echon "[vet] PASS" | echohl None
endif
2014-10-31 17:30:24 -04:00
endfunction
2015-12-08 08:20:04 -05:00
" ErrCheck calls 'errcheck' for the given packages. Any warnings are populated in
" the location list
function! go#lint#Errcheck(...) abort
2016-06-26 07:12:36 -04:00
if a:0 == 0
2017-07-06 08:57:35 -04:00
let import_path = go#package#ImportPath()
if import_path == -1
2016-06-26 07:12:36 -04:00
echohl Error | echomsg "vim-go: package is not inside GOPATH src" | echohl None
return
2015-12-08 08:20:04 -05:00
endif
2016-06-26 07:12:36 -04:00
else
2017-07-06 08:57:35 -04:00
let import_path = go#util#Shelljoin(a:000)
2016-06-26 07:12:36 -04:00
endif
2015-12-08 08:20:04 -05:00
2016-06-26 07:12:36 -04:00
let bin_path = go#path#CheckBinPath(g:go_errcheck_bin)
if empty(bin_path)
return
endif
echon "vim-go: " | echohl Identifier | echon "errcheck analysing ..." | echohl None
redraw
2017-09-02 06:43:18 -04:00
let command = go#util#Shellescape(bin_path) . ' -abspath ' . import_path
2016-06-26 07:12:36 -04:00
let out = go#tool#ExecuteInDir(command)
let l:listtype = go#list#Type("GoErrCheck")
2016-06-26 07:12:36 -04:00
if go#util#ShellError() != 0
let errformat = "%f:%l:%c:\ %m, %f:%l:%c\ %#%m"
" Parse and populate our location list
2016-12-27 09:46:49 -05:00
call go#list#ParseFormat(l:listtype, errformat, split(out, "\n"), 'Errcheck')
2016-06-26 07:12:36 -04:00
let errors = go#list#Get(l:listtype)
if empty(errors)
echohl Error | echomsg "GoErrCheck returned error" | echohl None
echo out
return
2015-12-08 08:20:04 -05:00
endif
2016-06-26 07:12:36 -04:00
if !empty(errors)
2017-07-06 08:57:35 -04:00
echohl Error | echomsg "GoErrCheck found errors" | echohl None
2016-12-27 09:46:49 -05:00
call go#list#Populate(l:listtype, errors, 'Errcheck')
2016-06-26 07:12:36 -04:00
call go#list#Window(l:listtype, len(errors))
if !empty(errors)
call go#list#JumpToFirst(l:listtype)
endif
2015-12-08 08:20:04 -05:00
endif
2016-06-26 07:12:36 -04:00
else
call go#list#Clean(l:listtype)
echon "vim-go: " | echohl Function | echon "[errcheck] PASS" | echohl None
endif
2015-12-08 08:20:04 -05:00
endfunction
2014-10-31 17:30:24 -04:00
2016-12-27 09:46:49 -05:00
function! go#lint#ToggleMetaLinterAutoSave() abort
2016-08-02 08:48:32 -04:00
if get(g:, "go_metalinter_autosave", 0)
let g:go_metalinter_autosave = 0
call go#util#EchoProgress("auto metalinter disabled")
return
end
let g:go_metalinter_autosave = 1
call go#util#EchoProgress("auto metalinter enabled")
endfunction
2018-02-04 06:35:08 -05:00
function! s:lint_job(args, autosave)
2018-03-31 10:56:26 -04:00
let state = {
\ 'status_dir': expand('%:p:h'),
\ 'started_at': reltime(),
\ 'messages': [],
\ 'exited': 0,
\ 'closed': 0,
\ 'exit_status': 0,
\ 'winnr': winnr(),
\ 'autosave': a:autosave
\ }
call go#statusline#Update(state.status_dir, {
2016-12-27 09:46:49 -05:00
\ 'desc': "current status",
\ 'type': "gometalinter",
\ 'state': "analysing",
\})
" autowrite is not enabled for jobs
call go#cmd#autowrite()
2018-02-04 06:35:08 -05:00
if a:autosave
2018-03-31 10:56:26 -04:00
let state.listtype = go#list#Type("GoMetaLinterAutoSave")
2018-02-04 06:35:08 -05:00
else
2018-03-31 10:56:26 -04:00
let state.listtype = go#list#Type("GoMetaLinter")
2018-02-04 06:35:08 -05:00
endif
2016-12-27 09:46:49 -05:00
2018-03-31 10:56:26 -04:00
function! s:callback(chan, msg) dict closure
call add(self.messages, a:msg)
2016-12-27 09:46:49 -05:00
endfunction
2018-03-31 10:56:26 -04:00
function! s:exit_cb(job, exitval) dict
let self.exited = 1
let self.exit_status = a:exitval
2018-02-04 06:35:08 -05:00
2016-12-27 09:46:49 -05:00
let status = {
\ 'desc': 'last status',
\ 'type': "gometaliner",
\ 'state': "finished",
\ }
2017-05-02 08:42:08 -04:00
if a:exitval
2016-12-27 09:46:49 -05:00
let status.state = "failed"
endif
2018-03-31 10:56:26 -04:00
let elapsed_time = reltimestr(reltime(self.started_at))
2016-12-27 09:46:49 -05:00
" strip whitespace
let elapsed_time = substitute(elapsed_time, '^\s*\(.\{-}\)\s*$', '\1', '')
let status.state .= printf(" (%ss)", elapsed_time)
2018-03-31 10:56:26 -04:00
call go#statusline#Update(self.status_dir, status)
2016-12-27 09:46:49 -05:00
2018-03-31 10:56:26 -04:00
if self.closed
call self.show_errors()
2016-12-27 09:46:49 -05:00
endif
2018-02-04 06:35:08 -05:00
endfunction
2018-03-31 10:56:26 -04:00
function! s:close_cb(ch) dict
let self.closed = 1
2018-02-04 06:35:08 -05:00
2018-03-31 10:56:26 -04:00
if self.exited
call self.show_errors()
2018-02-04 06:35:08 -05:00
endif
endfunction
2018-03-31 10:56:26 -04:00
function state.show_errors()
let l:winnr = winnr()
2018-02-04 06:35:08 -05:00
" make sure the current window is the window from which gometalinter was
" run when the listtype is locationlist so that the location list for the
" correct window will be populated.
2018-03-31 10:56:26 -04:00
if self.listtype == 'locationlist'
exe self.winnr . "wincmd w"
2018-02-04 06:35:08 -05:00
endif
let l:errorformat = '%f:%l:%c:%t%*[^:]:\ %m,%f:%l::%t%*[^:]:\ %m'
2018-03-31 10:56:26 -04:00
call go#list#ParseFormat(self.listtype, l:errorformat, self.messages, 'GoMetaLinter')
2018-02-04 06:35:08 -05:00
2018-03-31 10:56:26 -04:00
let errors = go#list#Get(self.listtype)
call go#list#Window(self.listtype, len(errors))
" move to the window that was active before processing the errors, because
" the user may have moved around within the window or even moved to a
" different window since saving. Moving back to current window as of the
" start of this function avoids the perception that the quickfix window
" steals focus when linting takes a while.
if self.autosave
exe l:winnr . "wincmd w"
endif
2016-12-27 09:46:49 -05:00
if get(g:, 'go_echo_command_info', 1)
call go#util#EchoSuccess("linting finished")
endif
endfunction
2018-03-31 10:56:26 -04:00
" explicitly bind the callbacks to state so that self within them always
" refers to state. See :help Partial for more information.
2016-12-27 09:46:49 -05:00
let start_options = {
2018-03-31 10:56:26 -04:00
\ 'callback': funcref("s:callback", [], state),
\ 'exit_cb': funcref("s:exit_cb", [], state),
\ 'close_cb': funcref("s:close_cb", [], state),
2016-12-27 09:46:49 -05:00
\ }
call job_start(a:args.cmd, start_options)
if get(g:, 'go_echo_command_info', 1)
call go#util#EchoProgress("linting started ...")
endif
endfunction
2016-06-26 07:12:36 -04:00
" vim: sw=2 ts=2 et