Skip to content

Commit f604437

Browse files
authored
Add native job support for neovim. (#234)
Support Syscall.CallAsync in neovim using neovim jobs. Rename system-vimjob.vroom to system-job.vroom and have it cover both vim and neovim implementations (adding delays as a workaround for async timing quirks in neovim, #235).
1 parent f2abdd1 commit f604437

File tree

5 files changed

+86
-22
lines changed

5 files changed

+86
-22
lines changed

.travis.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ services:
2020
- xvfb
2121
script:
2222
- '[ $CI_TARGET = neovim ] && VROOM_ARGS="--neovim" || VROOM_ARGS=""'
23-
- vroom $VROOM_ARGS --crawl --skip=vroom/system-vimjob.vroom
23+
- vroom $VROOM_ARGS --crawl --skip=vroom/system-job.vroom
2424
matrix:
2525
fast_finish: true
2626
allow_failures:

autoload/maktaba/syscall.vim

+12-8
Original file line numberDiff line numberDiff line change
@@ -126,7 +126,7 @@ endfunction
126126
" Returns whether the current vim session supports asynchronous calls.
127127
function! maktaba#syscall#IsAsyncAvailable() abort
128128
return !s:async_disabled && (
129-
\ (!s:vimjob_disabled && has('job')) ||
129+
\ (!s:vimjob_disabled && (has('job') || has('nvim'))) ||
130130
\ (has('clientserver') && !empty(v:servername)))
131131
endfunction
132132

@@ -312,12 +312,12 @@ endfunction
312312
" env_dict contains tab, buffer, path, column and line info. This fallback will
313313
" be deprecated and stop working in future versions of maktaba.
314314
"
315-
" Asynchronous calls are executed via vim's |job| support. If the vim instance
316-
" is missing job support, this will try to fall back to legacy |clientserver|
317-
" invocation, which has a few preconditions of its own (see below). If neither
318-
" option is available, asynchronous calls are not possible, and the call will
319-
" either throw |ERROR(MissingFeature)| or fall back to synchronous calls,
320-
" depending on the {allow_sync_fallback} parameter.
315+
" Asynchronous calls are executed via vim's |job| support, or neovim's
316+
" |job-control|. If the vim instance is missing job support, this will try to
317+
" fall back to legacy |clientserver| invocation, which has a few preconditions
318+
" of its own (see below). If neither option is available, asynchronous calls are
319+
" not possible, and the call will either throw |ERROR(MissingFeature)| or fall
320+
" back to synchronous calls, depending on the {allow_sync_fallback} parameter.
321321
"
322322
" The legacy fallback executes calls via |--remote-expr| using vim's
323323
" |clientserver| capabilities, so the preconditions for it are vim being
@@ -345,7 +345,7 @@ function! maktaba#syscall#CallAsync(Callback, allow_sync_fallback) abort dict
345345
if s:async_disabled
346346
let l:reason = 'disabled by maktaba#syscall#SetAsyncDisabled'
347347
else
348-
let l:reason = 'no +job support and no fallback available'
348+
let l:reason = 'no +job support, not nvim, and no fallback available'
349349
endif
350350
throw maktaba#error#MissingFeature(
351351
\ 'Cannot run async commands (%s). See :help Syscall.CallAsync',
@@ -357,6 +357,10 @@ function! maktaba#syscall#CallAsync(Callback, allow_sync_fallback) abort dict
357357
let l:vimjob_invocation =
358358
\ maktaba#syscall#async#CreateInvocation(self, l:invocation)
359359
call l:vimjob_invocation.Start()
360+
elseif !s:vimjob_disabled && has('nvim')
361+
let l:vimjob_invocation =
362+
\ maktaba#syscall#neovim#CreateInvocation(self, l:invocation)
363+
call l:vimjob_invocation.Start()
360364
else
361365
" TODO(#190): Remove clientserver implementation.
362366
let l:clientserver_invocation =

autoload/maktaba/syscall/neovim.vim

+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
" CallAsync implementation using neovim's job support.
2+
3+
""
4+
" @private
5+
function! maktaba#syscall#neovim#CreateInvocation(syscall, invocation) abort
6+
" We'll pass this entire invocation to jobstart, which will pass it to the
7+
" on_exit callback.
8+
return {
9+
\ '_syscall': a:syscall,
10+
\ '_invocation': a:invocation,
11+
\ 'Start': function('maktaba#syscall#neovim#Start'),
12+
\ 'stdout_buffered': 1,
13+
\ 'stderr_buffered': 1,
14+
\ 'on_exit': function('maktaba#syscall#neovim#HandleJobExit')}
15+
endfunction
16+
17+
18+
""
19+
" @private
20+
" @dict SyscallNeovimInvocation
21+
" Dispatches syscall through |jobstart()|, and invokes the invocation's
22+
" callback once the command completes, passing in stdout, stderr and exit code
23+
" to it.
24+
" The neovim |job_control| implementation for @function(#CallAsync).
25+
function! maktaba#syscall#neovim#Start() abort dict
26+
let self._job = jobstart(self._syscall.GetCommand(), self)
27+
" Send stdin immediately and close. Streaming input to stdin not supported.
28+
if has_key(self._syscall, 'stdin')
29+
call chansend(self._job, self._syscall.stdin)
30+
endif
31+
call chanclose(self._job, 'stdin')
32+
endfunction
33+
34+
35+
""
36+
" @private
37+
" @dict SyscallNeovimInvocation
38+
function! maktaba#syscall#neovim#HandleJobExit(
39+
\ unused_job,
40+
\ status,
41+
\ unused_event) abort dict
42+
" jobcontrol sets stdout and stderr when no callbacks are given.
43+
call self._invocation.Finish({
44+
\ 'status': a:status,
45+
\ 'stdout': join(self.stdout, "\n"),
46+
\ 'stderr': join(self.stderr, "\n")})
47+
endfunction

doc/maktaba.txt

+7-6
Original file line numberDiff line numberDiff line change
@@ -551,12 +551,13 @@ Syscall.CallAsync({callback}, {allow_sync_fallback}) *Syscall.CallAsync()*
551551
where env_dict contains tab, buffer, path, column and line info. This
552552
fallback will be deprecated and stop working in future versions of maktaba.
553553

554-
Asynchronous calls are executed via vim's |job| support. If the vim instance
555-
is missing job support, this will try to fall back to legacy |clientserver|
556-
invocation, which has a few preconditions of its own (see below). If neither
557-
option is available, asynchronous calls are not possible, and the call will
558-
either throw |ERROR(MissingFeature)| or fall back to synchronous calls,
559-
depending on the {allow_sync_fallback} parameter.
554+
Asynchronous calls are executed via vim's |job| support. For nvim, they are
555+
executed using |job_control|. If the vim instance is missing job support,
556+
this will try to fall back to legacy |clientserver| invocation, which has a
557+
few preconditions of its own (see below). If neither option is available,
558+
asynchronous calls are not possible, and the call will either throw
559+
|ERROR(MissingFeature)| or fall back to synchronous calls, depending on the
560+
{allow_sync_fallback} parameter.
560561

561562
The legacy fallback executes calls via |--remote-expr| using vim's
562563
|clientserver| capabilities, so the preconditions for it are vim being

vroom/system-vimjob.vroom vroom/system-job.vroom

+19-7
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,18 @@
11
In addition to the functionality featured in system.vroom, the maktaba#syscall#
22
helpers feature asynchronous execution powered by vim's native jobs feature.
33

4+
Note that we use delays of 0.1s on the asynchronous calls to ensure nvim has
5+
enough time to flush the contents of stdout to vroom. This shouldn't be
6+
necessary for non-test usage.
7+
(See https://github.com/google/vim-maktaba/pull/234 for more details.)
8+
9+
First, we need to work around for nvim prompting users to press enter.
10+
(See https://github.com/google/vim-codefmt/pull/131)
11+
12+
:if has('nvim')<CR>
13+
| set cmdheight=30<CR>
14+
|endif<CR>
15+
416
Before we dive in, let's get maktaba installed and set up the shell override:
517

618
@system (STRICT)
@@ -14,9 +26,9 @@ Before we dive in, let's get maktaba installed and set up the shell override:
1426
The examples featured in this file only work with vim instances that support
1527
vim jobs.
1628

17-
:if !has('job')<CR>
29+
:if !has('job') && !has('nvim')<CR>
1830
| echomsg maktaba#error#MissingFeature(
19-
| 'Must have +job support to run system-vimjob.vroom examples')<CR>
31+
| 'Must have +job support or nvim to run system-job.vroom examples')<CR>
2032
|endif
2133

2234

@@ -35,7 +47,7 @@ To execute system calls asynchronously, use CallAsync.
3547
:function Callback(result) abort<CR>
3648
| let g:callback_stdout = a:result.stdout<CR>
3749
|endfunction
38-
:let g:invocation = AsyncWait(g:syscall.CallAsync('Callback', 0), 2)
50+
:let g:invocation = AsyncWait(g:syscall.CallAsync('Callback', 0), 2) (0.2s)
3951
! echo hi
4052
:echomsg g:callback_stdout
4153
~ hi
@@ -46,7 +58,7 @@ It is also possible to force synchronous execution.
4658

4759
:call maktaba#syscall#SetAsyncDisabledForTesting(1)
4860
:call maktaba#syscall#ForceSyncFallbackAllowedForTesting(1)
49-
:let g:invocation = AsyncWait(g:syscall.CallAsync('Callback', 0), 2)
61+
:let g:invocation = AsyncWait(g:syscall.CallAsync('Callback', 0), 2) (0.2s)
5062
! echo hi.*
5163

5264
:call maktaba#syscall#SetAsyncDisabledForTesting(0)
@@ -55,21 +67,21 @@ It is also possible to force synchronous execution.
5567
Asynchronous calls can also take a stdin parameter.
5668

5769
:let g:syscall = maktaba#syscall#Create(['cat']).WithStdin("Hello")
58-
:let g:invocation = AsyncWait(g:syscall.CallAsync('Callback', 0), 2)
70+
:let g:invocation = AsyncWait(g:syscall.CallAsync('Callback', 0), 2) (0.2s)
5971
! cat
6072
:echomsg g:invocation.stdout
6173
~ Hello
6274

6375
And() and Or() can also be used to chain Asynchronous commands.
6476

6577
:let g:syscall = maktaba#syscall#Create('true').And('echo SUCCESS')
66-
:let g:invocation = AsyncWait(g:syscall.CallAsync('Callback', 0), 2)
78+
:let g:invocation = AsyncWait(g:syscall.CallAsync('Callback', 0), 2) (0.2s)
6779
! true && echo SUCCESS
6880
:echomsg g:invocation.stdout
6981
~ SUCCESS
7082

7183
:let g:syscall = maktaba#syscall#Create('false').And('echo FAILURE')
72-
:let g:invocation = AsyncWait(g:syscall.CallAsync('Callback', 0), 2)
84+
:let g:invocation = AsyncWait(g:syscall.CallAsync('Callback', 0), 2) (0.2s)
7385
! false && echo FAILURE
7486
:echomsg g:invocation.stdout
7587
~ FAILURE

0 commit comments

Comments
 (0)