mirror of
https://github.com/neovim/neovim.git
synced 2025-01-01 17:23:36 -07:00
2ea7bfc627
Tokenize p_sh if used as default in ex_terminal(). Previously p_sh was used as the first arg in a list when calling termopen(), this would try to call an untokenized version of shell, meaning if you had an argument in 'shell': set shell=/bin/bash\ --login the command would fail. Helped-by: oni-link <knil.ino@gmail.com> Closes #3999
196 lines
6.6 KiB
Lua
196 lines
6.6 KiB
Lua
local helpers = require('test.functional.helpers')(after_each)
|
|
local Screen = require('test.functional.ui.screen')
|
|
local clear, wait, nvim = helpers.clear, helpers.wait, helpers.nvim
|
|
local nvim_dir, source, eq = helpers.nvim_dir, helpers.source, helpers.eq
|
|
local execute, eval = helpers.execute, helpers.eval
|
|
|
|
if helpers.pending_win32(pending) then return end
|
|
|
|
describe(':terminal', function()
|
|
local screen
|
|
|
|
before_each(function()
|
|
clear()
|
|
screen = Screen.new(50, 4)
|
|
screen:attach({rgb=false})
|
|
end)
|
|
|
|
it("does not interrupt Press-ENTER prompt #2748", function()
|
|
-- Ensure that :messages shows Press-ENTER.
|
|
source([[
|
|
echomsg "msg1"
|
|
echomsg "msg2"
|
|
echomsg "msg3"
|
|
]])
|
|
-- Invoke a command that emits frequent terminal activity.
|
|
execute([[terminal while true; do echo X; done]])
|
|
helpers.feed([[<C-\><C-N>]])
|
|
wait()
|
|
helpers.sleep(10) -- Let some terminal activity happen.
|
|
execute("messages")
|
|
screen:expect([[
|
|
msg1 |
|
|
msg2 |
|
|
msg3 |
|
|
Press ENTER or type command to continue^ |
|
|
]])
|
|
end)
|
|
|
|
it("in normal-mode :split does not move cursor", function()
|
|
execute([[terminal while true; do echo foo; sleep .1; done]])
|
|
helpers.feed([[<C-\><C-N>M]]) -- move cursor away from last line
|
|
wait()
|
|
eq(3, eval("line('$')")) -- window height
|
|
eq(2, eval("line('.')")) -- cursor is in the middle
|
|
execute('vsplit')
|
|
eq(2, eval("line('.')")) -- cursor stays where we put it
|
|
execute('split')
|
|
eq(2, eval("line('.')")) -- cursor stays where we put it
|
|
end)
|
|
|
|
end)
|
|
|
|
describe(':terminal (with fake shell)', function()
|
|
local screen
|
|
|
|
before_each(function()
|
|
clear()
|
|
screen = Screen.new(50, 4)
|
|
screen:attach({rgb=false})
|
|
-- shell-test.c is a fake shell that prints its arguments and exits.
|
|
nvim('set_option', 'shell', nvim_dir..'/shell-test')
|
|
nvim('set_option', 'shellcmdflag', 'EXE')
|
|
end)
|
|
|
|
-- Invokes `:terminal {cmd}` using a fake shell (shell-test.c) which prints
|
|
-- the {cmd} and exits immediately .
|
|
local function terminal_with_fake_shell(cmd)
|
|
execute("terminal "..(cmd and cmd or ""))
|
|
end
|
|
|
|
it('with no argument, acts like termopen()', function()
|
|
terminal_with_fake_shell()
|
|
wait()
|
|
screen:expect([[
|
|
ready $ |
|
|
[Process exited 0] |
|
|
|
|
|
-- TERMINAL -- |
|
|
]])
|
|
end)
|
|
|
|
it("with no argument, and 'shell' is set to empty string", function()
|
|
nvim('set_option', 'shell', '')
|
|
terminal_with_fake_shell()
|
|
wait()
|
|
screen:expect([[
|
|
^ |
|
|
~ |
|
|
~ |
|
|
E91: 'shell' option is empty |
|
|
]])
|
|
end)
|
|
|
|
it("with no argument, but 'shell' has arguments, acts like termopen()", function()
|
|
nvim('set_option', 'shell', nvim_dir..'/shell-test -t jeff')
|
|
terminal_with_fake_shell()
|
|
wait()
|
|
screen:expect([[
|
|
jeff $ |
|
|
[Process exited 0] |
|
|
|
|
|
-- TERMINAL -- |
|
|
]])
|
|
end)
|
|
|
|
it('executes a given command through the shell', function()
|
|
terminal_with_fake_shell('echo hi')
|
|
wait()
|
|
screen:expect([[
|
|
ready $ echo hi |
|
|
|
|
|
[Process exited 0] |
|
|
-- TERMINAL -- |
|
|
]])
|
|
end)
|
|
|
|
it("executes a given command through the shell, when 'shell' has arguments", function()
|
|
nvim('set_option', 'shell', nvim_dir..'/shell-test -t jeff')
|
|
terminal_with_fake_shell('echo hi')
|
|
wait()
|
|
screen:expect([[
|
|
jeff $ echo hi |
|
|
|
|
|
[Process exited 0] |
|
|
-- TERMINAL -- |
|
|
]])
|
|
end)
|
|
|
|
it('allows quotes and slashes', function()
|
|
terminal_with_fake_shell([[echo 'hello' \ "world"]])
|
|
wait()
|
|
screen:expect([[
|
|
ready $ echo 'hello' \ "world" |
|
|
|
|
|
[Process exited 0] |
|
|
-- TERMINAL -- |
|
|
]])
|
|
end)
|
|
|
|
it('ex_terminal() double-free #4554', function()
|
|
source([[
|
|
autocmd BufNew * set shell=foo
|
|
terminal]])
|
|
-- Verify that BufNew actually fired (else the test is invalid).
|
|
eq('foo', eval('&shell'))
|
|
end)
|
|
|
|
it('ignores writes if the backing stream closes', function()
|
|
terminal_with_fake_shell()
|
|
helpers.feed('iiXXXXXXX')
|
|
wait()
|
|
-- Race: Though the shell exited (and streams were closed by SIGCHLD
|
|
-- handler), :terminal cleanup is pending on the main-loop.
|
|
-- This write should be ignored (not crash, #5445).
|
|
helpers.feed('iiYYYYYYY')
|
|
eq(2, eval("1+1")) -- Still alive?
|
|
end)
|
|
|
|
it('works with findfile()', function()
|
|
execute('terminal')
|
|
eq('term://', string.match(eval('bufname("%")'), "^term://"))
|
|
eq('scripts/shadacat.py', eval('findfile("scripts/shadacat.py", ".")'))
|
|
end)
|
|
|
|
it('works with :find', function()
|
|
terminal_with_fake_shell()
|
|
wait()
|
|
screen:expect([[
|
|
ready $ |
|
|
[Process exited 0] |
|
|
|
|
|
-- TERMINAL -- |
|
|
]])
|
|
eq('term://', string.match(eval('bufname("%")'), "^term://"))
|
|
helpers.feed([[<C-\><C-N>]])
|
|
execute([[find */shadacat.py]])
|
|
eq('scripts/shadacat.py', eval('bufname("%")'))
|
|
end)
|
|
|
|
it('works with gf', function()
|
|
terminal_with_fake_shell([[echo "scripts/shadacat.py"]])
|
|
wait()
|
|
screen:expect([[
|
|
ready $ echo "scripts/shadacat.py" |
|
|
|
|
|
[Process exited 0] |
|
|
-- TERMINAL -- |
|
|
]])
|
|
helpers.feed([[<C-\><C-N>]])
|
|
eq('term://', string.match(eval('bufname("%")'), "^term://"))
|
|
helpers.feed([[ggf"lgf]])
|
|
eq('scripts/shadacat.py', eval('bufname("%")'))
|
|
end)
|
|
|
|
end)
|