2016-11-08 06:34:03 -07:00
|
|
|
let s:shell_error = 0
|
|
|
|
|
2016-09-04 20:36:49 -07:00
|
|
|
function! s:is_bad_response(s) abort
|
2016-11-15 00:27:00 -07:00
|
|
|
return a:s =~? '\v(^unable)|(^error)|(^outdated)'
|
2016-09-04 20:36:49 -07:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:trim(s) abort
|
|
|
|
return substitute(a:s, '^\_s*\|\_s*$', '', 'g')
|
|
|
|
endfunction
|
|
|
|
|
2017-03-27 16:02:42 -07:00
|
|
|
" Convert '\' to '/'. Collapse '//' and '/./'.
|
|
|
|
function! s:normalize_path(s) abort
|
|
|
|
return substitute(substitute(a:s, '\', '/', 'g'), '/\./\|/\+', '/', 'g')
|
|
|
|
endfunction
|
|
|
|
|
2018-01-30 14:35:33 -07:00
|
|
|
" Returns TRUE if `cmd` exits with success, else FALSE.
|
|
|
|
function! s:cmd_ok(cmd) abort
|
|
|
|
call system(a:cmd)
|
|
|
|
return v:shell_error == 0
|
|
|
|
endfunction
|
|
|
|
|
2016-09-04 20:36:49 -07:00
|
|
|
" Simple version comparison.
|
|
|
|
function! s:version_cmp(a, b) abort
|
2016-11-15 00:27:00 -07:00
|
|
|
let a = split(a:a, '\.', 0)
|
|
|
|
let b = split(a:b, '\.', 0)
|
2016-09-04 20:36:49 -07:00
|
|
|
|
|
|
|
for i in range(len(a))
|
2016-11-15 00:27:00 -07:00
|
|
|
if str2nr(a[i]) > str2nr(b[i])
|
2016-09-04 20:36:49 -07:00
|
|
|
return 1
|
2016-11-15 00:27:00 -07:00
|
|
|
elseif str2nr(a[i]) < str2nr(b[i])
|
2016-09-04 20:36:49 -07:00
|
|
|
return -1
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
|
|
|
|
return 0
|
|
|
|
endfunction
|
|
|
|
|
2016-11-08 06:34:03 -07:00
|
|
|
" Handler for s:system() function.
|
2016-12-13 06:48:42 -07:00
|
|
|
function! s:system_handler(jobid, data, event) dict abort
|
2018-07-28 16:46:59 -07:00
|
|
|
if a:event ==# 'stderr'
|
2019-09-18 09:21:44 -07:00
|
|
|
if self.add_stderr_to_output
|
2018-07-28 16:46:59 -07:00
|
|
|
let self.output .= join(a:data, '')
|
2019-09-18 09:21:44 -07:00
|
|
|
else
|
|
|
|
let self.stderr .= join(a:data, '')
|
2018-07-28 16:46:59 -07:00
|
|
|
endif
|
|
|
|
elseif a:event ==# 'stdout'
|
2016-11-12 03:59:15 -07:00
|
|
|
let self.output .= join(a:data, '')
|
2017-04-28 14:18:10 -07:00
|
|
|
elseif a:event ==# 'exit'
|
2016-11-08 06:34:03 -07:00
|
|
|
let s:shell_error = a:data
|
|
|
|
endif
|
|
|
|
endfunction
|
|
|
|
|
2017-05-11 03:16:35 -07:00
|
|
|
" Attempts to construct a shell command from an args list.
|
|
|
|
" Only for display, to help users debug a failed command.
|
|
|
|
function! s:shellify(cmd) abort
|
|
|
|
if type(a:cmd) != type([])
|
|
|
|
return a:cmd
|
|
|
|
endif
|
|
|
|
return join(map(copy(a:cmd),
|
2018-04-30 07:48:32 -07:00
|
|
|
\'v:val =~# ''\m[^\-.a-zA-Z_/]'' ? shellescape(v:val) : v:val'), ' ')
|
2017-05-11 03:16:35 -07:00
|
|
|
endfunction
|
|
|
|
|
2016-11-08 06:34:03 -07:00
|
|
|
" Run a system command and timeout after 30 seconds.
|
|
|
|
function! s:system(cmd, ...) abort
|
|
|
|
let stdin = a:0 ? a:1 : ''
|
2017-01-08 08:23:13 -07:00
|
|
|
let ignore_error = a:0 > 2 ? a:3 : 0
|
2016-11-08 06:34:03 -07:00
|
|
|
let opts = {
|
2019-09-18 09:21:44 -07:00
|
|
|
\ 'add_stderr_to_output': a:0 > 1 ? a:2 : 0,
|
2016-11-12 03:59:15 -07:00
|
|
|
\ 'output': '',
|
2018-07-28 16:46:59 -07:00
|
|
|
\ 'stderr': '',
|
2016-11-08 06:34:03 -07:00
|
|
|
\ 'on_stdout': function('s:system_handler'),
|
2018-07-28 16:46:59 -07:00
|
|
|
\ 'on_stderr': function('s:system_handler'),
|
2016-11-08 06:34:03 -07:00
|
|
|
\ 'on_exit': function('s:system_handler'),
|
|
|
|
\ }
|
|
|
|
let jobid = jobstart(a:cmd, opts)
|
|
|
|
|
|
|
|
if jobid < 1
|
2018-04-30 07:50:01 -07:00
|
|
|
call health#report_error(printf('Command error (job=%d): `%s` (in %s)',
|
|
|
|
\ jobid, s:shellify(a:cmd), string(getcwd())))
|
2016-11-08 06:34:03 -07:00
|
|
|
let s:shell_error = 1
|
2016-11-12 03:59:15 -07:00
|
|
|
return opts.output
|
2016-11-08 06:34:03 -07:00
|
|
|
endif
|
|
|
|
|
|
|
|
if !empty(stdin)
|
|
|
|
call jobsend(jobid, stdin)
|
|
|
|
endif
|
|
|
|
|
|
|
|
let res = jobwait([jobid], 30000)
|
|
|
|
if res[0] == -1
|
2017-05-11 03:16:35 -07:00
|
|
|
call health#report_error(printf('Command timed out: %s', s:shellify(a:cmd)))
|
2016-11-08 06:34:03 -07:00
|
|
|
call jobstop(jobid)
|
2017-01-08 08:23:13 -07:00
|
|
|
elseif s:shell_error != 0 && !ignore_error
|
2019-09-18 09:21:44 -07:00
|
|
|
let emsg = printf("Command error (job=%d, exit code %d): `%s` (in %s)",
|
|
|
|
\ jobid, s:shell_error, s:shellify(a:cmd), string(getcwd()))
|
|
|
|
if !empty(opts.output)
|
|
|
|
let emsg .= "\noutput: " . opts.output
|
|
|
|
end
|
|
|
|
if !empty(opts.stderr)
|
|
|
|
let emsg .= "\nstderr: " . opts.stderr
|
|
|
|
end
|
|
|
|
call health#report_error(emsg)
|
2016-11-08 06:34:03 -07:00
|
|
|
endif
|
|
|
|
|
2016-11-12 03:59:15 -07:00
|
|
|
return opts.output
|
2016-11-08 06:34:03 -07:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:systemlist(cmd, ...) abort
|
|
|
|
let stdout = split(s:system(a:cmd, a:0 ? a:1 : ''), "\n")
|
|
|
|
if a:0 > 1 && !empty(a:2)
|
|
|
|
return filter(stdout, '!empty(v:val)')
|
|
|
|
endif
|
|
|
|
return stdout
|
|
|
|
endfunction
|
|
|
|
|
2016-09-04 20:36:49 -07:00
|
|
|
" Fetch the contents of a URL.
|
|
|
|
function! s:download(url) abort
|
2019-04-20 02:32:14 -07:00
|
|
|
let has_curl = executable('curl')
|
|
|
|
if has_curl && system(['curl', '-V']) =~# 'Protocols:.*https'
|
2017-01-08 08:46:15 -07:00
|
|
|
let rv = s:system(['curl', '-sL', a:url], '', 1, 1)
|
|
|
|
return s:shell_error ? 'curl error with '.a:url.': '.s:shell_error : rv
|
2016-10-13 08:53:38 -07:00
|
|
|
elseif executable('python')
|
2016-09-04 20:36:49 -07:00
|
|
|
let script = "
|
|
|
|
\try:\n
|
|
|
|
\ from urllib.request import urlopen\n
|
|
|
|
\except ImportError:\n
|
|
|
|
\ from urllib2 import urlopen\n
|
|
|
|
\\n
|
2016-10-13 08:53:38 -07:00
|
|
|
\response = urlopen('".a:url."')\n
|
|
|
|
\print(response.read().decode('utf8'))\n
|
2016-09-04 20:36:49 -07:00
|
|
|
\"
|
2016-11-08 06:34:03 -07:00
|
|
|
let rv = s:system(['python', '-c', script])
|
|
|
|
return empty(rv) && s:shell_error
|
|
|
|
\ ? 'python urllib.request error: '.s:shell_error
|
2016-10-13 08:53:38 -07:00
|
|
|
\ : rv
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
2019-04-20 02:32:14 -07:00
|
|
|
return 'missing `curl` '
|
|
|
|
\ .(has_curl ? '(with HTTPS support) ' : '')
|
|
|
|
\ .'and `python`, cannot make web request'
|
2016-09-04 20:36:49 -07:00
|
|
|
endfunction
|
|
|
|
|
2016-10-28 18:39:29 -07:00
|
|
|
" Check for clipboard tools.
|
|
|
|
function! s:check_clipboard() abort
|
2017-04-12 07:37:19 -07:00
|
|
|
call health#report_start('Clipboard (optional)')
|
2016-10-28 18:39:29 -07:00
|
|
|
|
2018-02-07 16:27:54 -07:00
|
|
|
if !empty($TMUX) && executable('tmux') && executable('pbpaste') && !s:cmd_ok('pbpaste')
|
2018-01-30 14:35:33 -07:00
|
|
|
let tmux_version = matchstr(system('tmux -V'), '\d\+\.\d\+')
|
|
|
|
call health#report_error('pbcopy does not work with tmux version: '.tmux_version,
|
|
|
|
\ ['Install tmux 2.6+. https://superuser.com/q/231130',
|
|
|
|
\ 'or use tmux with reattach-to-user-namespace. https://superuser.com/a/413233'])
|
|
|
|
endif
|
|
|
|
|
2016-10-28 18:39:29 -07:00
|
|
|
let clipboard_tool = provider#clipboard#Executable()
|
2017-08-19 17:13:04 -07:00
|
|
|
if exists('g:clipboard') && empty(clipboard_tool)
|
|
|
|
call health#report_error(
|
|
|
|
\ provider#clipboard#Error(),
|
|
|
|
\ ["Use the example in :help g:clipboard as a template, or don't set g:clipboard at all."])
|
|
|
|
elseif empty(clipboard_tool)
|
2016-10-28 18:39:29 -07:00
|
|
|
call health#report_warn(
|
2017-08-19 17:13:04 -07:00
|
|
|
\ 'No clipboard tool found. Clipboard registers (`"+` and `"*`) will not work.',
|
2017-04-17 16:57:19 -07:00
|
|
|
\ [':help clipboard'])
|
2016-10-28 18:39:29 -07:00
|
|
|
else
|
|
|
|
call health#report_ok('Clipboard tool found: '. clipboard_tool)
|
|
|
|
endif
|
|
|
|
endfunction
|
2016-09-04 20:36:49 -07:00
|
|
|
|
2018-11-17 07:27:09 -07:00
|
|
|
" Get the latest Neovim Python client (pynvim) version from PyPI.
|
2016-09-04 20:36:49 -07:00
|
|
|
function! s:latest_pypi_version() abort
|
2016-10-13 08:53:38 -07:00
|
|
|
let pypi_version = 'unable to get pypi response'
|
2018-11-17 07:27:09 -07:00
|
|
|
let pypi_response = s:download('https://pypi.python.org/pypi/pynvim/json')
|
2016-10-13 08:53:38 -07:00
|
|
|
if !empty(pypi_response)
|
|
|
|
try
|
|
|
|
let pypi_data = json_decode(pypi_response)
|
|
|
|
catch /E474/
|
|
|
|
return 'error: '.pypi_response
|
|
|
|
endtry
|
|
|
|
let pypi_version = get(get(pypi_data, 'info', {}), 'version', 'unable to parse')
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
2016-10-13 08:53:38 -07:00
|
|
|
return pypi_version
|
2016-09-04 20:36:49 -07:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
" Get version information using the specified interpreter. The interpreter is
|
|
|
|
" used directly in case breaking changes were introduced since the last time
|
|
|
|
" Neovim's Python client was updated.
|
|
|
|
"
|
|
|
|
" Returns: [
|
|
|
|
" {python executable version},
|
|
|
|
" {current nvim version},
|
|
|
|
" {current pypi nvim status},
|
|
|
|
" {installed version status}
|
|
|
|
" ]
|
|
|
|
function! s:version_info(python) abort
|
|
|
|
let pypi_version = s:latest_pypi_version()
|
2016-11-08 06:34:03 -07:00
|
|
|
let python_version = s:trim(s:system([
|
2016-09-04 20:36:49 -07:00
|
|
|
\ a:python,
|
|
|
|
\ '-c',
|
|
|
|
\ 'import sys; print(".".join(str(x) for x in sys.version_info[:3]))',
|
|
|
|
\ ]))
|
|
|
|
|
|
|
|
if empty(python_version)
|
2017-05-10 06:33:11 -07:00
|
|
|
let python_version = 'unable to parse '.a:python.' response'
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
|
|
|
|
2016-11-08 06:34:03 -07:00
|
|
|
let nvim_path = s:trim(s:system([
|
2018-04-30 07:54:35 -07:00
|
|
|
\ a:python, '-c',
|
|
|
|
\ 'import sys; sys.path.remove(""); ' .
|
2019-01-01 10:04:41 -07:00
|
|
|
\ 'import neovim; print(neovim.__file__)']))
|
2016-10-22 08:35:01 -07:00
|
|
|
if s:shell_error || empty(nvim_path)
|
2019-01-01 10:04:41 -07:00
|
|
|
return [python_version, 'unable to load neovim Python module', pypi_version,
|
2016-10-22 08:35:01 -07:00
|
|
|
\ nvim_path]
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
|
|
|
|
2016-10-08 07:34:54 -07:00
|
|
|
" Assuming that multiple versions of a package are installed, sort them
|
|
|
|
" numerically in descending order.
|
2017-04-28 14:18:10 -07:00
|
|
|
function! s:compare(metapath1, metapath2) abort
|
2016-10-08 07:34:54 -07:00
|
|
|
let a = matchstr(fnamemodify(a:metapath1, ':p:h:t'), '[0-9.]\+')
|
|
|
|
let b = matchstr(fnamemodify(a:metapath2, ':p:h:t'), '[0-9.]\+')
|
|
|
|
return a == b ? 0 : a > b ? 1 : -1
|
|
|
|
endfunction
|
|
|
|
|
2019-01-01 10:04:41 -07:00
|
|
|
" Try to get neovim.VERSION (added in 0.1.11dev).
|
2017-05-10 06:33:11 -07:00
|
|
|
let nvim_version = s:system([a:python, '-c',
|
2019-01-01 10:04:41 -07:00
|
|
|
\ 'from neovim import VERSION as v; '.
|
2017-01-08 08:23:13 -07:00
|
|
|
\ 'print("{}.{}.{}{}".format(v.major, v.minor, v.patch, v.prerelease))'],
|
|
|
|
\ '', 1, 1)
|
|
|
|
if empty(nvim_version)
|
2019-01-01 10:04:41 -07:00
|
|
|
let nvim_version = 'unable to find neovim Python module version'
|
2017-01-08 08:23:13 -07:00
|
|
|
let base = fnamemodify(nvim_path, ':h')
|
|
|
|
let metas = glob(base.'-*/METADATA', 1, 1)
|
|
|
|
\ + glob(base.'-*/PKG-INFO', 1, 1)
|
|
|
|
\ + glob(base.'.egg-info/PKG-INFO', 1, 1)
|
|
|
|
let metas = sort(metas, 's:compare')
|
|
|
|
|
|
|
|
if !empty(metas)
|
|
|
|
for meta_line in readfile(metas[0])
|
|
|
|
if meta_line =~# '^Version:'
|
|
|
|
let nvim_version = matchstr(meta_line, '^Version: \zs\S\+')
|
|
|
|
break
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
endif
|
2016-10-08 07:34:54 -07:00
|
|
|
endif
|
2016-09-04 20:36:49 -07:00
|
|
|
|
2017-01-08 05:32:55 -07:00
|
|
|
let nvim_path_base = fnamemodify(nvim_path, ':~:h')
|
|
|
|
let version_status = 'unknown; '.nvim_path_base
|
2016-09-04 20:36:49 -07:00
|
|
|
if !s:is_bad_response(nvim_version) && !s:is_bad_response(pypi_version)
|
|
|
|
if s:version_cmp(nvim_version, pypi_version) == -1
|
2017-01-08 05:32:55 -07:00
|
|
|
let version_status = 'outdated; from '.nvim_path_base
|
2016-09-04 20:36:49 -07:00
|
|
|
else
|
|
|
|
let version_status = 'up to date'
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
|
|
|
|
return [python_version, nvim_version, pypi_version, version_status]
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
" Check the Python interpreter's usability.
|
|
|
|
function! s:check_bin(bin) abort
|
2017-03-27 16:02:42 -07:00
|
|
|
if !filereadable(a:bin) && (!has('win32') || !filereadable(a:bin.'.exe'))
|
2016-09-04 20:36:49 -07:00
|
|
|
call health#report_error(printf('"%s" was not found.', a:bin))
|
|
|
|
return 0
|
|
|
|
elseif executable(a:bin) != 1
|
|
|
|
call health#report_error(printf('"%s" is not executable.', a:bin))
|
|
|
|
return 0
|
|
|
|
endif
|
|
|
|
return 1
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:check_python(version) abort
|
2017-04-12 07:37:19 -07:00
|
|
|
call health#report_start('Python ' . a:version . ' provider (optional)')
|
2016-09-04 20:36:49 -07:00
|
|
|
|
2017-02-13 13:55:12 -07:00
|
|
|
let pyname = 'python'.(a:version == 2 ? '' : '3')
|
2019-01-06 09:19:57 -07:00
|
|
|
let python_exe = ''
|
2016-09-04 20:36:49 -07:00
|
|
|
let venv = exists('$VIRTUAL_ENV') ? resolve($VIRTUAL_ENV) : ''
|
2017-02-13 13:55:12 -07:00
|
|
|
let host_prog_var = pyname.'_host_prog'
|
2017-02-13 16:00:29 -07:00
|
|
|
let loaded_var = 'g:loaded_'.pyname.'_provider'
|
2016-09-04 20:36:49 -07:00
|
|
|
let python_multiple = []
|
|
|
|
|
2017-02-13 16:00:29 -07:00
|
|
|
if exists(loaded_var) && !exists('*provider#'.pyname.'#Call')
|
2019-09-21 16:17:22 -07:00
|
|
|
let v = eval(loaded_var)
|
|
|
|
call health#report_info('Disabled ('.loaded_var.'='.v.').'.(0 is v ? '' : ' This might be due to some previous error.'))
|
|
|
|
if 0 is v
|
|
|
|
return
|
|
|
|
endif
|
2017-02-13 13:55:12 -07:00
|
|
|
endif
|
|
|
|
|
2019-01-06 07:44:39 -07:00
|
|
|
let [pyenv, pyenv_root] = s:check_for_pyenv()
|
2017-08-26 15:29:12 -07:00
|
|
|
|
2016-09-04 20:36:49 -07:00
|
|
|
if exists('g:'.host_prog_var)
|
|
|
|
call health#report_info(printf('Using: g:%s = "%s"', host_prog_var, get(g:, host_prog_var)))
|
|
|
|
endif
|
|
|
|
|
2019-01-06 09:19:57 -07:00
|
|
|
let [pyname, pythonx_errors] = provider#pythonx#Detect(a:version)
|
|
|
|
|
2017-02-13 13:55:12 -07:00
|
|
|
if empty(pyname)
|
2019-01-07 06:46:58 -07:00
|
|
|
call health#report_warn('No Python executable found that can `import neovim`. '
|
2019-01-06 09:19:57 -07:00
|
|
|
\ . 'Using the first available executable for diagnostics.')
|
2018-07-28 16:44:46 -07:00
|
|
|
elseif exists('g:'.host_prog_var)
|
2019-01-06 09:19:57 -07:00
|
|
|
let python_exe = pyname
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
|
|
|
|
2019-01-06 09:19:57 -07:00
|
|
|
" No Python executable could `import neovim`.
|
|
|
|
if !empty(pythonx_errors)
|
|
|
|
call health#report_error('Python provider error:', pythonx_errors)
|
2016-09-04 20:36:49 -07:00
|
|
|
|
2019-01-06 09:19:57 -07:00
|
|
|
elseif !empty(pyname) && empty(python_exe)
|
2016-09-04 20:36:49 -07:00
|
|
|
if !exists('g:'.host_prog_var)
|
|
|
|
call health#report_info(printf('`g:%s` is not set. Searching for '
|
2017-02-13 13:55:12 -07:00
|
|
|
\ . '%s in the environment.', host_prog_var, pyname))
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
|
|
|
|
|
|
|
if !empty(pyenv)
|
2019-01-06 09:19:57 -07:00
|
|
|
let python_exe = s:trim(s:system([pyenv, 'which', pyname], '', 1))
|
2016-09-04 20:36:49 -07:00
|
|
|
|
2019-01-06 09:19:57 -07:00
|
|
|
if empty(python_exe)
|
2017-02-13 13:55:12 -07:00
|
|
|
call health#report_warn(printf('pyenv could not find %s.', pyname))
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
|
|
|
endif
|
|
|
|
|
2019-01-06 09:19:57 -07:00
|
|
|
if empty(python_exe)
|
|
|
|
let python_exe = exepath(pyname)
|
2016-09-04 20:36:49 -07:00
|
|
|
|
|
|
|
if exists('$PATH')
|
2017-02-02 18:15:35 -07:00
|
|
|
for path in split($PATH, has('win32') ? ';' : ':')
|
2017-03-27 16:02:42 -07:00
|
|
|
let path_bin = s:normalize_path(path.'/'.pyname)
|
2019-01-06 09:19:57 -07:00
|
|
|
if path_bin != s:normalize_path(python_exe)
|
2017-03-27 16:02:42 -07:00
|
|
|
\ && index(python_multiple, path_bin) == -1
|
2016-09-04 20:36:49 -07:00
|
|
|
\ && executable(path_bin)
|
|
|
|
call add(python_multiple, path_bin)
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
|
|
|
|
if len(python_multiple)
|
|
|
|
" This is worth noting since the user may install something
|
|
|
|
" that changes $PATH, like homebrew.
|
2017-01-08 13:26:00 -07:00
|
|
|
call health#report_info(printf('Multiple %s executables found. '
|
2017-02-13 13:55:12 -07:00
|
|
|
\ . 'Set `g:%s` to avoid surprises.', pyname, host_prog_var))
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
|
|
|
|
2019-01-06 09:19:57 -07:00
|
|
|
if python_exe =~# '\<shims\>'
|
|
|
|
call health#report_warn(printf('`%s` appears to be a pyenv shim.', python_exe), [
|
2017-08-26 15:29:12 -07:00
|
|
|
\ '`pyenv` is not in $PATH, your pyenv installation is broken. '
|
|
|
|
\ .'Set `g:'.host_prog_var.'` to avoid surprises.',
|
2016-09-04 20:36:49 -07:00
|
|
|
\ ])
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
|
2019-01-06 09:19:57 -07:00
|
|
|
if !empty(python_exe) && !exists('g:'.host_prog_var)
|
2018-03-04 12:04:52 -07:00
|
|
|
if empty(venv) && !empty(pyenv)
|
2019-01-06 09:19:57 -07:00
|
|
|
\ && !empty(pyenv_root) && resolve(python_exe) !~# '^'.pyenv_root.'/'
|
2016-09-04 20:36:49 -07:00
|
|
|
call health#report_warn('pyenv is not set up optimally.', [
|
2017-01-08 13:26:00 -07:00
|
|
|
\ printf('Create a virtualenv specifically '
|
|
|
|
\ . 'for Neovim using pyenv, and set `g:%s`. This will avoid '
|
2018-11-17 07:27:09 -07:00
|
|
|
\ . 'the need to install the pynvim module in each '
|
2016-09-04 20:36:49 -07:00
|
|
|
\ . 'version/virtualenv.', host_prog_var)
|
|
|
|
\ ])
|
2018-03-04 12:04:52 -07:00
|
|
|
elseif !empty(venv)
|
2016-09-04 20:36:49 -07:00
|
|
|
if !empty(pyenv_root)
|
|
|
|
let venv_root = pyenv_root
|
|
|
|
else
|
|
|
|
let venv_root = fnamemodify(venv, ':h')
|
|
|
|
endif
|
|
|
|
|
2019-01-06 09:19:57 -07:00
|
|
|
if resolve(python_exe) !~# '^'.venv_root.'/'
|
2016-09-04 20:36:49 -07:00
|
|
|
call health#report_warn('Your virtualenv is not set up optimally.', [
|
2017-01-08 13:26:00 -07:00
|
|
|
\ printf('Create a virtualenv specifically '
|
|
|
|
\ . 'for Neovim and use `g:%s`. This will avoid '
|
2018-11-17 07:27:09 -07:00
|
|
|
\ . 'the need to install the pynvim module in each '
|
2016-09-04 20:36:49 -07:00
|
|
|
\ . 'virtualenv.', host_prog_var)
|
|
|
|
\ ])
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
endif
|
|
|
|
|
2019-01-06 09:19:57 -07:00
|
|
|
if empty(python_exe) && !empty(pyname)
|
2016-09-04 20:36:49 -07:00
|
|
|
" An error message should have already printed.
|
2017-02-13 13:55:12 -07:00
|
|
|
call health#report_error(printf('`%s` was not found.', pyname))
|
2019-01-06 09:19:57 -07:00
|
|
|
elseif !empty(python_exe) && !s:check_bin(python_exe)
|
|
|
|
let python_exe = ''
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
|
|
|
|
2017-12-29 10:35:05 -07:00
|
|
|
" Check if $VIRTUAL_ENV is valid.
|
2019-01-06 09:19:57 -07:00
|
|
|
if exists('$VIRTUAL_ENV') && !empty(python_exe)
|
|
|
|
if $VIRTUAL_ENV ==# matchstr(python_exe, '^\V'.$VIRTUAL_ENV)
|
2018-06-24 01:28:21 -07:00
|
|
|
call health#report_info('$VIRTUAL_ENV matches executable')
|
|
|
|
else
|
2017-12-27 05:53:01 -07:00
|
|
|
call health#report_warn(
|
|
|
|
\ '$VIRTUAL_ENV exists but appears to be inactive. '
|
|
|
|
\ . 'This could lead to unexpected results.',
|
|
|
|
\ [ 'If you are using Zsh, see: http://vi.stackexchange.com/a/7654' ])
|
|
|
|
endif
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
|
|
|
|
|
|
|
" Diagnostic output
|
2019-01-06 09:19:57 -07:00
|
|
|
call health#report_info('Executable: ' . (empty(python_exe) ? 'Not found' : python_exe))
|
2016-09-04 20:36:49 -07:00
|
|
|
if len(python_multiple)
|
|
|
|
for path_bin in python_multiple
|
|
|
|
call health#report_info('Other python executable: ' . path_bin)
|
|
|
|
endfor
|
|
|
|
endif
|
|
|
|
|
2018-11-20 01:57:43 -07:00
|
|
|
let pip = 'pip' . (a:version == 2 ? '' : '3')
|
|
|
|
|
2019-01-07 06:46:58 -07:00
|
|
|
if empty(python_exe)
|
|
|
|
" No Python executable can import 'neovim'. Check if any Python executable
|
|
|
|
" can import 'pynvim'. If so, that Python failed to import 'neovim' as
|
|
|
|
" well, which is most probably due to a failed pip upgrade:
|
|
|
|
" https://github.com/neovim/neovim/wiki/Following-HEAD#20181118
|
|
|
|
let [pynvim_exe, errors] = provider#pythonx#DetectByModule('pynvim', a:version)
|
|
|
|
if !empty(pynvim_exe)
|
|
|
|
call health#report_error(
|
|
|
|
\ 'Detected pip upgrade failure: Python executable can import "pynvim" but '
|
|
|
|
\ . 'not "neovim": '. pynvim_exe,
|
|
|
|
\ "Use that Python version to reinstall \"pynvim\" and optionally \"neovim\".\n"
|
|
|
|
\ . pip ." uninstall pynvim neovim\n"
|
|
|
|
\ . pip ." install pynvim\n"
|
|
|
|
\ . pip ." install neovim # only if needed by third-party software")
|
|
|
|
endif
|
|
|
|
else
|
2019-01-06 09:19:57 -07:00
|
|
|
let [pyversion, current, latest, status] = s:version_info(python_exe)
|
2019-01-07 06:46:58 -07:00
|
|
|
|
2016-09-04 20:36:49 -07:00
|
|
|
if a:version != str2nr(pyversion)
|
2017-01-08 13:26:00 -07:00
|
|
|
call health#report_warn('Unexpected Python version.' .
|
2016-09-04 20:36:49 -07:00
|
|
|
\ ' This could lead to confusing error messages.')
|
|
|
|
endif
|
2019-01-07 06:46:58 -07:00
|
|
|
|
2016-09-04 20:36:49 -07:00
|
|
|
if a:version == 3 && str2float(pyversion) < 3.3
|
|
|
|
call health#report_warn('Python 3.3+ is recommended.')
|
|
|
|
endif
|
|
|
|
|
2018-11-18 13:38:34 -07:00
|
|
|
call health#report_info('Python version: ' . pyversion)
|
2019-01-07 06:46:58 -07:00
|
|
|
|
2017-01-08 05:32:55 -07:00
|
|
|
if s:is_bad_response(status)
|
2018-11-18 13:38:34 -07:00
|
|
|
call health#report_info(printf('pynvim version: %s (%s)', current, status))
|
2019-01-01 12:21:53 -07:00
|
|
|
else
|
|
|
|
call health#report_info(printf('pynvim version: %s', current))
|
2017-01-08 05:32:55 -07:00
|
|
|
endif
|
2016-09-04 20:36:49 -07:00
|
|
|
|
|
|
|
if s:is_bad_response(current)
|
|
|
|
call health#report_error(
|
2018-11-18 13:38:34 -07:00
|
|
|
\ "pynvim is not installed.\nError: ".current,
|
2018-11-20 01:57:43 -07:00
|
|
|
\ ['Run in shell: '. pip .' install pynvim'])
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
|
|
|
|
|
|
|
if s:is_bad_response(latest)
|
2017-01-08 08:42:35 -07:00
|
|
|
call health#report_warn('Could not contact PyPI to get latest version.')
|
2016-10-13 08:53:38 -07:00
|
|
|
call health#report_error('HTTP request failed: '.latest)
|
2017-01-08 08:42:35 -07:00
|
|
|
elseif s:is_bad_response(status)
|
2018-11-18 13:38:34 -07:00
|
|
|
call health#report_warn(printf('Latest pynvim is NOT installed: %s', latest))
|
2017-01-08 08:42:35 -07:00
|
|
|
elseif !s:is_bad_response(current)
|
2018-11-18 13:38:34 -07:00
|
|
|
call health#report_ok(printf('Latest pynvim is installed.'))
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
|
|
|
endif
|
2019-01-06 07:44:39 -07:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
" Check if pyenv is available and a valid pyenv root can be found, then return
|
|
|
|
" their respective paths. If either of those is invalid, return two empty
|
|
|
|
" strings, effectivly ignoring pyenv.
|
|
|
|
function! s:check_for_pyenv() abort
|
|
|
|
let pyenv_path = resolve(exepath('pyenv'))
|
|
|
|
|
|
|
|
if empty(pyenv_path)
|
|
|
|
return ['', '']
|
|
|
|
endif
|
|
|
|
|
|
|
|
call health#report_info('pyenv: Path: '. pyenv_path)
|
|
|
|
|
|
|
|
let pyenv_root = exists('$PYENV_ROOT') ? resolve($PYENV_ROOT) : ''
|
|
|
|
|
|
|
|
if empty(pyenv_root)
|
|
|
|
let pyenv_root = s:trim(s:system([pyenv_path, 'root']))
|
|
|
|
call health#report_info('pyenv: $PYENV_ROOT is not set. Infer from `pyenv root`.')
|
|
|
|
endif
|
|
|
|
|
|
|
|
if !isdirectory(pyenv_root)
|
|
|
|
call health#report_warn(
|
|
|
|
\ printf('pyenv: Root does not exist: %s. '
|
|
|
|
\ . 'Ignoring pyenv for all following checks.', pyenv_root))
|
|
|
|
return ['', '']
|
|
|
|
endif
|
|
|
|
|
|
|
|
call health#report_info('pyenv: Root: '.pyenv_root)
|
2016-09-04 20:36:49 -07:00
|
|
|
|
2019-01-06 07:44:39 -07:00
|
|
|
return [pyenv_path, pyenv_root]
|
2016-09-04 20:36:49 -07:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:check_ruby() abort
|
2017-04-12 07:37:19 -07:00
|
|
|
call health#report_start('Ruby provider (optional)')
|
2017-01-06 08:11:02 -07:00
|
|
|
|
2017-02-13 16:00:29 -07:00
|
|
|
let loaded_var = 'g:loaded_ruby_provider'
|
|
|
|
if exists(loaded_var) && !exists('*provider#ruby#Call')
|
|
|
|
call health#report_info('Disabled. '.loaded_var.'='.eval(loaded_var))
|
2017-02-13 13:55:12 -07:00
|
|
|
return
|
|
|
|
endif
|
|
|
|
|
2017-01-06 08:11:02 -07:00
|
|
|
if !executable('ruby') || !executable('gem')
|
|
|
|
call health#report_warn(
|
2017-04-28 14:18:10 -07:00
|
|
|
\ '`ruby` and `gem` must be in $PATH.',
|
|
|
|
\ ['Install Ruby and verify that `ruby` and `gem` commands work.'])
|
2017-01-06 08:11:02 -07:00
|
|
|
return
|
2016-11-12 03:59:15 -07:00
|
|
|
endif
|
2017-01-06 08:11:02 -07:00
|
|
|
call health#report_info('Ruby: '. s:system('ruby -v'))
|
|
|
|
|
|
|
|
let host = provider#ruby#Detect()
|
|
|
|
if empty(host)
|
2018-02-21 18:30:39 -07:00
|
|
|
call health#report_warn('`neovim-ruby-host` not found.',
|
2017-12-17 09:47:52 -07:00
|
|
|
\ ['Run `gem install neovim` to ensure the neovim RubyGem is installed.',
|
|
|
|
\ 'Run `gem environment` to ensure the gem bin directory is in $PATH.',
|
|
|
|
\ 'If you are using rvm/rbenv/chruby, try "rehashing".',
|
|
|
|
\ 'See :help g:ruby_host_prog for non-standard gem installations.'])
|
2017-01-06 08:11:02 -07:00
|
|
|
return
|
|
|
|
endif
|
|
|
|
call health#report_info('Host: '. host)
|
|
|
|
|
2017-04-28 07:41:16 -07:00
|
|
|
let latest_gem_cmd = has('win32') ? 'cmd /c gem list -ra ^^neovim$' : 'gem list -ra ^neovim$'
|
2017-01-06 08:11:02 -07:00
|
|
|
let latest_gem = s:system(split(latest_gem_cmd))
|
|
|
|
if s:shell_error || empty(latest_gem)
|
|
|
|
call health#report_error('Failed to run: '. latest_gem_cmd,
|
|
|
|
\ ["Make sure you're connected to the internet.",
|
2017-04-28 14:18:10 -07:00
|
|
|
\ 'Are you behind a firewall or proxy?'])
|
2017-01-06 08:11:02 -07:00
|
|
|
return
|
|
|
|
endif
|
2017-06-03 17:14:35 -07:00
|
|
|
let latest_gem = get(split(latest_gem, 'neovim (\|, \|)$' ), 1, 'not found')
|
2017-01-06 08:11:02 -07:00
|
|
|
|
|
|
|
let current_gem_cmd = host .' --version'
|
|
|
|
let current_gem = s:system(current_gem_cmd)
|
|
|
|
if s:shell_error
|
|
|
|
call health#report_error('Failed to run: '. current_gem_cmd,
|
2017-04-28 14:18:10 -07:00
|
|
|
\ ['Report this issue with the output of: ', current_gem_cmd])
|
2017-01-06 08:11:02 -07:00
|
|
|
return
|
2016-09-04 20:36:49 -07:00
|
|
|
endif
|
|
|
|
|
2017-01-06 08:11:02 -07:00
|
|
|
if s:version_cmp(current_gem, latest_gem) == -1
|
|
|
|
call health#report_warn(
|
|
|
|
\ printf('Gem "neovim" is out-of-date. Installed: %s, latest: %s',
|
|
|
|
\ current_gem, latest_gem),
|
|
|
|
\ ['Run in shell: gem update neovim'])
|
|
|
|
else
|
2017-02-13 13:55:12 -07:00
|
|
|
call health#report_ok('Latest "neovim" gem is installed: '. current_gem)
|
2017-01-06 08:11:02 -07:00
|
|
|
endif
|
2016-09-04 20:36:49 -07:00
|
|
|
endfunction
|
|
|
|
|
2017-10-29 11:06:47 -07:00
|
|
|
function! s:check_node() abort
|
2017-11-28 23:50:11 -07:00
|
|
|
call health#report_start('Node.js provider (optional)')
|
2017-10-29 11:06:47 -07:00
|
|
|
|
|
|
|
let loaded_var = 'g:loaded_node_provider'
|
|
|
|
if exists(loaded_var) && !exists('*provider#node#Call')
|
|
|
|
call health#report_info('Disabled. '.loaded_var.'='.eval(loaded_var))
|
|
|
|
return
|
|
|
|
endif
|
|
|
|
|
2018-06-11 12:13:19 -07:00
|
|
|
if !executable('node') || (!executable('npm') && !executable('yarn'))
|
2017-10-29 11:06:47 -07:00
|
|
|
call health#report_warn(
|
2018-06-11 12:13:19 -07:00
|
|
|
\ '`node` and `npm` (or `yarn`) must be in $PATH.',
|
|
|
|
\ ['Install Node.js and verify that `node` and `npm` (or `yarn`) commands work.'])
|
2017-10-29 11:06:47 -07:00
|
|
|
return
|
|
|
|
endif
|
2017-11-28 23:50:11 -07:00
|
|
|
let node_v = get(split(s:system('node -v'), "\n"), 0, '')
|
|
|
|
call health#report_info('Node.js: '. node_v)
|
|
|
|
if !s:shell_error && s:version_cmp(node_v[1:], '6.0.0') < 0
|
|
|
|
call health#report_warn('Neovim node.js host does not support '.node_v)
|
2017-12-17 07:03:07 -07:00
|
|
|
" Skip further checks, they are nonsense if nodejs is too old.
|
|
|
|
return
|
2017-11-28 23:50:11 -07:00
|
|
|
endif
|
|
|
|
if !provider#node#can_inspect()
|
|
|
|
call health#report_warn('node.js on this system does not support --inspect-brk so $NVIM_NODE_HOST_DEBUG is ignored.')
|
|
|
|
endif
|
2017-10-29 11:06:47 -07:00
|
|
|
|
|
|
|
let host = provider#node#Detect()
|
|
|
|
if empty(host)
|
2018-06-11 12:13:19 -07:00
|
|
|
call health#report_warn('Missing "neovim" npm (or yarn) package.',
|
2017-10-29 11:06:47 -07:00
|
|
|
\ ['Run in shell: npm install -g neovim',
|
2018-06-11 12:13:19 -07:00
|
|
|
\ 'Run in shell (if you use yarn): yarn global add neovim'])
|
2017-10-29 11:06:47 -07:00
|
|
|
return
|
|
|
|
endif
|
2017-11-28 23:50:11 -07:00
|
|
|
call health#report_info('Neovim node.js host: '. host)
|
2017-10-29 11:06:47 -07:00
|
|
|
|
2019-07-14 11:48:53 -07:00
|
|
|
let manager = executable('npm') ? 'npm' : 'yarn'
|
|
|
|
let latest_npm_cmd = has('win32') ?
|
|
|
|
\ 'cmd /c '. manager .' info neovim --json' :
|
|
|
|
\ manager .' info neovim --json'
|
2017-10-29 11:06:47 -07:00
|
|
|
let latest_npm = s:system(split(latest_npm_cmd))
|
|
|
|
if s:shell_error || empty(latest_npm)
|
|
|
|
call health#report_error('Failed to run: '. latest_npm_cmd,
|
|
|
|
\ ["Make sure you're connected to the internet.",
|
|
|
|
\ 'Are you behind a firewall or proxy?'])
|
|
|
|
return
|
|
|
|
endif
|
|
|
|
if !empty(latest_npm)
|
|
|
|
try
|
|
|
|
let pkg_data = json_decode(latest_npm)
|
|
|
|
catch /E474/
|
|
|
|
return 'error: '.latest_npm
|
|
|
|
endtry
|
|
|
|
let latest_npm = get(get(pkg_data, 'dist-tags', {}), 'latest', 'unable to parse')
|
|
|
|
endif
|
|
|
|
|
2017-11-28 23:50:11 -07:00
|
|
|
let current_npm_cmd = ['node', host, '--version']
|
2017-10-29 11:06:47 -07:00
|
|
|
let current_npm = s:system(current_npm_cmd)
|
|
|
|
if s:shell_error
|
2017-12-17 07:03:07 -07:00
|
|
|
call health#report_error('Failed to run: '. string(current_npm_cmd),
|
|
|
|
\ ['Report this issue with the output of: ', string(current_npm_cmd)])
|
2017-10-29 11:06:47 -07:00
|
|
|
return
|
|
|
|
endif
|
|
|
|
|
|
|
|
if s:version_cmp(current_npm, latest_npm) == -1
|
|
|
|
call health#report_warn(
|
|
|
|
\ printf('Package "neovim" is out-of-date. Installed: %s, latest: %s',
|
|
|
|
\ current_npm, latest_npm),
|
2019-07-14 11:48:53 -07:00
|
|
|
\ ['Run in shell: npm install -g neovim',
|
|
|
|
\ 'Run in shell (if you use yarn): yarn global add neovim'])
|
2017-10-29 11:06:47 -07:00
|
|
|
else
|
2018-06-11 12:13:19 -07:00
|
|
|
call health#report_ok('Latest "neovim" npm/yarn package is installed: '. current_npm)
|
2017-10-29 11:06:47 -07:00
|
|
|
endif
|
|
|
|
endfunction
|
|
|
|
|
2016-09-04 20:36:49 -07:00
|
|
|
function! health#provider#check() abort
|
2016-10-28 18:39:29 -07:00
|
|
|
call s:check_clipboard()
|
2016-09-04 20:36:49 -07:00
|
|
|
call s:check_python(2)
|
|
|
|
call s:check_python(3)
|
|
|
|
call s:check_ruby()
|
2017-10-29 11:06:47 -07:00
|
|
|
call s:check_node()
|
2016-09-04 20:36:49 -07:00
|
|
|
endfunction
|