123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278 |
- 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('<Ignore>') -- 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<CR>')
- -- need to first process input
- poke_eventloop()
- feed('<Ignore>') -- 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('<Ignore>') -- 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 termopen(), exited, and deleted by pressing a key', function()
- command([[let id = termopen('echo')]])
- 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<CR>')
- -- need to first process input
- poke_eventloop()
- feed('<Ignore>') -- 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 termopen(), exited, and deleted by :bdelete', function()
- command([[let id = termopen('echo')]])
- 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('<Ignore>') -- 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 = termopen(']] .. sh .. [[')]])
- 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 termopen()', function()
- command([[autocmd TermOpen * call input('')]])
- async_meths.nvim_command('terminal foobar')
- screen:expect {
- grid = [[
- |
- {0:~ }|*2
- ^ |
- ]],
- }
- feed('<CR>')
- screen:expect {
- grid = [[
- ^ready $ foobar |
- |
- [Process exited 0] |
- |
- ]],
- }
- feed('i<CR>')
- screen:expect {
- grid = [[
- ^ |
- {0:~ }|*2
- |
- ]],
- }
- assert_alive()
- end)
- it('wipes buffer and processes events when using termopen()', function()
- command([[autocmd TermOpen * bwipe! | call input('')]])
- async_meths.nvim_command('terminal foobar')
- screen:expect {
- grid = [[
- |
- {0:~ }|*2
- ^ |
- ]],
- }
- feed('<CR>')
- 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('<CR>')
- 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)
|