Skip to content

Commit 5983ad0

Browse files
committed
patch 7.4.1493
Problem: Wrong callback invoked for zero-id messages. Solution: Don't use the first one-time callback when the sequence number doesn't match.
1 parent 35ca0e7 commit 5983ad0

4 files changed

Lines changed: 79 additions & 6 deletions

File tree

src/channel.c

Lines changed: 6 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -1480,15 +1480,18 @@ may_invoke_callback(channel_T *channel, int part)
14801480
int seq_nr = -1;
14811481
ch_mode_T ch_mode = channel->ch_part[part].ch_mode;
14821482
cbq_T *cbhead = &channel->ch_part[part].ch_cb_head;
1483-
cbq_T *cbitem = cbhead->cq_next;
1483+
cbq_T *cbitem;
14841484
char_u *callback = NULL;
14851485
buf_T *buffer = NULL;
14861486

14871487
if (channel->ch_nb_close_cb != NULL)
14881488
/* this channel is handled elsewhere (netbeans) */
14891489
return FALSE;
14901490

1491-
/* use a message-specific callback, part callback or channel callback */
1491+
/* Use a message-specific callback, part callback or channel callback */
1492+
for (cbitem = cbhead->cq_next; cbitem != NULL; cbitem = cbitem->cq_next)
1493+
if (cbitem->cq_seq_nr == 0)
1494+
break;
14921495
if (cbitem != NULL)
14931496
callback = cbitem->cq_callback;
14941497
else if (channel->ch_part[part].ch_callback != NULL)
@@ -1610,16 +1613,13 @@ may_invoke_callback(channel_T *channel, int part)
16101613
int done = FALSE;
16111614

16121615
/* invoke the one-time callback with the matching nr */
1613-
while (cbitem != NULL)
1614-
{
1616+
for (cbitem = cbhead->cq_next; cbitem != NULL; cbitem = cbitem->cq_next)
16151617
if (cbitem->cq_seq_nr == seq_nr)
16161618
{
16171619
invoke_one_time_callback(channel, cbhead, cbitem, argv);
16181620
done = TRUE;
16191621
break;
16201622
}
1621-
cbitem = cbitem->cq_next;
1622-
}
16231623
if (!done)
16241624
ch_logn(channel, "Dropping message %d without callback", seq_nr);
16251625
}

src/testdir/test_channel.py

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -143,6 +143,11 @@ def handle(self):
143143
print("sending: {}".format(cmd))
144144
self.request.sendall(cmd.encode('utf-8'))
145145
response = ""
146+
elif decoded[1] == 'send zero':
147+
cmd = '[0,"zero index"]'
148+
print("sending: {}".format(cmd))
149+
self.request.sendall(cmd.encode('utf-8'))
150+
response = "sent zero"
146151
elif decoded[1] == 'close me':
147152
print("closing")
148153
self.request.close()

src/testdir/test_channel.vim

Lines changed: 66 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -294,6 +294,72 @@ endfunc
294294

295295
"""""""""
296296

297+
let s:ch_reply = ''
298+
func s:ChHandler(chan, msg)
299+
unlet s:ch_reply
300+
let s:ch_reply = a:msg
301+
endfunc
302+
303+
let s:zero_reply = ''
304+
func s:OneHandler(chan, msg)
305+
unlet s:zero_reply
306+
let s:zero_reply = a:msg
307+
endfunc
308+
309+
func s:channel_zero(port)
310+
let handle = ch_open('localhost:' . a:port, s:chopt)
311+
if ch_status(handle) == "fail"
312+
call assert_false(1, "Can't open channel")
313+
return
314+
endif
315+
316+
" Check that eval works.
317+
call assert_equal('got it', ch_evalexpr(handle, 'hello!'))
318+
319+
" Check that eval works if a zero id message is sent back.
320+
let s:ch_reply = ''
321+
call assert_equal('sent zero', ch_evalexpr(handle, 'send zero'))
322+
sleep 10m
323+
if s:has_handler
324+
call assert_equal('zero index', s:ch_reply)
325+
else
326+
call assert_equal('', s:ch_reply)
327+
endif
328+
329+
" Check that handler works if a zero id message is sent back.
330+
let s:ch_reply = ''
331+
let s:zero_reply = ''
332+
call ch_sendexpr(handle, 'send zero', {'callback': 's:OneHandler'})
333+
" Somehow the second message takes a bit of time.
334+
for i in range(50)
335+
if s:zero_reply == 'sent zero'
336+
break
337+
endif
338+
sleep 10m
339+
endfor
340+
if s:has_handler
341+
call assert_equal('zero index', s:ch_reply)
342+
else
343+
call assert_equal('', s:ch_reply)
344+
endif
345+
call assert_equal('sent zero', s:zero_reply)
346+
endfunc
347+
348+
func Test_zero_reply()
349+
call ch_log('Test_zero_reply()')
350+
" Run with channel handler
351+
let s:has_handler = 1
352+
let s:chopt.callback = 's:ChHandler'
353+
call s:run_server('s:channel_zero')
354+
unlet s:chopt.callback
355+
356+
" Run without channel handler
357+
let s:has_handler = 0
358+
call s:run_server('s:channel_zero')
359+
endfunc
360+
361+
"""""""""
362+
297363
let s:reply1 = ""
298364
func s:HandleRaw1(chan, msg)
299365
unlet s:reply1

src/version.c

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -743,6 +743,8 @@ static char *(features[]) =
743743

744744
static int included_patches[] =
745745
{ /* Add new patch number below this line */
746+
/**/
747+
1493,
746748
/**/
747749
1492,
748750
/**/

0 commit comments

Comments
 (0)