1
0
mirror of https://github.com/SpaceVim/SpaceVim.git synced 2025-01-24 06:20:05 +08:00
SpaceVim/bundle/vim-hug-neovim-rpc/autoload/neovim_rpc.vim
2020-06-18 23:07:37 +08:00

205 lines
5.5 KiB
VimL

if has('pythonx')
let g:neovim_rpc#py = 'pythonx'
let s:pyeval = function('pyxeval')
elseif has('python3')
let g:neovim_rpc#py = 'python3'
let s:pyeval = function('py3eval')
else
let g:neovim_rpc#py = 'python'
let s:pyeval = function('pyeval')
endif
func! s:py(cmd)
execute g:neovim_rpc#py a:cmd
endfunc
func! neovim_rpc#serveraddr()
if exists('g:_neovim_rpc_nvim_server')
return g:_neovim_rpc_nvim_server
endif
if &encoding !=? "utf-8"
throw '[vim-hug-neovim-rpc] requires `:set encoding=utf-8`'
endif
try
call s:py('import pynvim')
catch
try
call s:py('import neovim')
catch
call neovim_rpc#_error("failed executing: " .
\ g:neovim_rpc#py . " import [pynvim|neovim]")
call neovim_rpc#_error(v:exception)
throw '[vim-hug-neovim-rpc] requires one of `:' . g:neovim_rpc#py .
\ ' import [pynvim|neovim]` command to work'
endtry
endtry
call s:py('import neovim_rpc_server')
let l:servers = s:pyeval('neovim_rpc_server.start()')
let g:_neovim_rpc_nvim_server = l:servers[0]
let g:_neovim_rpc_vim_server = l:servers[1]
let g:_neovim_rpc_main_channel = ch_open(g:_neovim_rpc_vim_server)
" identify myself
call ch_sendexpr(g:_neovim_rpc_main_channel,'neovim_rpc_setup')
return g:_neovim_rpc_nvim_server
endfunc
" elegant python function call wrapper
func! neovim_rpc#pyxcall(func,...)
call s:py('import vim, json')
let g:neovim_rpc#_tmp_args = copy(a:000)
let l:ret = s:pyeval(a:func . '(*vim.vars["neovim_rpc#_tmp_args"])')
unlet g:neovim_rpc#_tmp_args
return l:ret
endfunc
" supported opt keys:
" - on_stdout
" - on_stderr
" - on_exit
" - detach
func! neovim_rpc#jobstart(cmd,...)
let l:opts = {}
if len(a:000)
let l:opts = a:1
endif
let l:opts['_close'] = 0
let l:opts['_exit'] = 0
let l:real_opts = {'mode': 'raw'}
if has_key(l:opts,'detach') && l:opts['detach']
let l:real_opts['stoponexit'] = ''
endif
if has_key(l:opts,'on_stdout')
let l:real_opts['out_cb'] = function('neovim_rpc#_on_stdout')
endif
if has_key(l:opts,'on_stderr')
let l:real_opts['err_cb'] = function('neovim_rpc#_on_stderr')
endif
let l:real_opts['exit_cb'] = function('neovim_rpc#_on_exit')
let l:real_opts['close_cb'] = function('neovim_rpc#_on_close')
let l:job = job_start(a:cmd, l:real_opts)
let l:jobid = ch_info(l:job)['id']
let g:_neovim_rpc_jobs[l:jobid] = {'cmd':a:cmd, 'opts': l:opts, 'job': l:job}
return l:jobid
endfunc
func! neovim_rpc#jobstop(jobid)
let l:job = g:_neovim_rpc_jobs[a:jobid]['job']
return job_stop(l:job)
endfunc
func! neovim_rpc#rpcnotify(channel,event,...)
call neovim_rpc#pyxcall('neovim_rpc_server.rpcnotify',a:channel,a:event,a:000)
endfunc
let s:rspid = 1
func! neovim_rpc#rpcrequest(channel, event, ...)
let s:rspid = s:rspid + 1
" a unique key for storing response
let rspid = '' . s:rspid
" neovim's rpcrequest doesn't have timeout
let opt = {'timeout': 24 * 60 * 60 * 1000}
let args = ['rpcrequest', a:channel, a:event, a:000, rspid]
call ch_evalexpr(g:_neovim_rpc_main_channel, args, opt)
let expr = 'neovim_rpc_server.responses.pop("' . rspid . '")'
call s:py('import neovim_rpc_server, json')
let [err, result] = s:pyeval(expr)
if err
if type(err) == type('')
throw err
endif
throw err[1]
endif
return result
endfunc
func! neovim_rpc#_on_stdout(job,data)
let l:jobid = ch_info(a:job)['id']
let l:opts = g:_neovim_rpc_jobs[l:jobid]['opts']
" convert to neovim style function call
call call(l:opts['on_stdout'],[l:jobid,split(a:data,"\n",1),'stdout'],l:opts)
endfunc
func! neovim_rpc#_on_stderr(job,data)
let l:jobid = ch_info(a:job)['id']
let l:opts = g:_neovim_rpc_jobs[l:jobid]['opts']
" convert to neovim style function call
call call(l:opts['on_stderr'],[l:jobid,split(a:data,"\n",1),'stderr'],l:opts)
endfunc
func! neovim_rpc#_on_exit(job,status)
let l:jobid = ch_info(a:job)['id']
let l:opts = g:_neovim_rpc_jobs[l:jobid]['opts']
let l:opts['_exit'] = 1
" cleanup when both close_cb and exit_cb is called
if l:opts['_close'] && l:opts['_exit']
unlet g:_neovim_rpc_jobs[l:jobid]
endif
if has_key(l:opts, 'on_exit')
" convert to neovim style function call
call call(l:opts['on_exit'],[l:jobid,a:status,'exit'],l:opts)
endif
endfunc
func! neovim_rpc#_on_close(job)
let l:jobid = ch_info(a:job)['id']
let l:opts = g:_neovim_rpc_jobs[l:jobid]['opts']
let l:opts['_close'] = 1
" cleanup when both close_cb and exit_cb is called
if l:opts['_close'] && l:opts['_exit']
unlet g:_neovim_rpc_jobs[l:jobid]
endif
endfunc
func! neovim_rpc#_callback()
execute g:neovim_rpc#py . ' neovim_rpc_server.process_pending_requests()'
endfunc
let g:_neovim_rpc_main_channel = -1
let g:_neovim_rpc_jobs = {}
let s:leaving = 0
func! neovim_rpc#_error(msg)
if mode() == 'i'
" NOTE: side effect, sorry, but this is necessary
set nosmd
endif
echohl ErrorMsg
echom '[vim-hug-neovim-rpc] ' . a:msg
echohl None
endfunc
func! neovim_rpc#_nvim_err_write(msg)
if mode() == 'i'
" NOTE: side effect, sorry, but this is necessary
set nosmd
endif
echohl ErrorMsg
let g:error = a:msg
echom a:msg
echohl None
endfunc
func! neovim_rpc#_nvim_out_write(msg)
echom a:msg
endfunc