196 lines
6.2 KiB
VimL
196 lines
6.2 KiB
VimL
|
" don't spam the user when Vim is started in Vi compatibility mode
|
||
|
let s:cpo_save = &cpo
|
||
|
set cpo&vim
|
||
|
|
||
|
" Write a Go file to a temporary directory and append this directory to
|
||
|
" $GOPATH.
|
||
|
"
|
||
|
" The file will written to a:path, which is relative to the temporary
|
||
|
" directory, and this file will be loaded as the current buffer.
|
||
|
"
|
||
|
" A Go module will be configured in the first segment of a:path within the
|
||
|
" temporary directory. The module's name will be prefixed with vim-go.test/
|
||
|
" followed by the first segment in a:path.
|
||
|
"
|
||
|
" The current directory will be changed to the parent directory of module
|
||
|
" root.
|
||
|
"
|
||
|
" The cursor will be placed on the character before any 0x1f byte.
|
||
|
"
|
||
|
" The full path to the created directory is returned, it is the caller's
|
||
|
" responsibility to clean that up!
|
||
|
fun! gotest#write_file(path, contents) abort
|
||
|
let l:dir = go#util#tempdir("vim-go-test/testrun/")
|
||
|
let $GOPATH .= ':' . l:dir
|
||
|
let l:full_path = l:dir . '/src/' . a:path
|
||
|
|
||
|
call mkdir(fnamemodify(l:full_path, ':h'), 'p')
|
||
|
call writefile(a:contents, l:full_path)
|
||
|
call s:setupproject(printf('%s/src', l:dir), a:path)
|
||
|
|
||
|
silent exe 'e! ' . a:path
|
||
|
|
||
|
" Set cursor.
|
||
|
let l:lnum = 1
|
||
|
for l:line in a:contents
|
||
|
let l:m = stridx(l:line, "\x1f")
|
||
|
if l:m > -1
|
||
|
call cursor(l:lnum, l:m)
|
||
|
call setline('.', substitute(getline('.'), "\x1f", '', ''))
|
||
|
silent noautocmd w!
|
||
|
|
||
|
call go#lsp#DidClose(expand('%:p'))
|
||
|
call go#lsp#DidOpen(expand('%:p'))
|
||
|
|
||
|
break
|
||
|
endif
|
||
|
|
||
|
let l:lnum += 1
|
||
|
endfor
|
||
|
|
||
|
return l:dir
|
||
|
endfun
|
||
|
|
||
|
" Load a fixture file from test-fixtures.
|
||
|
"
|
||
|
" The file will be copied to a new GOPATH-compliant temporary directory and
|
||
|
" loaded as the current buffer.
|
||
|
"
|
||
|
" A Go module will be configured in the first segment of a:path within the
|
||
|
" temporary directory. The module's name will be prefixed with vim-go.test/
|
||
|
" followed by the first segment in a:path.
|
||
|
"
|
||
|
" The current directory will be changed to the parent directory of module
|
||
|
" root.
|
||
|
fun! gotest#load_fixture(path) abort
|
||
|
if go#util#has_job()
|
||
|
call go#lsp#CleanWorkspaces()
|
||
|
endif
|
||
|
let l:dir = go#util#tempdir("vim-go-test/testrun/")
|
||
|
let $GOPATH .= ':' . l:dir
|
||
|
let l:full_path = l:dir . '/src/' . a:path
|
||
|
|
||
|
call mkdir(fnamemodify(l:full_path, ':h'), 'p')
|
||
|
call s:setupproject(printf('%s/src', l:dir), a:path)
|
||
|
silent exe 'noautocmd e! ' . a:path
|
||
|
silent exe printf('read %s/test-fixtures/%s', g:vim_go_root, a:path)
|
||
|
silent noautocmd w!
|
||
|
if go#util#has_job()
|
||
|
call go#lsp#AddWorkspaceDirectory(fnamemodify(l:full_path, ':p:h'))
|
||
|
endif
|
||
|
|
||
|
return l:dir
|
||
|
endfun
|
||
|
|
||
|
" Diff the contents of the current buffer to a:want, which should be a list.
|
||
|
" If a:skipHeader is true we won't bother with the package and import
|
||
|
" declarations; so e.g.:
|
||
|
"
|
||
|
" let l:diff = s:diff_buffer(1, ['_ = mail.Address{}'])
|
||
|
"
|
||
|
" will pass, whereas otherwise you'd have to:
|
||
|
"
|
||
|
" let l:diff = s:diff_buffer(0, ['package main', 'import "net/mail", '_ = mail.Address{}'])
|
||
|
fun! gotest#assert_buffer(skipHeader, want) abort
|
||
|
let l:buffer = go#util#GetLines()
|
||
|
|
||
|
if a:skipHeader
|
||
|
for l:lnum in range(0, len(l:buffer) - 1)
|
||
|
" Bit rudimentary, but works reasonably well.
|
||
|
if match(l:buffer[l:lnum], '^\v(func|var|const|import \(|\))') > -1
|
||
|
" vint bug: https://github.com/Kuniwak/vint/issues/179
|
||
|
" vint: -ProhibitUsingUndeclaredVariable
|
||
|
let l:buffer = l:buffer[l:lnum:len(l:buffer)]
|
||
|
break
|
||
|
endif
|
||
|
endfor
|
||
|
endif
|
||
|
|
||
|
" Using ' is often easier so we don't have to escape ".
|
||
|
let l:want = map(a:want, 'substitute(v:val, "\\\\t", "\t", "")')
|
||
|
|
||
|
let l:tmp = go#util#tempdir('assert_buffer')
|
||
|
try
|
||
|
call writefile(l:buffer, l:tmp . '/have')
|
||
|
call writefile(l:want, l:tmp . '/want')
|
||
|
call go#fmt#run('gofmt', l:tmp . '/have', l:tmp . '/have')
|
||
|
call go#fmt#run('gofmt', l:tmp . '/want', l:tmp . '/want')
|
||
|
let [l:out, l:err] = go#util#Exec(["diff", "-u", l:tmp . '/have', l:tmp . '/want'])
|
||
|
finally
|
||
|
call delete(l:tmp . '/have')
|
||
|
call delete(l:tmp . '/want')
|
||
|
call delete(l:tmp, 'd')
|
||
|
endtry
|
||
|
|
||
|
if l:err || l:out != ''
|
||
|
let v:errors = extend(v:errors, split(l:out, "\n"))
|
||
|
endif
|
||
|
endfun
|
||
|
|
||
|
" Diff the contents of the current buffer to the fixture file in a:path.
|
||
|
fun! gotest#assert_fixture(path) abort
|
||
|
let l:want = readfile(printf('%s/test-fixtures/%s', g:vim_go_root, a:path))
|
||
|
call gotest#assert_buffer(0, l:want)
|
||
|
endfun
|
||
|
|
||
|
func! gotest#assert_quickfix(got, want) abort
|
||
|
call assert_equal(len(a:want), len(a:got), "number of errors")
|
||
|
if len(a:want) != len(a:got)
|
||
|
return assert_equal(a:want, a:got)
|
||
|
endif
|
||
|
|
||
|
let l:retval = 0
|
||
|
let i = 0
|
||
|
|
||
|
while i < len(a:want)
|
||
|
let want_item = a:want[i]
|
||
|
let got_item = a:got[i]
|
||
|
let i += 1
|
||
|
|
||
|
let l:retval = assert_equal(want_item.bufnr, got_item.bufnr, "bufnr") || l:retval
|
||
|
let l:retval = assert_equal(want_item.lnum, got_item.lnum, "lnum") || l:retval
|
||
|
let l:retval = assert_equal(want_item.col, got_item.col, "col") || l:retval
|
||
|
let l:retval = assert_equal(want_item.vcol, got_item.vcol, "vcol") || l:retval
|
||
|
let l:retval = assert_equal(want_item.nr, got_item.nr, "nr") || l:retval
|
||
|
let l:retval = assert_equal(want_item.pattern, got_item.pattern, "pattern") || l:retval
|
||
|
let l:retval = assert_equal(want_item.text, got_item.text, "text") || l:retval
|
||
|
let l:retval = assert_equal(want_item.type, got_item.type, "type") || l:retval
|
||
|
let l:retval = assert_equal(want_item.valid, got_item.valid, "valid") || l:retval
|
||
|
endwhile
|
||
|
|
||
|
return l:retval
|
||
|
endfunc
|
||
|
|
||
|
" s:setupproject sets up a Go module in dir rooted at the first segment of
|
||
|
" path and changes the current directory to the parent directory of the
|
||
|
" project root.
|
||
|
func! s:setupproject(dir, path) abort
|
||
|
let l:projectdir = s:projectdir(a:path)
|
||
|
let l:mod = printf('vim-go.test/%s', l:projectdir)
|
||
|
let l:modroot = printf('%s/%s', a:dir, l:projectdir)
|
||
|
call s:creategomod(l:mod, l:modroot)
|
||
|
call go#util#Chdir(a:dir)
|
||
|
endfunc
|
||
|
|
||
|
func! s:creategomod(modname, dir) abort
|
||
|
call go#util#ExecInWorkDir(['go', 'mod', 'init', a:modname], a:dir)
|
||
|
endfunc
|
||
|
|
||
|
" s:project dir returns the first element of path.
|
||
|
func! s:projectdir(path) abort
|
||
|
let l:path = a:path
|
||
|
let l:next = fnamemodify(l:path, ':h')
|
||
|
while l:next isnot '.'
|
||
|
let l:path = l:next
|
||
|
let l:next = fnamemodify(l:path, ':h')
|
||
|
endwhile
|
||
|
|
||
|
return l:path
|
||
|
endfunc
|
||
|
|
||
|
" restore Vi compatibility settings
|
||
|
let &cpo = s:cpo_save
|
||
|
unlet s:cpo_save
|
||
|
|
||
|
" vim: sw=2 ts=2 et
|