local t = require('test.testutil') local n = require('test.functional.testnvim')() local Screen = require('test.functional.ui.screen') local clear = n.clear local eq = t.eq local eval = n.eval local command = n.command local pcall_err = t.pcall_err local feed = n.feed local poke_eventloop = n.poke_eventloop local is_os = t.is_os local api = n.api local async_meths = n.async_meths local testprg = n.testprg local assert_alive = n.assert_alive describe('terminal channel is closed and later released if', function() local screen before_each(function() clear() screen = Screen.new() end) it('opened by nvim_open_term() and deleted by :bdelete!', function() command([[let id = nvim_open_term(0, {})]]) local chans = eval('len(nvim_list_chans())') -- channel hasn't been released yet eq( "Vim(call):Can't send data to closed stream", pcall_err(command, [[bdelete! | call chansend(id, 'test')]]) ) feed('') -- add input to separate two RPC requests -- channel has been released after one main loop iteration eq(chans - 1, eval('len(nvim_list_chans())')) end) it('opened by nvim_open_term(), closed by chanclose(), and deleted by pressing a key', function() command('let id = nvim_open_term(0, {})') local chans = eval('len(nvim_list_chans())') -- channel has been closed but not released eq( "Vim(call):Can't send data to closed stream", pcall_err(command, [[call chanclose(id) | call chansend(id, 'test')]]) ) screen:expect({ any = '%[Terminal closed%]' }) eq(chans, eval('len(nvim_list_chans())')) -- delete terminal feed('i') -- need to first process input poke_eventloop() feed('') -- add input to separate two RPC requests -- channel has been released after another main loop iteration eq(chans - 1, eval('len(nvim_list_chans())')) end) it('opened by nvim_open_term(), closed by chanclose(), and deleted by :bdelete', function() command('let id = nvim_open_term(0, {})') local chans = eval('len(nvim_list_chans())') -- channel has been closed but not released eq( "Vim(call):Can't send data to closed stream", pcall_err(command, [[call chanclose(id) | call chansend(id, 'test')]]) ) screen:expect({ any = '%[Terminal closed%]' }) eq(chans, eval('len(nvim_list_chans())')) -- channel still hasn't been released yet eq( "Vim(call):Can't send data to closed stream", pcall_err(command, [[bdelete | call chansend(id, 'test')]]) ) feed('') -- add input to separate two RPC requests -- channel has been released after one main loop iteration eq(chans - 1, eval('len(nvim_list_chans())')) end) it('opened by jobstart(…,{term=true}), exited, and deleted by pressing a key', function() command([[let id = jobstart('echo',{'term':v:true})]]) local chans = eval('len(nvim_list_chans())') -- wait for process to exit screen:expect({ any = '%[Process exited 0%]' }) -- process has exited but channel has't been released eq( "Vim(call):Can't send data to closed stream", pcall_err(command, [[call chansend(id, 'test')]]) ) eq(chans, eval('len(nvim_list_chans())')) -- delete terminal feed('i') -- need to first process input poke_eventloop() feed('') -- add input to separate two RPC requests -- channel has been released after another main loop iteration eq(chans - 1, eval('len(nvim_list_chans())')) end) -- This indirectly covers #16264 it('opened by jobstart(…,{term=true}), exited, and deleted by :bdelete', function() command([[let id = jobstart('echo', {'term':v:true})]]) local chans = eval('len(nvim_list_chans())') -- wait for process to exit screen:expect({ any = '%[Process exited 0%]' }) -- process has exited but channel hasn't been released eq( "Vim(call):Can't send data to closed stream", pcall_err(command, [[call chansend(id, 'test')]]) ) eq(chans, eval('len(nvim_list_chans())')) -- channel still hasn't been released yet eq( "Vim(call):Can't send data to closed stream", pcall_err(command, [[bdelete | call chansend(id, 'test')]]) ) feed('') -- add input to separate two RPC requests -- channel has been released after one main loop iteration eq(chans - 1, eval('len(nvim_list_chans())')) end) end) it('chansend sends lines to terminal channel in proper order', function() clear({ args = { '--cmd', 'set laststatus=2' } }) local screen = Screen.new(100, 20) screen._default_attr_ids = nil local shells = is_os('win') and { 'cmd.exe', 'pwsh.exe -nop', 'powershell.exe -nop' } or { 'sh' } for _, sh in ipairs(shells) do command([[let id = jobstart(']] .. sh .. [[', {'term':v:true})]]) command([[call chansend(id, ['echo "hello"', 'echo "world"', ''])]]) screen:expect { any = [[echo "hello".*echo "world"]], } command('bdelete!') screen:expect { any = '%[No Name%]', } end end) describe('no crash when TermOpen autocommand', function() local screen before_each(function() clear() api.nvim_set_option_value('shell', testprg('shell-test'), {}) command('set shellcmdflag=EXE shellredir= shellpipe= shellquote= shellxquote=') screen = Screen.new(60, 4) screen:set_default_attr_ids({ [0] = { bold = true, foreground = Screen.colors.Blue }, }) end) it('processes job exit event when using jobstart(…,{term=true})', function() command([[autocmd TermOpen * call input('')]]) async_meths.nvim_command('terminal foobar') screen:expect { grid = [[ | {0:~ }|*2 ^ | ]], } feed('') screen:expect { grid = [[ ^ready $ foobar | | [Process exited 0] | | ]], } feed('i') screen:expect { grid = [[ ^ | {0:~ }|*2 | ]], } assert_alive() end) it('wipes buffer and processes events when using jobstart(…,{term=true})', function() command([[autocmd TermOpen * bwipe! | call input('')]]) async_meths.nvim_command('terminal foobar') screen:expect { grid = [[ | {0:~ }|*2 ^ | ]], } feed('') screen:expect { grid = [[ ^ | {0:~ }|*2 | ]], } assert_alive() end) it('wipes buffer and processes events when using nvim_open_term()', function() command([[autocmd TermOpen * bwipe! | call input('')]]) async_meths.nvim_open_term(0, {}) screen:expect { grid = [[ | {0:~ }|*2 ^ | ]], } feed('') screen:expect { grid = [[ ^ | {0:~ }|*2 | ]], } assert_alive() end) end) describe('nvim_open_term', function() local screen before_each(function() clear() screen = Screen.new(8, 10) end) it('with force_crlf=true converts newlines', function() local win = api.nvim_get_current_win() local buf = api.nvim_create_buf(false, true) local term = api.nvim_open_term(buf, { force_crlf = true }) api.nvim_win_set_buf(win, buf) api.nvim_chan_send(term, 'here\nthere\nfoo\r\nbar\n\ntest') screen:expect { grid = [[ ^here | there | foo | bar | | test | |*4 ]], } api.nvim_chan_send(term, '\nfirst') screen:expect { grid = [[ ^here | there | foo | bar | | test | first | |*3 ]], } end) it('with force_crlf=false does not convert newlines', function() local win = api.nvim_get_current_win() local buf = api.nvim_create_buf(false, true) local term = api.nvim_open_term(buf, { force_crlf = false }) api.nvim_win_set_buf(win, buf) api.nvim_chan_send(term, 'here\nthere') screen:expect { grid = [[ ^here | there | |*8 ]] } end) end)