node.vim 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. if exists('g:loaded_node_provider')
  2. finish
  3. endif
  4. let g:loaded_node_provider = 1
  5. function! s:is_minimum_version(version, min_major, min_minor) abort
  6. if empty(a:version)
  7. let nodejs_version = get(split(system(['node', '-v']), "\n"), 0, '')
  8. if v:shell_error || nodejs_version[0] !=# 'v'
  9. return 0
  10. endif
  11. else
  12. let nodejs_version = a:version
  13. endif
  14. " Remove surrounding junk. Example: 'v4.12.0' => '4.12.0'
  15. let nodejs_version = matchstr(nodejs_version, '\(\d\.\?\)\+')
  16. " [major, minor, patch]
  17. let v_list = split(nodejs_version, '\.')
  18. return len(v_list) == 3
  19. \ && ((str2nr(v_list[0]) > str2nr(a:min_major))
  20. \ || (str2nr(v_list[0]) == str2nr(a:min_major)
  21. \ && str2nr(v_list[1]) >= str2nr(a:min_minor)))
  22. endfunction
  23. let s:NodeHandler = {
  24. \ 'stdout_buffered': v:true,
  25. \ 'result': ''
  26. \ }
  27. function! s:NodeHandler.on_exit(job_id, data, event) abort
  28. let bin_dir = join(get(self, 'stdout', []), '')
  29. let entry_point = bin_dir . self.entry_point
  30. let self.result = filereadable(entry_point) ? entry_point : ''
  31. endfunction
  32. " Support for --inspect-brk requires node 6.12+ or 7.6+ or 8+
  33. " Return 1 if it is supported
  34. " Return 0 otherwise
  35. function! provider#node#can_inspect() abort
  36. if !executable('node')
  37. return 0
  38. endif
  39. let ver = get(split(system(['node', '-v']), "\n"), 0, '')
  40. if v:shell_error || ver[0] !=# 'v'
  41. return 0
  42. endif
  43. return (ver[1] ==# '6' && s:is_minimum_version(ver, 6, 12))
  44. \ || s:is_minimum_version(ver, 7, 6)
  45. endfunction
  46. function! provider#node#Detect() abort
  47. let minver = [6, 0]
  48. if exists('g:node_host_prog')
  49. return [expand(g:node_host_prog, v:true), '']
  50. endif
  51. if !executable('node')
  52. return ['', 'node not found (or not executable)']
  53. endif
  54. if !s:is_minimum_version(v:null, minver[0], minver[1])
  55. return ['', printf('node version %s.%s not found', minver[0], minver[1])]
  56. endif
  57. let npm_opts = {}
  58. if executable('npm')
  59. let npm_opts = deepcopy(s:NodeHandler)
  60. let npm_opts.entry_point = '/neovim/bin/cli.js'
  61. let npm_opts.job_id = jobstart('npm --loglevel silent root -g', npm_opts)
  62. endif
  63. let yarn_opts = {}
  64. if executable('yarn')
  65. let yarn_opts = deepcopy(s:NodeHandler)
  66. let yarn_opts.entry_point = '/node_modules/neovim/bin/cli.js'
  67. " `yarn global dir` is slow (> 250ms), try the default path first
  68. " XXX: The following code is not portable
  69. " https://github.com/yarnpkg/yarn/issues/2049#issuecomment-263183768
  70. if has('unix')
  71. let yarn_default_path = $HOME . '/.config/yarn/global/' . yarn_opts.entry_point
  72. if filereadable(yarn_default_path)
  73. return [yarn_default_path, '']
  74. endif
  75. endif
  76. let yarn_opts.job_id = jobstart('yarn global dir', yarn_opts)
  77. endif
  78. let pnpm_opts = {}
  79. if executable('pnpm')
  80. let pnpm_opts = deepcopy(s:NodeHandler)
  81. let pnpm_opts.entry_point = '/neovim/bin/cli.js'
  82. let pnpm_opts.job_id = jobstart('pnpm --loglevel silent root -g', pnpm_opts)
  83. endif
  84. " npm returns the directory faster, so let's check that first
  85. if !empty(npm_opts)
  86. let result = jobwait([npm_opts.job_id])
  87. if result[0] == 0 && npm_opts.result != ''
  88. return [npm_opts.result, '']
  89. endif
  90. endif
  91. if !empty(yarn_opts)
  92. let result = jobwait([yarn_opts.job_id])
  93. if result[0] == 0 && yarn_opts.result != ''
  94. return [yarn_opts.result, '']
  95. endif
  96. endif
  97. if !empty(pnpm_opts)
  98. let result = jobwait([pnpm_opts.job_id])
  99. if result[0] == 0 && pnpm_opts.result != ''
  100. return [pnpm_opts.result, '']
  101. endif
  102. endif
  103. return ['', 'failed to detect node']
  104. endfunction
  105. function! provider#node#Prog() abort
  106. return s:prog
  107. endfunction
  108. function! provider#node#Require(host) abort
  109. if s:err != ''
  110. echoerr s:err
  111. return
  112. endif
  113. let args = ['node']
  114. if !empty($NVIM_NODE_HOST_DEBUG) && provider#node#can_inspect()
  115. call add(args, '--inspect-brk')
  116. endif
  117. call add(args, provider#node#Prog())
  118. return provider#Poll(args, a:host.orig_name, '$NVIM_NODE_LOG_FILE')
  119. endfunction
  120. function! provider#node#Call(method, args) abort
  121. if s:err != ''
  122. echoerr s:err
  123. return
  124. endif
  125. if !exists('s:host')
  126. try
  127. let s:host = remote#host#Require('node')
  128. catch
  129. let s:err = v:exception
  130. echohl WarningMsg
  131. echomsg v:exception
  132. echohl None
  133. return
  134. endtry
  135. endif
  136. return call('rpcrequest', insert(insert(a:args, 'node_'.a:method), s:host))
  137. endfunction
  138. let s:err = ''
  139. let [s:prog, s:_] = provider#node#Detect()
  140. let g:loaded_node_provider = empty(s:prog) ? 1 : 2
  141. if g:loaded_node_provider != 2
  142. let s:err = 'Cannot find the "neovim" node package. Try :checkhealth'
  143. endif
  144. call remote#host#RegisterPlugin('node-provider', 'node', [])