551 lines
16 KiB
VimL
Raw Permalink Normal View History

if exists('g:polyglot_disabled') && index(g:polyglot_disabled, 'rust') != -1
finish
endif
2014-07-29 13:03:49 +02:00
" Author: Kevin Ballard
" Description: Helper functions for Rust commands/mappings
" Last Modified: May 27, 2014
2017-09-27 19:57:29 +02:00
" For bugs, patches and license go to https://github.com/rust-lang/rust.vim
2014-07-29 13:03:49 +02:00
2019-03-04 09:15:44 +01:00
function! rust#Load()
2018-07-08 15:16:28 +02:00
" Utility call to get this script loaded, for debugging
endfunction
function! rust#GetConfigVar(name, default)
" Local buffer variable with same name takes predeence over global
2019-03-04 09:15:44 +01:00
if has_key(b:, a:name)
2018-07-08 15:16:28 +02:00
return get(b:, a:name)
endif
2019-03-04 09:15:44 +01:00
if has_key(g:, a:name)
2018-07-08 15:16:28 +02:00
return get(g:, a:name)
endif
return a:default
endfunction
2019-03-04 09:15:44 +01:00
" Include expression {{{1
function! rust#IncludeExpr(fname) abort
" Remove leading 'crate::' to deal with 2018 edition style 'use'
" statements
let l:fname = substitute(a:fname, '^crate::', '', '')
" Remove trailing colons arising from lines like
"
" use foo::{Bar, Baz};
let l:fname = substitute(l:fname, ':\+$', '', '')
" Replace '::' with '/'
let l:fname = substitute(l:fname, '::', '/', 'g')
" When we have
"
" use foo::bar::baz;
"
" we can't tell whether baz is a module or a function; and we can't tell
" which modules correspond to files.
"
" So we work our way up, trying
"
" foo/bar/baz.rs
" foo/bar.rs
" foo.rs
while l:fname !=# '.'
let l:path = findfile(l:fname)
if !empty(l:path)
return l:fname
endif
let l:fname = fnamemodify(l:fname, ':h')
endwhile
return l:fname
endfunction
" Jump {{{1
2018-07-08 15:16:28 +02:00
2014-07-29 13:03:49 +02:00
function! rust#Jump(mode, function) range
2018-07-08 15:16:28 +02:00
let cnt = v:count1
normal! m'
if a:mode ==# 'v'
norm! gv
endif
let foldenable = &foldenable
set nofoldenable
while cnt > 0
execute "call <SID>Jump_" . a:function . "()"
let cnt = cnt - 1
endwhile
let &foldenable = foldenable
2014-07-29 13:03:49 +02:00
endfunction
function! s:Jump_Back()
2018-07-08 15:16:28 +02:00
call search('{', 'b')
keepjumps normal! w99[{
2014-07-29 13:03:49 +02:00
endfunction
function! s:Jump_Forward()
2018-07-08 15:16:28 +02:00
normal! j0
call search('{', 'b')
keepjumps normal! w99[{%
call search('{')
2014-07-29 13:03:49 +02:00
endfunction
" Run {{{1
function! rust#Run(bang, args)
2018-07-08 15:16:28 +02:00
let args = s:ShellTokenize(a:args)
if a:bang
let idx = index(l:args, '--')
if idx != -1
let rustc_args = idx == 0 ? [] : l:args[:idx-1]
let args = l:args[idx+1:]
else
let rustc_args = l:args
let args = []
endif
else
let rustc_args = []
endif
let b:rust_last_rustc_args = l:rustc_args
let b:rust_last_args = l:args
call s:WithPath(function("s:Run"), rustc_args, args)
2014-07-29 13:03:49 +02:00
endfunction
2015-12-06 11:31:38 +01:00
function! s:Run(dict, rustc_args, args)
2018-07-08 15:16:28 +02:00
let exepath = a:dict.tmpdir.'/'.fnamemodify(a:dict.path, ':t:r')
if has('win32')
let exepath .= '.exe'
endif
let relpath = get(a:dict, 'tmpdir_relpath', a:dict.path)
let rustc_args = [relpath, '-o', exepath] + a:rustc_args
let rustc = exists("g:rustc_path") ? g:rustc_path : "rustc"
let pwd = a:dict.istemp ? a:dict.tmpdir : ''
let output = s:system(pwd, shellescape(rustc) . " " . join(map(rustc_args, 'shellescape(v:val)')))
2018-10-08 19:00:59 +02:00
if output !=# ''
2018-07-08 15:16:28 +02:00
echohl WarningMsg
echo output
echohl None
endif
if !v:shell_error
exe '!' . shellescape(exepath) . " " . join(map(a:args, 'shellescape(v:val)'))
endif
2014-07-29 13:03:49 +02:00
endfunction
" Expand {{{1
function! rust#Expand(bang, args)
2018-07-08 15:16:28 +02:00
let args = s:ShellTokenize(a:args)
if a:bang && !empty(l:args)
let pretty = remove(l:args, 0)
else
let pretty = "expanded"
endif
call s:WithPath(function("s:Expand"), pretty, args)
2014-07-29 13:03:49 +02:00
endfunction
2015-12-06 11:31:38 +01:00
function! s:Expand(dict, pretty, args)
2018-07-08 15:16:28 +02:00
try
let rustc = exists("g:rustc_path") ? g:rustc_path : "rustc"
if a:pretty =~? '^\%(everybody_loops$\|flowgraph=\)'
let flag = '--xpretty'
else
let flag = '--pretty'
endif
let relpath = get(a:dict, 'tmpdir_relpath', a:dict.path)
let args = [relpath, '-Z', 'unstable-options', l:flag, a:pretty] + a:args
let pwd = a:dict.istemp ? a:dict.tmpdir : ''
let output = s:system(pwd, shellescape(rustc) . " " . join(map(args, 'shellescape(v:val)')))
if v:shell_error
echohl WarningMsg
echo output
echohl None
else
new
silent put =output
1
d
setl filetype=rust
setl buftype=nofile
setl bufhidden=hide
setl noswapfile
" give the buffer a nice name
let suffix = 1
let basename = fnamemodify(a:dict.path, ':t:r')
while 1
let bufname = basename
if suffix > 1 | let bufname .= ' ('.suffix.')' | endif
let bufname .= '.pretty.rs'
if bufexists(bufname)
let suffix += 1
continue
endif
exe 'silent noautocmd keepalt file' fnameescape(bufname)
break
endwhile
endif
endtry
2014-07-29 13:03:49 +02:00
endfunction
function! rust#CompleteExpand(lead, line, pos)
2018-10-08 19:00:59 +02:00
if a:line[: a:pos-1] =~# '^RustExpand!\s*\S*$'
2018-07-08 15:16:28 +02:00
" first argument and it has a !
let list = ["normal", "expanded", "typed", "expanded,identified", "flowgraph=", "everybody_loops"]
if !empty(a:lead)
call filter(list, "v:val[:len(a:lead)-1] == a:lead")
endif
return list
endif
return glob(escape(a:lead, "*?[") . '*', 0, 1)
2014-07-29 13:03:49 +02:00
endfunction
" Emit {{{1
function! rust#Emit(type, args)
2018-07-08 15:16:28 +02:00
let args = s:ShellTokenize(a:args)
call s:WithPath(function("s:Emit"), a:type, args)
2014-07-29 13:03:49 +02:00
endfunction
2015-12-06 11:31:38 +01:00
function! s:Emit(dict, type, args)
2018-07-08 15:16:28 +02:00
try
let output_path = a:dict.tmpdir.'/output'
let rustc = exists("g:rustc_path") ? g:rustc_path : "rustc"
let relpath = get(a:dict, 'tmpdir_relpath', a:dict.path)
let args = [relpath, '--emit', a:type, '-o', output_path] + a:args
let pwd = a:dict.istemp ? a:dict.tmpdir : ''
let output = s:system(pwd, shellescape(rustc) . " " . join(map(args, 'shellescape(v:val)')))
2018-10-08 19:00:59 +02:00
if output !=# ''
2018-07-08 15:16:28 +02:00
echohl WarningMsg
echo output
echohl None
endif
if !v:shell_error
new
exe 'silent keepalt read' fnameescape(output_path)
1
d
2018-10-08 19:00:59 +02:00
if a:type ==# "llvm-ir"
2018-07-08 15:16:28 +02:00
setl filetype=llvm
let extension = 'll'
2018-10-08 19:00:59 +02:00
elseif a:type ==# "asm"
2018-07-08 15:16:28 +02:00
setl filetype=asm
let extension = 's'
endif
setl buftype=nofile
setl bufhidden=hide
setl noswapfile
if exists('l:extension')
" give the buffer a nice name
let suffix = 1
let basename = fnamemodify(a:dict.path, ':t:r')
while 1
let bufname = basename
if suffix > 1 | let bufname .= ' ('.suffix.')' | endif
let bufname .= '.'.extension
if bufexists(bufname)
let suffix += 1
continue
endif
exe 'silent noautocmd keepalt file' fnameescape(bufname)
break
endwhile
endif
endif
endtry
2014-07-29 13:03:49 +02:00
endfunction
" Utility functions {{{1
2015-12-06 11:31:38 +01:00
" Invokes func(dict, ...)
" Where {dict} is a dictionary with the following keys:
" 'path' - The path to the file
" 'tmpdir' - The path to a temporary directory that will be deleted when the
" function returns.
" 'istemp' - 1 if the path is a file inside of {dict.tmpdir} or 0 otherwise.
" If {istemp} is 1 then an additional key is provided:
" 'tmpdir_relpath' - The {path} relative to the {tmpdir}.
"
" {dict.path} may be a path to a file inside of {dict.tmpdir} or it may be the
" existing path of the current buffer. If the path is inside of {dict.tmpdir}
" then it is guaranteed to have a '.rs' extension.
2014-07-29 13:03:49 +02:00
function! s:WithPath(func, ...)
2018-07-08 15:16:28 +02:00
let buf = bufnr('')
let saved = {}
let dict = {}
try
let saved.write = &write
set write
let dict.path = expand('%')
let pathisempty = empty(dict.path)
" Always create a tmpdir in case the wrapped command wants it
let dict.tmpdir = tempname()
call mkdir(dict.tmpdir)
if pathisempty || !saved.write
let dict.istemp = 1
" if we're doing this because of nowrite, preserve the filename
if !pathisempty
let filename = expand('%:t:r').".rs"
else
let filename = 'unnamed.rs'
endif
let dict.tmpdir_relpath = filename
let dict.path = dict.tmpdir.'/'.filename
2018-10-08 19:00:59 +02:00
let saved.mod = &modified
set nomodified
2018-07-08 15:16:28 +02:00
silent exe 'keepalt write! ' . fnameescape(dict.path)
if pathisempty
silent keepalt 0file
endif
else
let dict.istemp = 0
update
endif
call call(a:func, [dict] + a:000)
finally
if bufexists(buf)
for [opt, value] in items(saved)
silent call setbufvar(buf, '&'.opt, value)
unlet value " avoid variable type mismatches
endfor
endif
if has_key(dict, 'tmpdir') | silent call s:RmDir(dict.tmpdir) | endif
endtry
2014-07-29 13:03:49 +02:00
endfunction
function! rust#AppendCmdLine(text)
2018-07-08 15:16:28 +02:00
call setcmdpos(getcmdpos())
let cmd = getcmdline() . a:text
return cmd
2014-07-29 13:03:49 +02:00
endfunction
2015-12-06 11:31:38 +01:00
" Tokenize the string according to sh parsing rules
function! s:ShellTokenize(text)
2018-07-08 15:16:28 +02:00
" states:
" 0: start of word
" 1: unquoted
" 2: unquoted backslash
" 3: double-quote
" 4: double-quoted backslash
" 5: single-quote
let l:state = 0
let l:current = ''
let l:args = []
for c in split(a:text, '\zs')
if l:state == 0 || l:state == 1 " unquoted
if l:c ==# ' '
if l:state == 0 | continue | endif
call add(l:args, l:current)
let l:current = ''
let l:state = 0
elseif l:c ==# '\'
let l:state = 2
elseif l:c ==# '"'
let l:state = 3
elseif l:c ==# "'"
let l:state = 5
else
let l:current .= l:c
let l:state = 1
endif
elseif l:state == 2 " unquoted backslash
if l:c !=# "\n" " can it even be \n?
let l:current .= l:c
endif
let l:state = 1
elseif l:state == 3 " double-quote
if l:c ==# '\'
let l:state = 4
elseif l:c ==# '"'
let l:state = 1
else
let l:current .= l:c
endif
elseif l:state == 4 " double-quoted backslash
if stridx('$`"\', l:c) >= 0
let l:current .= l:c
elseif l:c ==# "\n" " is this even possible?
" skip it
else
let l:current .= '\'.l:c
endif
let l:state = 3
elseif l:state == 5 " single-quoted
2018-10-08 19:00:59 +02:00
if l:c ==# "'"
2018-07-08 15:16:28 +02:00
let l:state = 1
else
let l:current .= l:c
endif
endif
endfor
if l:state != 0
call add(l:args, l:current)
endif
return l:args
2015-12-06 11:31:38 +01:00
endfunction
2014-07-29 13:03:49 +02:00
function! s:RmDir(path)
2018-07-08 15:16:28 +02:00
" sanity check; make sure it's not empty, /, or $HOME
if empty(a:path)
echoerr 'Attempted to delete empty path'
return 0
2018-10-08 19:00:59 +02:00
elseif a:path ==# '/' || a:path ==# $HOME
2018-07-08 15:16:28 +02:00
echoerr 'Attempted to delete protected path: ' . a:path
return 0
endif
return system("rm -rf " . shellescape(a:path))
2014-07-29 13:03:49 +02:00
endfunction
2015-12-06 11:31:38 +01:00
" Executes {cmd} with the cwd set to {pwd}, without changing Vim's cwd.
" If {pwd} is the empty string then it doesn't change the cwd.
function! s:system(pwd, cmd)
2018-07-08 15:16:28 +02:00
let cmd = a:cmd
if !empty(a:pwd)
let cmd = 'cd ' . shellescape(a:pwd) . ' && ' . cmd
endif
return system(cmd)
2015-12-06 11:31:38 +01:00
endfunction
" Playpen Support {{{1
" Parts of gist.vim by Yasuhiro Matsumoto <mattn.jp@gmail.com> reused
" gist.vim available under the BSD license, available at
" http://github.com/mattn/gist-vim
function! s:has_webapi()
2017-09-27 19:57:29 +02:00
if !exists("*webapi#http#post")
2018-07-08 15:16:28 +02:00
try
call webapi#http#post()
catch
endtry
2017-09-27 19:57:29 +02:00
endif
return exists("*webapi#http#post")
2015-12-06 11:31:38 +01:00
endfunction
function! rust#Play(count, line1, line2, ...) abort
2017-09-27 19:57:29 +02:00
redraw
let l:rust_playpen_url = get(g:, 'rust_playpen_url', 'https://play.rust-lang.org/')
let l:rust_shortener_url = get(g:, 'rust_shortener_url', 'https://is.gd/')
if !s:has_webapi()
2018-07-08 15:16:28 +02:00
echohl ErrorMsg | echomsg ':RustPlay depends on webapi.vim (https://github.com/mattn/webapi-vim)' | echohl None
return
2017-09-27 19:57:29 +02:00
endif
let bufname = bufname('%')
if a:count < 1
2018-07-08 15:16:28 +02:00
let content = join(getline(a:line1, a:line2), "\n")
2017-09-27 19:57:29 +02:00
else
2018-07-08 15:16:28 +02:00
let save_regcont = @"
let save_regtype = getregtype('"')
silent! normal! gvy
let content = @"
call setreg('"', save_regcont, save_regtype)
2017-09-27 19:57:29 +02:00
endif
2018-10-08 19:00:59 +02:00
let url = l:rust_playpen_url."?code=".webapi#http#encodeURI(content)
2017-09-27 19:57:29 +02:00
2018-10-08 19:00:59 +02:00
if strlen(url) > 5000
echohl ErrorMsg | echomsg 'Buffer too large, max 5000 encoded characters ('.strlen(url).')' | echohl None
2018-07-08 15:16:28 +02:00
return
2017-09-27 19:57:29 +02:00
endif
2018-10-08 19:00:59 +02:00
let payload = "format=simple&url=".webapi#http#encodeURI(url)
2017-09-27 19:57:29 +02:00
let res = webapi#http#post(l:rust_shortener_url.'create.php', payload, {})
2018-10-08 19:00:59 +02:00
if res.status[0] ==# '2'
let url = res.content
endif
2017-09-27 19:57:29 +02:00
2018-10-08 19:00:59 +02:00
let footer = ''
2017-09-27 19:57:29 +02:00
if exists('g:rust_clip_command')
2018-07-08 15:16:28 +02:00
call system(g:rust_clip_command, url)
2018-10-08 19:00:59 +02:00
if !v:shell_error
let footer = ' (copied to clipboard)'
endif
2017-09-27 19:57:29 +02:00
endif
2018-10-08 19:00:59 +02:00
redraw | echomsg 'Done: '.url.footer
endfunction
2017-09-27 19:57:29 +02:00
2018-10-08 19:00:59 +02:00
" Run a test under the cursor or all tests {{{1
" Finds a test function name under the cursor. Returns empty string when a
" test function is not found.
function! s:SearchTestFunctionNameUnderCursor() abort
let cursor_line = line('.')
" Find #[test] attribute
2018-12-26 10:41:57 +01:00
if search('\m\C#\[test\]', 'bcW') is 0
2018-10-08 19:00:59 +02:00
return ''
endif
" Move to an opening brace of the test function
2018-12-26 10:41:57 +01:00
let test_func_line = search('\m\C^\s*fn\s\+\h\w*\s*(.\+{$', 'eW')
2018-10-08 19:00:59 +02:00
if test_func_line is 0
return ''
endif
" Search the end of test function (closing brace) to ensure that the
" cursor position is within function definition
normal! %
if line('.') < cursor_line
return ''
endif
2018-12-26 10:41:57 +01:00
return matchstr(getline(test_func_line), '\m\C^\s*fn\s\+\zs\h\w*')
2018-10-08 19:00:59 +02:00
endfunction
function! rust#Test(all, options) abort
2018-12-26 10:41:57 +01:00
let manifest = findfile('Cargo.toml', expand('%:p:h') . ';')
if manifest ==# ''
2018-10-08 19:00:59 +02:00
return rust#Run(1, '--test ' . a:options)
endif
2018-12-26 10:41:57 +01:00
if exists(':terminal')
let cmd = 'terminal '
else
let cmd = '!'
let manifest = shellescape(manifest)
endif
2018-10-08 19:00:59 +02:00
if a:all
2018-12-26 10:41:57 +01:00
if a:options ==# ''
execute cmd . 'cargo test --manifest-path' manifest
else
execute cmd . 'cargo test --manifest-path' manifest a:options
endif
return
endif
2018-10-08 19:00:59 +02:00
let saved = getpos('.')
try
let func_name = s:SearchTestFunctionNameUnderCursor()
if func_name ==# ''
echohl ErrorMsg
echo 'No test function was found under the cursor. Please add ! to command if you want to run all tests'
echohl None
return
endif
2018-12-26 10:41:57 +01:00
if a:options ==# ''
2019-03-04 09:15:44 +01:00
execute cmd . 'cargo test --manifest-path' manifest func_name
2018-12-26 10:41:57 +01:00
else
2019-03-04 09:15:44 +01:00
execute cmd . 'cargo test --manifest-path' manifest func_name a:options
2018-12-26 10:41:57 +01:00
endif
return
2018-10-08 19:00:59 +02:00
finally
call setpos('.', saved)
endtry
2015-12-06 11:31:38 +01:00
endfunction
2014-07-29 13:03:49 +02:00
" }}}1
2018-07-08 15:16:28 +02:00
" vim: set et sw=4 sts=4 ts=8: