mirror of
				https://github.com/neovim/neovim.git
				synced 2025-10-26 12:27:24 +00:00 
			
		
		
		
	 e40946a5be
			
		
	
	e40946a5be
	
	
	
		
			
			- Default to powershell. - Avoid hardcoded "-c". - Remove ^M character from received lines. - pending_win32(): clear() is unnecessary and it pollutes the tests. Closes #3973 Helped-by: Rui Abreu Ferreira <raf-ep@gmx.com>
		
			
				
	
	
		
			227 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			227 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
| -- Test server -> client RPC scenarios. Note: unlike `rpcnotify`, to evaluate
 | |
| -- `rpcrequest` calls we need the client event loop to be running.
 | |
| local helpers = require('test.functional.helpers')(after_each)
 | |
| local clear, nvim, eval = helpers.clear, helpers.nvim, helpers.eval
 | |
| local eq, neq, run, stop = helpers.eq, helpers.neq, helpers.run, helpers.stop
 | |
| local nvim_prog, command, funcs = helpers.nvim_prog, helpers.command, helpers.funcs
 | |
| local source, next_message = helpers.source, helpers.next_message
 | |
| local ok = helpers.ok
 | |
| local meths = helpers.meths
 | |
| 
 | |
| describe('server -> client', function()
 | |
|   local cid
 | |
| 
 | |
|   before_each(function()
 | |
|     clear()
 | |
|     cid = nvim('get_api_info')[1]
 | |
|   end)
 | |
| 
 | |
|   describe('simple call', function()
 | |
|     it('works', function()
 | |
|       local function on_setup()
 | |
|         eq({4, 5, 6}, eval('rpcrequest('..cid..', "scall", 1, 2, 3)'))
 | |
|         stop()
 | |
|       end
 | |
| 
 | |
|       local function on_request(method, args)
 | |
|         eq('scall', method)
 | |
|         eq({1, 2, 3}, args)
 | |
|         nvim('command', 'let g:result = [4, 5, 6]')
 | |
|         return eval('g:result')
 | |
|       end
 | |
|       run(on_request, nil, on_setup)
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
|   describe('empty string handling in arrays', function()
 | |
|     -- Because the msgpack encoding for an empty string was interpreted as an
 | |
|     -- error, msgpack arrays with an empty string looked like
 | |
|     -- [..., '', 0, ..., 0] after the conversion, regardless of the array
 | |
|     -- elements following the empty string.
 | |
|     it('works', function()
 | |
|       local function on_setup()
 | |
|         eq({1, 2, '', 3, 'asdf'}, eval('rpcrequest('..cid..', "nstring")'))
 | |
|         stop()
 | |
|       end
 | |
| 
 | |
|       local function on_request()
 | |
|         -- No need to evaluate the args, we are only interested in
 | |
|         -- a response that contains an array with an empty string.
 | |
|         return {1, 2, '', 3, 'asdf'}
 | |
|       end
 | |
|       run(on_request, nil, on_setup)
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
|   describe('recursive call', function()
 | |
|     it('works', function()
 | |
|       local function on_setup()
 | |
|         nvim('set_var', 'result1', 0)
 | |
|         nvim('set_var', 'result2', 0)
 | |
|         nvim('set_var', 'result3', 0)
 | |
|         nvim('set_var', 'result4', 0)
 | |
|         nvim('command', 'let g:result1 = rpcrequest('..cid..', "rcall", 2)')
 | |
|         eq(4, nvim('get_var', 'result1'))
 | |
|         eq(8, nvim('get_var', 'result2'))
 | |
|         eq(16, nvim('get_var', 'result3'))
 | |
|         eq(32, nvim('get_var', 'result4'))
 | |
|         stop()
 | |
|       end
 | |
| 
 | |
|       local function on_request(method, args)
 | |
|         eq('rcall', method)
 | |
|         local n = unpack(args) * 2
 | |
|         if n <= 16 then
 | |
|           local cmd
 | |
|           if n == 4 then
 | |
|             cmd = 'let g:result2 = rpcrequest('..cid..', "rcall", '..n..')'
 | |
|           elseif n == 8 then
 | |
|             cmd = 'let g:result3 = rpcrequest('..cid..', "rcall", '..n..')'
 | |
|           elseif n == 16 then
 | |
|             cmd = 'let g:result4 = rpcrequest('..cid..', "rcall", '..n..')'
 | |
|           end
 | |
|           nvim('command', cmd)
 | |
|         end
 | |
|         return n
 | |
|       end
 | |
|       run(on_request, nil, on_setup)
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
|   describe('requests and notifications interleaved', function()
 | |
|     -- This tests that the following scenario won't happen:
 | |
|     --
 | |
|     -- server->client [request     ] (1)
 | |
|     -- client->server [request     ] (2) triggered by (1)
 | |
|     -- server->client [notification] (3) triggered by (2)
 | |
|     -- server->client [response    ] (4) response to (2)
 | |
|     -- client->server [request     ] (4) triggered by (3)
 | |
|     -- server->client [request     ] (5) triggered by (4)
 | |
|     -- client->server [response    ] (6) response to (1)
 | |
|     --
 | |
|     -- If the above scenario ever happens, the client connection will be closed
 | |
|     -- because (6) is returned after request (5) is sent, and nvim
 | |
|     -- only deals with one server->client request at a time. (In other words,
 | |
|     -- the client cannot send a response to a request that is not at the top
 | |
|     -- of nvim's request stack).
 | |
|     --
 | |
|     -- But above scenario shoudn't happen by the way notifications are dealt in
 | |
|     -- Nvim: they are only sent after there are no pending server->client
 | |
|     -- request(the request stack fully unwinds). So (3) is only sent after the
 | |
|     -- client returns (6).
 | |
|     it('works', function()
 | |
|       local expected = 300
 | |
|       local notified = 0
 | |
|       local function on_setup()
 | |
|         eq('notified!', eval('rpcrequest('..cid..', "notify")'))
 | |
|       end
 | |
| 
 | |
|       local function on_request(method)
 | |
|         eq('notify', method)
 | |
|         eq(1, eval('rpcnotify('..cid..', "notification")'))
 | |
|         return 'notified!'
 | |
|       end
 | |
| 
 | |
|       local function on_notification(method)
 | |
|         eq('notification', method)
 | |
|         if notified == expected then
 | |
|           stop()
 | |
|           return
 | |
|         end
 | |
|         notified = notified + 1
 | |
|         eq('notified!', eval('rpcrequest('..cid..', "notify")'))
 | |
|       end
 | |
| 
 | |
|       run(on_request, on_notification, on_setup)
 | |
|       eq(expected, notified)
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
|   describe('when the client is a recursive vim instance', function()
 | |
|     if os.getenv("TRAVIS") and helpers.os_name() == "osx" then
 | |
|       -- XXX: Hangs Travis macOS since e9061117a5b8f195c3f26a5cb94e18ddd7752d86.
 | |
|       pending("[Hangs on Travis macOS. #5002]", function() end)
 | |
|       return
 | |
|     end
 | |
| 
 | |
|     before_each(function()
 | |
|       command("let vim = rpcstart('"..nvim_prog.."', ['-u', 'NONE', '-i', 'NONE', '--cmd', 'set noswapfile', '--embed'])")
 | |
|       neq(0, eval('vim'))
 | |
|     end)
 | |
| 
 | |
|     after_each(function() command('call rpcstop(vim)') end)
 | |
| 
 | |
|     it('can send/recieve notifications and make requests', function()
 | |
|       nvim('command', "call rpcnotify(vim, 'vim_set_current_line', 'SOME TEXT')")
 | |
| 
 | |
|       -- Wait for the notification to complete.
 | |
|       nvim('command', "call rpcrequest(vim, 'vim_eval', '0')")
 | |
| 
 | |
|       eq('SOME TEXT', eval("rpcrequest(vim, 'vim_get_current_line')"))
 | |
|     end)
 | |
| 
 | |
|     it('can communicate buffers, tabpages, and windows', function()
 | |
|       eq({1}, eval("rpcrequest(vim, 'nvim_list_tabpages')"))
 | |
|       -- Window IDs start at 1000 (LOWEST_WIN_ID in vim.h)
 | |
|       eq({1000}, eval("rpcrequest(vim, 'nvim_list_wins')"))
 | |
| 
 | |
|       local buf = eval("rpcrequest(vim, 'nvim_list_bufs')")[1]
 | |
|       eq(1, buf)
 | |
| 
 | |
|       eval("rpcnotify(vim, 'buffer_set_line', "..buf..", 0, 'SOME TEXT')")
 | |
|       nvim('command', "call rpcrequest(vim, 'vim_eval', '0')")  -- wait
 | |
| 
 | |
|       eq('SOME TEXT', eval("rpcrequest(vim, 'buffer_get_line', "..buf..", 0)"))
 | |
| 
 | |
|       -- Call get_lines(buf, range [0,0], strict_indexing)
 | |
|       eq({'SOME TEXT'}, eval("rpcrequest(vim, 'buffer_get_lines', "..buf..", 0, 1, 1)"))
 | |
|     end)
 | |
| 
 | |
|     it('returns an error if the request failed', function()
 | |
|       local status, err = pcall(eval, "rpcrequest(vim, 'does-not-exist')")
 | |
|       eq(false, status)
 | |
|       ok(nil ~= string.match(err, 'Failed to evaluate expression'))
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
|   describe('when using jobstart', function()
 | |
|     local jobid
 | |
|     before_each(function()
 | |
|       local channel = nvim('get_api_info')[1]
 | |
|       nvim('set_var', 'channel', channel)
 | |
|       source([[
 | |
|         function! s:OnEvent(id, data, event)
 | |
|           call rpcnotify(g:channel, a:event, 0, a:data)
 | |
|         endfunction
 | |
|         let g:job_opts = {
 | |
|         \ 'on_stderr': function('s:OnEvent'),
 | |
|         \ 'on_exit': function('s:OnEvent'),
 | |
|         \ 'user': 0,
 | |
|         \ 'rpc': v:true
 | |
|         \ }
 | |
|       ]])
 | |
|       local lua_prog = arg[-1]
 | |
|       meths.set_var("args", {lua_prog, 'test/functional/api/rpc_fixture.lua'})
 | |
|       jobid = eval("jobstart(g:args, g:job_opts)")
 | |
|       neq(0, 'jobid')
 | |
|     end)
 | |
| 
 | |
|     after_each(function()
 | |
|       funcs.jobstop(jobid)
 | |
|     end)
 | |
| 
 | |
|     if helpers.pending_win32(pending) then return end
 | |
| 
 | |
|     it('rpc and text stderr can be combined', function()
 | |
|       eq("ok",funcs.rpcrequest(jobid, "poll"))
 | |
|       funcs.rpcnotify(jobid, "ping")
 | |
|       eq({'notification', 'pong', {}}, next_message())
 | |
|       eq("done!",funcs.rpcrequest(jobid, "write_stderr", "fluff\n"))
 | |
|       eq({'notification', 'stderr', {0, {'fluff', ''}}}, next_message())
 | |
|       funcs.rpcrequest(jobid, "exit")
 | |
|       eq({'notification', 'exit', {0, 0}}, next_message())
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
| end)
 |