376 lines
10 KiB
VimL
376 lines
10 KiB
VimL
|
" don't spam the user when Vim is started in Vi compatibility mode
|
||
|
let s:cpo_save = &cpo
|
||
|
set cpo&vim
|
||
|
|
||
|
function! go#cmd#autowrite() abort
|
||
|
if &autowrite == 1 || &autowriteall == 1
|
||
|
silent! wall
|
||
|
else
|
||
|
for l:nr in range(0, bufnr('$'))
|
||
|
if buflisted(l:nr) && getbufvar(l:nr, '&modified')
|
||
|
" Sleep one second to make sure people see the message. Otherwise it is
|
||
|
" often immediately overwritten by the async messages (which also
|
||
|
" doesn't invoke the "hit ENTER" prompt).
|
||
|
call go#util#EchoWarning('[No write since last change]')
|
||
|
sleep 1
|
||
|
return
|
||
|
endif
|
||
|
endfor
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
" Build builds the source code without producing any output binary. We live in
|
||
|
" an editor so the best is to build it to catch errors and fix them. By
|
||
|
" default it tries to call simply 'go build', but it first tries to get all
|
||
|
" dependent files for the current folder and passes it to go build.
|
||
|
function! go#cmd#Build(bang, ...) abort
|
||
|
" Create our command arguments. go build discards any results when it
|
||
|
" compiles multiple packages. So we pass the `errors` package just as an
|
||
|
" placeholder with the current folder (indicated with '.').
|
||
|
let l:args =
|
||
|
\ ['build', '-tags', go#config#BuildTags()] +
|
||
|
\ map(copy(a:000), "expand(v:val)") +
|
||
|
\ [".", "errors"]
|
||
|
|
||
|
" Vim and Neovim async
|
||
|
if go#util#has_job()
|
||
|
call s:cmd_job({
|
||
|
\ 'cmd': ['go'] + args,
|
||
|
\ 'bang': a:bang,
|
||
|
\ 'for': 'GoBuild',
|
||
|
\ 'statustype': 'build'
|
||
|
\})
|
||
|
|
||
|
" Vim without async
|
||
|
else
|
||
|
let l:status = {
|
||
|
\ 'desc': 'current status',
|
||
|
\ 'type': 'build',
|
||
|
\ 'state': "started",
|
||
|
\ }
|
||
|
call go#statusline#Update(expand('%:p:h'), l:status)
|
||
|
|
||
|
let default_makeprg = &makeprg
|
||
|
let &makeprg = "go " . join(go#util#Shelllist(args), ' ')
|
||
|
|
||
|
let l:listtype = go#list#Type("GoBuild")
|
||
|
" execute make inside the source folder so we can parse the errors
|
||
|
" correctly
|
||
|
try
|
||
|
let l:dir = go#util#Chdir(expand("%:p:h"))
|
||
|
if l:listtype == "locationlist"
|
||
|
silent! exe 'lmake!'
|
||
|
else
|
||
|
silent! exe 'make!'
|
||
|
endif
|
||
|
redraw!
|
||
|
finally
|
||
|
call go#util#Chdir(l:dir)
|
||
|
let &makeprg = default_makeprg
|
||
|
endtry
|
||
|
|
||
|
let errors = go#list#Get(l:listtype)
|
||
|
call go#list#Window(l:listtype, len(errors))
|
||
|
if !empty(errors) && !a:bang
|
||
|
call go#list#JumpToFirst(l:listtype)
|
||
|
let l:status.state = 'failed'
|
||
|
else
|
||
|
let l:status.state = 'success'
|
||
|
if go#config#EchoCommandInfo()
|
||
|
call go#util#EchoSuccess("[build] SUCCESS")
|
||
|
endif
|
||
|
endif
|
||
|
call go#statusline#Update(expand('%:p:h'), l:status)
|
||
|
endif
|
||
|
|
||
|
endfunction
|
||
|
|
||
|
|
||
|
" BuildTags sets or shows the current build tags used for tools
|
||
|
function! go#cmd#BuildTags(bang, ...) abort
|
||
|
if a:0
|
||
|
let v = a:1
|
||
|
if v == '""' || v == "''"
|
||
|
let v = ""
|
||
|
endif
|
||
|
call go#config#SetBuildTags(v)
|
||
|
let tags = go#config#BuildTags()
|
||
|
if empty(tags)
|
||
|
call go#util#EchoSuccess("build tags are cleared")
|
||
|
else
|
||
|
call go#util#EchoSuccess("build tags are changed to: " . tags)
|
||
|
endif
|
||
|
|
||
|
return
|
||
|
endif
|
||
|
|
||
|
let tags = go#config#BuildTags()
|
||
|
if empty(tags)
|
||
|
call go#util#EchoSuccess("build tags are not set")
|
||
|
else
|
||
|
call go#util#EchoSuccess("current build tags: " . tags)
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
|
||
|
" Run runs the current file (and their dependencies if any) in a new terminal.
|
||
|
function! go#cmd#RunTerm(bang, mode, files) abort
|
||
|
let cmd = ["go", "run"]
|
||
|
if len(go#config#BuildTags()) > 0
|
||
|
call extend(cmd, ["-tags", go#config#BuildTags()])
|
||
|
endif
|
||
|
|
||
|
if empty(a:files)
|
||
|
call extend(cmd, go#tool#Files())
|
||
|
else
|
||
|
call extend(cmd, map(copy(a:files), funcref('s:expandRunArgs')))
|
||
|
endif
|
||
|
call go#term#newmode(a:bang, cmd, s:runerrorformat(), a:mode)
|
||
|
endfunction
|
||
|
|
||
|
" Run runs the current file (and their dependencies if any) and outputs it.
|
||
|
" This is intended to test small programs and play with them. It's not
|
||
|
" suitable for long running apps, because vim is blocking by default and
|
||
|
" calling long running apps will block the whole UI.
|
||
|
function! go#cmd#Run(bang, ...) abort
|
||
|
if go#config#TermEnabled()
|
||
|
call go#cmd#RunTerm(a:bang, '', a:000)
|
||
|
return
|
||
|
endif
|
||
|
|
||
|
if go#util#has_job()
|
||
|
" NOTE(arslan): 'term': 'open' case is not implement for +jobs. This means
|
||
|
" executions waiting for stdin will not work. That's why we don't do
|
||
|
" anything. Once this is implemented we're going to make :GoRun async
|
||
|
endif
|
||
|
|
||
|
let l:status = {
|
||
|
\ 'desc': 'current status',
|
||
|
\ 'type': 'run',
|
||
|
\ 'state': "started",
|
||
|
\ }
|
||
|
|
||
|
call go#statusline#Update(expand('%:p:h'), l:status)
|
||
|
|
||
|
let l:cmd = ['go', 'run']
|
||
|
let l:tags = go#config#BuildTags()
|
||
|
if len(l:tags) > 0
|
||
|
let l:cmd = l:cmd + ['-tags', l:tags]
|
||
|
endif
|
||
|
|
||
|
if a:0 == 0
|
||
|
let l:files = go#tool#Files()
|
||
|
else
|
||
|
let l:files = map(copy(a:000), funcref('s:expandRunArgs'))
|
||
|
endif
|
||
|
|
||
|
let l:cmd = l:cmd + l:files
|
||
|
|
||
|
if go#util#IsWin()
|
||
|
if go#util#HasDebug('shell-commands')
|
||
|
call go#util#EchoInfo(printf('shell command: %s', string(l:cmd)))
|
||
|
endif
|
||
|
try
|
||
|
let l:dir = go#util#Chdir(expand("%:p:h"))
|
||
|
exec printf('!%s', go#util#Shelljoin(l:cmd, 1))
|
||
|
finally
|
||
|
call go#util#Chdir(l:dir)
|
||
|
endtry
|
||
|
|
||
|
let l:status.state = 'success'
|
||
|
if v:shell_error
|
||
|
let l:status.state = 'failed'
|
||
|
if go#config#EchoCommandInfo()
|
||
|
redraws!
|
||
|
call go#util#EchoError('[run] FAILED')
|
||
|
endif
|
||
|
else
|
||
|
if go#config#EchoCommandInfo()
|
||
|
redraws!
|
||
|
call go#util#EchoSuccess('[run] SUCCESS')
|
||
|
endif
|
||
|
endif
|
||
|
|
||
|
call go#statusline#Update(expand('%:p:h'), l:status)
|
||
|
return
|
||
|
endif
|
||
|
|
||
|
" :make expands '%' and '#' wildcards, so they must also be escaped
|
||
|
let l:default_makeprg = &makeprg
|
||
|
let &makeprg = go#util#Shelljoin(l:cmd, 1)
|
||
|
|
||
|
let l:listtype = go#list#Type("GoRun")
|
||
|
|
||
|
let l:status.state = 'success'
|
||
|
|
||
|
let l:dir = go#util#Chdir(expand("%:p:h"))
|
||
|
try
|
||
|
" backup user's errorformat, will be restored once we are finished
|
||
|
let l:old_errorformat = &errorformat
|
||
|
let &errorformat = s:runerrorformat()
|
||
|
|
||
|
if go#util#HasDebug('shell-commands')
|
||
|
call go#util#EchoInfo(printf('shell command: %s', string(l:cmd)))
|
||
|
endif
|
||
|
|
||
|
if l:listtype == "locationlist"
|
||
|
exe 'lmake!'
|
||
|
else
|
||
|
exe 'make!'
|
||
|
endif
|
||
|
finally
|
||
|
call go#util#Chdir(l:dir)
|
||
|
let &errorformat = l:old_errorformat
|
||
|
let &makeprg = l:default_makeprg
|
||
|
endtry
|
||
|
|
||
|
let l:errors = go#list#Get(l:listtype)
|
||
|
|
||
|
call go#list#Window(l:listtype, len(l:errors))
|
||
|
if !empty(l:errors)
|
||
|
let l:status.state = 'failed'
|
||
|
if !a:bang
|
||
|
call go#list#JumpToFirst(l:listtype)
|
||
|
endif
|
||
|
endif
|
||
|
call go#statusline#Update(expand('%:p:h'), l:status)
|
||
|
endfunction
|
||
|
|
||
|
" Install installs the package by simple calling 'go install'. If any argument
|
||
|
" is given(which are passed directly to 'go install') it tries to install
|
||
|
" those packages. Errors are populated in the location window.
|
||
|
function! go#cmd#Install(bang, ...) abort
|
||
|
" use vim's job functionality to call it asynchronously
|
||
|
if go#util#has_job()
|
||
|
" expand all wildcards(i.e: '%' to the current file name)
|
||
|
let goargs = map(copy(a:000), "expand(v:val)")
|
||
|
|
||
|
call s:cmd_job({
|
||
|
\ 'cmd': ['go', 'install', '-tags', go#config#BuildTags()] + goargs,
|
||
|
\ 'bang': a:bang,
|
||
|
\ 'for': 'GoInstall',
|
||
|
\ 'statustype': 'install'
|
||
|
\})
|
||
|
return
|
||
|
endif
|
||
|
|
||
|
let default_makeprg = &makeprg
|
||
|
|
||
|
" :make expands '%' and '#' wildcards, so they must also be escaped
|
||
|
let goargs = go#util#Shelljoin(map(copy(a:000), "expand(v:val)"), 1)
|
||
|
let &makeprg = "go install " . goargs
|
||
|
|
||
|
let l:listtype = go#list#Type("GoInstall")
|
||
|
" execute make inside the source folder so we can parse the errors
|
||
|
" correctly
|
||
|
try
|
||
|
let l:dir = go#util#Chdir(expand("%:p:h"))
|
||
|
if l:listtype == "locationlist"
|
||
|
silent! exe 'lmake!'
|
||
|
else
|
||
|
silent! exe 'make!'
|
||
|
endif
|
||
|
redraw!
|
||
|
finally
|
||
|
call go#util#Chdir(l:dir)
|
||
|
let &makeprg = default_makeprg
|
||
|
endtry
|
||
|
|
||
|
let errors = go#list#Get(l:listtype)
|
||
|
call go#list#Window(l:listtype, len(errors))
|
||
|
if !empty(errors) && !a:bang
|
||
|
call go#list#JumpToFirst(l:listtype)
|
||
|
else
|
||
|
call go#util#EchoSuccess("installed to ". go#path#Default())
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
" Generate runs 'go generate' in similar fashion to go#cmd#Build()
|
||
|
function! go#cmd#Generate(bang, ...) abort
|
||
|
let default_makeprg = &makeprg
|
||
|
|
||
|
" :make expands '%' and '#' wildcards, so they must also be escaped
|
||
|
let goargs = go#util#Shelljoin(map(copy(a:000), "expand(v:val)"), 1)
|
||
|
if go#util#ShellError() != 0
|
||
|
let &makeprg = "go generate " . goargs
|
||
|
else
|
||
|
let gofiles = go#util#Shelljoin(go#tool#Files(), 1)
|
||
|
let &makeprg = "go generate " . goargs . ' ' . gofiles
|
||
|
endif
|
||
|
|
||
|
let l:status = {
|
||
|
\ 'desc': 'current status',
|
||
|
\ 'type': 'generate',
|
||
|
\ 'state': "started",
|
||
|
\ }
|
||
|
call go#statusline#Update(expand('%:p:h'), l:status)
|
||
|
|
||
|
if go#config#EchoCommandInfo()
|
||
|
call go#util#EchoProgress('generating ...')
|
||
|
endif
|
||
|
|
||
|
let l:listtype = go#list#Type("GoGenerate")
|
||
|
|
||
|
try
|
||
|
if l:listtype == "locationlist"
|
||
|
silent! exe 'lmake!'
|
||
|
else
|
||
|
silent! exe 'make!'
|
||
|
endif
|
||
|
finally
|
||
|
redraw!
|
||
|
let &makeprg = default_makeprg
|
||
|
endtry
|
||
|
|
||
|
let errors = go#list#Get(l:listtype)
|
||
|
call go#list#Window(l:listtype, len(errors))
|
||
|
if !empty(errors)
|
||
|
let l:status.status = 'failed'
|
||
|
if !a:bang
|
||
|
call go#list#JumpToFirst(l:listtype)
|
||
|
endif
|
||
|
else
|
||
|
let l:status.status = 'success'
|
||
|
if go#config#EchoCommandInfo()
|
||
|
redraws!
|
||
|
call go#util#EchoSuccess('[generate] SUCCESS')
|
||
|
endif
|
||
|
endif
|
||
|
call go#statusline#Update(expand(':%:p:h'), l:status)
|
||
|
endfunction
|
||
|
|
||
|
function! s:runerrorformat()
|
||
|
let l:panicaddress = "%\\t%#%f:%l +0x%[0-9A-Fa-f]%\\+"
|
||
|
let l:errorformat = '%A' . l:panicaddress . "," . &errorformat
|
||
|
return l:errorformat
|
||
|
endfunction
|
||
|
|
||
|
" s:expandRunArgs expands arguments for go#cmd#Run according to the
|
||
|
" documentation of :GoRun. When val is a readable file, it is expanded to the
|
||
|
" full path so that go run can be executed in the current buffer's directory.
|
||
|
" val is return unaltered otherwise to support non-file arguments to go run.
|
||
|
function! s:expandRunArgs(idx, val) abort
|
||
|
let l:val = expand(a:val)
|
||
|
if !filereadable(l:val)
|
||
|
return l:val
|
||
|
endif
|
||
|
|
||
|
return fnamemodify(l:val, ':p')")
|
||
|
endfunction
|
||
|
" ---------------------
|
||
|
" | Vim job callbacks |
|
||
|
" ---------------------
|
||
|
|
||
|
function! s:cmd_job(args) abort
|
||
|
" autowrite is not enabled for jobs
|
||
|
call go#cmd#autowrite()
|
||
|
|
||
|
call go#job#Spawn(a:args.cmd, a:args)
|
||
|
endfunction
|
||
|
|
||
|
" restore Vi compatibility settings
|
||
|
let &cpo = s:cpo_save
|
||
|
unlet s:cpo_save
|
||
|
|
||
|
" vim: sw=2 ts=2 et
|