Files
macvim-mirror/src/testdir/test_terminal3.vim
T
Christian Brabandt 54ffef7eb8 patch 9.2.0569: out-of-bounds access in libvterm CSI 8 t resize
Problem:  In the bundled libvterm the CSI 8 ; rows ; cols t sequence reaches
          on_resize() without validating its arguments.  Missing, zero or
          negative dimensions cause a negative-size memmove() in
          resize_buffer() and out-of-bounds accesses in set_lineinfo() and
          DECALN, all reachable from output rendered in a terminal window
          (Yukihiro Nakamura).
Solution: Reject missing, zero or negative dimensions before calling
          on_resize().  Also clamp a negative cell width in on_text() as
          hardening for the bundled libvterm.

Signed-off-by: Christian Brabandt <cb@256bit.org>
2026-05-31 14:27:16 +00:00

1321 lines
43 KiB
VimL

" Tests for the terminal window.
" This is split in two, because it can take a lot of time.
" See test_terminal.vim and test_terminal2.vim for further tests.
CheckFeature terminal
source util/screendump.vim
source util/mouse.vim
import './util/vim9.vim' as v9
let $PROMPT_COMMAND=''
func Test_terminal_altscreen()
" somehow doesn't work on MS-Windows
CheckUnix
let cmd = "cat Xtext\<CR>"
let buf = term_start(&shell, {})
call TermWait(buf)
call writefile(["\<Esc>[?1047h"], 'Xtext', 'D')
call term_sendkeys(buf, cmd)
call WaitForAssert({-> assert_equal(1, term_getaltscreen(buf))})
call writefile(["\<Esc>[?1047l"], 'Xtext')
call term_sendkeys(buf, cmd)
call WaitForAssert({-> assert_equal(0, term_getaltscreen(buf))})
call term_sendkeys(buf, "exit\r")
exe buf . "bwipe!"
endfunc
func Test_terminal_shell_option()
if has('unix')
" exec is a shell builtin command, should fail without a shell.
term exec ls runtest.vim
call WaitForAssert({-> assert_match('job failed', term_getline(bufnr(), 1))})
bwipe!
term ++shell exec ls runtest.vim
call WaitForAssert({-> assert_match('runtest.vim', term_getline(bufnr(), 1))})
bwipe!
elseif has('win32')
" dir is a shell builtin command, should fail without a shell.
" However, if dir.exe (which might be provided by Cygwin/MSYS2) exists in
" the %PATH%, "term dir" succeeds unintentionally. Use dir.com instead.
try
term dir.com /b runtest.vim
throw 'dir.com without a shell must fail, so you will never get here'
catch /CreateProcess failed/
" ignore:
" winpty simply fails with "CreateProcess failed".
" compty fails with "CreateProcess failed: {localized failure reason}".
endtry
bwipe!
" This should execute the dir builtin command even with ".com".
term ++shell dir.com /b runtest.vim
call WaitForAssert({-> assert_match('runtest.vim', term_getline(bufnr(), 1))})
bwipe!
else
throw 'Skipped: does not work on this platform'
endif
endfunc
func Test_terminal_invalid_arg()
call assert_fails('terminal ++xyz', 'E181:')
endfunc
" Check a terminal with different colors
func Terminal_color(group_name, highlight_cmds, highlight_opt, open_cmds)
CheckScreendump
CheckRunVimInTerminal
CheckUnix
let lines = [
\ 'call setline(1, range(20))',
\ 'func NotifyParent()',
\ ' call echoraw("' .. TermNotifyParentCmd(v:true) .. '")',
\ 'endfunc',
\ 'func OpenTerm()',
\ ' set noruler',
\ " call term_start('cat', #{vertical: 1, "
\ .. 'exit_cb: {->NotifyParent()}, '
\ .. a:highlight_opt .. "})",
\ ' call NotifyParent()',
\ ] + a:open_cmds + [
\ 'endfunc',
\ ] + a:highlight_cmds + [
\ 'call NotifyParent()',
\ ]
call writefile(lines, 'XtermStart', 'D')
let buf = RunVimInTerminal('-S XtermStart', #{rows: 15})
call WaitForChildNotification()
call term_sendkeys(buf, ":call OpenTerm()\<CR>")
call WaitForChildNotification()
call term_sendkeys(buf, "hello\<CR>")
call VerifyScreenDump(buf, 'Test_terminal_color_' .. a:group_name, {})
call term_sendkeys(buf, "\<C-D>")
call WaitForChildNotification()
call StopVimInTerminal(buf)
endfunc
func Test_terminal_color_Terminal()
call Terminal_color("Terminal", [
\ "highlight Terminal ctermfg=blue ctermbg=yellow",
\ ], "", [])
endfunc
func Test_terminal_color_group()
call Terminal_color("MyTermCol", [
\ "highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue",
\ ], "term_highlight: 'MyTermCol',", [])
endfunc
func Test_terminal_color_wincolor()
call Terminal_color("MyWinCol", [
\ "highlight MyWinCol ctermfg=red ctermbg=darkyellow",
\ ], "", [
\ 'set wincolor=MyWinCol',
\ ])
endfunc
func Test_terminal_color_group_over_Terminal()
call Terminal_color("MyTermCol_over_Terminal", [
\ "highlight Terminal ctermfg=blue ctermbg=yellow",
\ "highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue",
\ ], "term_highlight: 'MyTermCol',", [])
endfunc
func Test_terminal_color_wincolor_over_group()
call Terminal_color("MyWinCol_over_group", [
\ "highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue",
\ "highlight MyWinCol ctermfg=red ctermbg=darkyellow",
\ ], "term_highlight: 'MyTermCol',", [
\ 'set wincolor=MyWinCol',
\ ])
endfunc
func Test_terminal_color_wincolor_split()
CheckScreendump
CheckRunVimInTerminal
CheckUnix
let lines = [
\ 'call setline(1, range(20))',
\ 'func OpenTerm()',
\ ' set noruler',
\ " call term_start('cat', #{vertical: 1, term_highlight: 'MyTermCol'})",
\ 'endfunc',
\ 'highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue',
\ 'highlight MyWinCol ctermfg=red ctermbg=darkyellow',
\ 'highlight MyWinCol2 ctermfg=black ctermbg=blue',
\ ]
call writefile(lines, 'XtermStart', 'D')
let buf = RunVimInTerminal('-S XtermStart', #{rows: 15})
call TermWait(buf, 100)
call term_sendkeys(buf, ":call OpenTerm()\<CR>")
call TermWait(buf, 50)
call term_sendkeys(buf, "hello\<CR>")
call TermWait(buf, 50)
call term_sendkeys(buf, "\<C-W>:split\<CR>")
call term_sendkeys(buf, "\<C-W>:set wincolor=MyWinCol\<CR>")
call VerifyScreenDump(buf, 'Test_terminal_wincolor_split_MyWinCol', {})
call term_sendkeys(buf, "\<C-W>b:2sb\<CR>")
call term_sendkeys(buf, "\<C-W>:set wincolor=MyWinCol2\<CR>")
call VerifyScreenDump(buf, 'Test_terminal_wincolor_split_MyWinCol2', {})
call term_sendkeys(buf, "\<C-D>")
call TermWait(buf, 50)
call StopVimInTerminal(buf)
endfunc
func Test_terminal_color_transp_Terminal()
call Terminal_color("transp_Terminal", [
\ "highlight Terminal ctermfg=blue",
\ ], "", [])
endfunc
func Test_terminal_color_transp_group()
call Terminal_color("transp_MyTermCol", [
\ "highlight MyTermCol ctermfg=darkgreen",
\ ], "term_highlight: 'MyTermCol',", [])
endfunc
func Test_terminal_color_transp_wincolor()
call Terminal_color("transp_MyWinCol", [
\ "highlight MyWinCol ctermfg=red",
\ ], "", [
\ 'set wincolor=MyWinCol',
\ ])
endfunc
func Test_terminal_color_gui_Terminal()
CheckFeature termguicolors
call Terminal_color("gui_Terminal", [
\ "set termguicolors",
\ "highlight Terminal guifg=#3344ff guibg=#b0a700",
\ ], "", [])
endfunc
func Test_terminal_color_gui_group()
CheckFeature termguicolors
call Terminal_color("gui_MyTermCol", [
\ "set termguicolors",
\ "highlight MyTermCol guifg=#007800 guibg=#6789ff",
\ ], "term_highlight: 'MyTermCol',", [])
endfunc
func Test_terminal_color_gui_wincolor()
CheckFeature termguicolors
call Terminal_color("gui_MyWinCol", [
\ "set termguicolors",
\ "highlight MyWinCol guifg=#fe1122 guibg=#818100",
\ ], "", [
\ 'set wincolor=MyWinCol',
\ ])
endfunc
func Test_terminal_color_gui_transp_Terminal()
CheckFeature termguicolors
call Terminal_color("gui_transp_Terminal", [
\ "set termguicolors",
\ "highlight Terminal guifg=#3344ff",
\ ], "", [])
endfunc
func Test_terminal_color_gui_transp_group()
CheckFeature termguicolors
call Terminal_color("gui_transp_MyTermCol", [
\ "set termguicolors",
\ "highlight MyTermCol guifg=#007800",
\ ], "term_highlight: 'MyTermCol',", [])
endfunc
func Test_terminal_color_gui_transp_wincolor()
CheckFeature termguicolors
call Terminal_color("gui_transp_MyWinCol", [
\ "set termguicolors",
\ "highlight MyWinCol guifg=#fe1122",
\ ], "", [
\ 'set wincolor=MyWinCol',
\ ])
endfunc
func Test_terminal_in_popup()
CheckScreendump
CheckRunVimInTerminal
let text =<< trim END
some text
to edit
in a popup window
END
call writefile(text, 'Xtext', 'D')
let cmd = GetVimCommandCleanTerm()
let lines = [
\ 'call setline(1, range(20))',
\ 'hi PopTerm ctermbg=grey',
\ 'func OpenTerm(setColor)',
\ " set noruler",
\ " let s:buf = term_start('" .. cmd .. " Xtext', #{hidden: 1, term_finish: 'close'})",
\ ' let g:winid = popup_create(s:buf, #{minwidth: 45, minheight: 7, border: [], drag: 1, resize: 1})',
\ ' if a:setColor',
\ ' call win_execute(g:winid, "set wincolor=PopTerm")',
\ ' endif',
\ 'endfunc',
\ 'func HidePopup()',
\ ' call popup_hide(g:winid)',
\ 'endfunc',
\ 'func ClosePopup()',
\ ' call popup_close(g:winid)',
\ 'endfunc',
\ 'func ReopenPopup()',
\ ' call popup_create(s:buf, #{minwidth: 40, minheight: 6, border: []})',
\ 'endfunc',
\ ]
call writefile(lines, 'XtermPopup', 'D')
let buf = RunVimInTerminal('-S XtermPopup', #{rows: 15})
call TermWait(buf,0)
call term_sendkeys(buf, ":call OpenTerm(0)\<CR>")
call TermWait(buf,0)
call term_sendkeys(buf, ":\<CR>")
call TermWait(buf,0)
call term_sendkeys(buf, "\<C-W>:echo getwinvar(g:winid, \"&buftype\") win_gettype(g:winid)\<CR>")
call VerifyScreenDump(buf, 'Test_terminal_popup_1', {})
call term_sendkeys(buf, ":q\<CR>")
call VerifyScreenDump(buf, 'Test_terminal_popup_2', {})
call term_sendkeys(buf, ":call OpenTerm(1)\<CR>")
call TermWait(buf,0)
call term_sendkeys(buf, ":set hlsearch\<CR>")
call TermWait(buf,0)
call term_sendkeys(buf, "/edit\<CR>")
call VerifyScreenDump(buf, 'Test_terminal_popup_3', {})
call term_sendkeys(buf, "\<C-W>:call HidePopup()\<CR>")
call VerifyScreenDump(buf, 'Test_terminal_popup_4', {})
call term_sendkeys(buf, "\<CR>")
call TermWait(buf,0)
call term_sendkeys(buf, "\<C-W>:call ClosePopup()\<CR>")
call VerifyScreenDump(buf, 'Test_terminal_popup_5', {})
call term_sendkeys(buf, "\<C-W>:call ReopenPopup()\<CR>")
call VerifyScreenDump(buf, 'Test_terminal_popup_6', {})
" Go to terminal-Normal mode and visually select text.
call term_sendkeys(buf, "\<C-W>Ngg/in\<CR>vww")
call VerifyScreenDump(buf, 'Test_terminal_popup_7', {})
" Back to job mode, redraws
call term_sendkeys(buf, "A")
call VerifyScreenDump(buf, 'Test_terminal_popup_8', {})
call TermWait(buf,0)
call term_sendkeys(buf, ":q\<CR>")
call WaitForAssert({-> assert_equal(0, match(term_getline(buf, 6), '^5\s*$'))}, 250) " wait for terminal to vanish
call StopVimInTerminal(buf)
endfunc
" Check a terminal in popup window uses the default minimum size.
func Test_terminal_in_popup_min_size()
CheckScreendump
CheckRunVimInTerminal
let text =<< trim END
another text
to show
in a popup window
END
call writefile(text, 'Xtext', 'D')
let lines = [
\ 'call setline(1, range(20))',
\ 'func OpenTerm()',
\ " let s:buf = term_start('cat Xtext', #{hidden: 1})",
\ ' let g:winid = popup_create(s:buf, #{ border: []})',
\ 'endfunc',
\ ]
call writefile(lines, 'XtermPopup', 'D')
let buf = RunVimInTerminal('-S XtermPopup', #{rows: 15})
call TermWait(buf, 100)
call term_sendkeys(buf, ":set noruler\<CR>")
call term_sendkeys(buf, ":call OpenTerm()\<CR>")
call TermWait(buf, 50)
call term_sendkeys(buf, ":\<CR>")
call VerifyScreenDump(buf, 'Test_terminal_popup_m1', {})
call TermWait(buf, 50)
call term_sendkeys(buf, ":q\<CR>")
call TermWait(buf, 50) " wait for terminal to vanish
call StopVimInTerminal(buf)
endfunc
" Check a terminal in popup window with different colors
func Terminal_in_popup_color(group_name, highlight_cmds, highlight_opt, popup_cmds, popup_opt)
CheckScreendump
CheckRunVimInTerminal
CheckUnix
let lines = [
\ 'call setline(1, range(20))',
\ 'func NotifyParent(...)',
\ ' call echoraw("' .. TermNotifyParentCmd(v:true) .. '")',
\ 'endfunc',
\ 'func OpenTerm()',
\ " let s:buf = term_start('cat', #{hidden: 1, term_finish: 'close', "
\ .. a:highlight_opt .. "})",
\ ' let g:winid = popup_create(s:buf, #{border: [], '
\ .. 'callback: {->NotifyParent()}, '
\ .. a:popup_opt .. '})',
\ ] + a:popup_cmds + [
\ ' call NotifyParent()',
\ 'endfunc',
\ ] + a:highlight_cmds + [
\ 'call NotifyParent()',
\ ]
call writefile(lines, 'XtermPopup', 'D')
let buf = RunVimInTerminal('-S XtermPopup', #{rows: 15})
call WaitForChildNotification()
call term_sendkeys(buf, ":set noruler\<CR>")
call term_sendkeys(buf, ":call OpenTerm()\<CR>")
call WaitForChildNotification()
call term_sendkeys(buf, "hello\<CR>")
call VerifyScreenDump(buf, 'Test_terminal_popup_' .. a:group_name, {})
call term_sendkeys(buf, "\<C-D>")
call WaitForChildNotification()
call StopVimInTerminal(buf)
endfunc
func Test_terminal_in_popup_color_Terminal()
call Terminal_in_popup_color("Terminal", [
\ "highlight Terminal ctermfg=blue ctermbg=yellow",
\ ], "", [], "")
endfunc
func Test_terminal_in_popup_color_group()
call Terminal_in_popup_color("MyTermCol", [
\ "highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue",
\ ], "term_highlight: 'MyTermCol',", [], "")
endfunc
func Test_terminal_in_popup_color_wincolor()
call Terminal_in_popup_color("MyWinCol", [
\ "highlight MyWinCol ctermfg=red ctermbg=darkyellow",
\ ], "", [
\ 'call setwinvar(g:winid, "&wincolor", "MyWinCol")',
\ ], "")
endfunc
func Test_terminal_in_popup_color_popup_highlight()
call Terminal_in_popup_color("MyPopupHlCol", [
\ "highlight MyPopupHlCol ctermfg=cyan ctermbg=green",
\ ], "", [], "highlight: 'MyPopupHlCol'")
endfunc
func Test_terminal_in_popup_color_group_over_Terminal()
call Terminal_in_popup_color("MyTermCol_over_Terminal", [
\ "highlight Terminal ctermfg=blue ctermbg=yellow",
\ "highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue",
\ ], "term_highlight: 'MyTermCol',", [], "")
endfunc
func Test_terminal_in_popup_color_wincolor_over_group()
call Terminal_in_popup_color("MyWinCol_over_group", [
\ "highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue",
\ "highlight MyWinCol ctermfg=red ctermbg=darkyellow",
\ ], "term_highlight: 'MyTermCol',", [
\ 'call setwinvar(g:winid, "&wincolor", "MyWinCol")',
\ ], "")
endfunc
func Test_terminal_in_popup_color_transp_Terminal()
call Terminal_in_popup_color("transp_Terminal", [
\ "highlight Terminal ctermfg=blue",
\ ], "", [], "")
endfunc
func Test_terminal_in_popup_color_transp_group()
call Terminal_in_popup_color("transp_MyTermCol", [
\ "highlight MyTermCol ctermfg=darkgreen",
\ ], "term_highlight: 'MyTermCol',", [], "")
endfunc
func Test_terminal_in_popup_color_transp_wincolor()
call Terminal_in_popup_color("transp_MyWinCol", [
\ "highlight MyWinCol ctermfg=red",
\ ], "", [
\ 'call setwinvar(g:winid, "&wincolor", "MyWinCol")',
\ ], "")
endfunc
func Test_terminal_in_popup_color_transp_popup_highlight()
call Terminal_in_popup_color("transp_MyPopupHlCol", [
\ "highlight MyPopupHlCol ctermfg=cyan",
\ ], "", [], "highlight: 'MyPopupHlCol'")
endfunc
func Test_terminal_in_popup_color_gui_Terminal()
CheckFeature termguicolors
call Terminal_in_popup_color("gui_Terminal", [
\ "set termguicolors",
\ "highlight Terminal guifg=#3344ff guibg=#b0a700",
\ ], "", [], "")
endfunc
func Test_terminal_in_popup_color_gui_group()
CheckFeature termguicolors
call Terminal_in_popup_color("gui_MyTermCol", [
\ "set termguicolors",
\ "highlight MyTermCol guifg=#007800 guibg=#6789ff",
\ ], "term_highlight: 'MyTermCol',", [], "")
endfunc
func Test_terminal_in_popup_color_gui_wincolor()
CheckFeature termguicolors
call Terminal_in_popup_color("gui_MyWinCol", [
\ "set termguicolors",
\ "highlight MyWinCol guifg=#fe1122 guibg=#818100",
\ ], "", [
\ 'call setwinvar(g:winid, "&wincolor", "MyWinCol")',
\ ], "")
endfunc
func Test_terminal_in_popup_color_gui_popup_highlight()
CheckFeature termguicolors
call Terminal_in_popup_color("gui_MyPopupHlCol", [
\ "set termguicolors",
\ "highlight MyPopupHlCol guifg=#00e8f0 guibg=#126521",
\ ], "", [], "highlight: 'MyPopupHlCol'")
endfunc
func Test_terminal_in_popup_color_gui_transp_Terminal()
CheckFeature termguicolors
call Terminal_in_popup_color("gui_transp_Terminal", [
\ "set termguicolors",
\ "highlight Terminal guifg=#3344ff",
\ ], "", [], "")
endfunc
func Test_terminal_in_popup_color_gui_transp_group()
CheckFeature termguicolors
call Terminal_in_popup_color("gui_transp_MyTermCol", [
\ "set termguicolors",
\ "highlight MyTermCol guifg=#007800",
\ ], "term_highlight: 'MyTermCol',", [], "")
endfunc
func Test_terminal_in_popup_color_gui_transp_wincolor()
CheckFeature termguicolors
call Terminal_in_popup_color("gui_transp_MyWinCol", [
\ "set termguicolors",
\ "highlight MyWinCol guifg=#fe1122",
\ ], "", [
\ 'call setwinvar(g:winid, "&wincolor", "MyWinCol")',
\ ], "")
endfunc
func Test_terminal_in_popup_color_gui_transp_popup_highlight()
CheckFeature termguicolors
call Terminal_in_popup_color("gui_transp_MyPopupHlCol", [
\ "set termguicolors",
\ "highlight MyPopupHlCol guifg=#00e8f0",
\ ], "", [], "highlight: 'MyPopupHlCol'")
endfunc
func Test_double_popup_terminal()
let buf1 = term_start(&shell, #{hidden: 1})
let win1 = popup_create(buf1, {})
let buf2 = term_start(&shell, #{hidden: 1})
call assert_fails('call popup_create(buf2, {})', 'E861:')
call popup_close(win1)
exe buf1 .. 'bwipe!'
exe buf2 .. 'bwipe!'
endfunc
func Test_escape_popup_terminal()
set hidden
" Cannot escape a terminal popup window using win_gotoid
let prev_win = win_getid()
eval term_start('sh', #{hidden: 1, term_finish: 'close'})->popup_create({})
call assert_fails("call win_gotoid(" .. prev_win .. ")", 'E863:')
call popup_clear(1)
set hidden&
endfunc
func Test_issue_5607()
let wincount = winnr('$')
exe 'terminal' &shell &shellcmdflag 'exit'
let job = term_getjob(bufnr())
call WaitForAssert({-> assert_equal("dead", job_status(job))})
let old_wincolor = &wincolor
try
set wincolor=
finally
let &wincolor = old_wincolor
bw!
endtry
endfunc
func Test_hidden_terminal()
let buf = term_start(&shell, #{hidden: 1})
call assert_equal('', bufname('^$'))
call StopShellInTerminal(buf)
endfunc
func Test_term_nasty_callback()
CheckExecutable sh
set hidden
let g:buf0 = term_start('sh', #{hidden: 1, term_finish: 'close'})
call popup_create(g:buf0, {})
call assert_fails("call term_start(['sh', '-c'], #{curwin: 1})", 'E863:')
call popup_clear(1)
set hidden&
endfunc
func Test_term_and_startinsert()
CheckRunVimInTerminal
CheckUnix
let lines =<< trim EOL
put='some text'
term
startinsert
EOL
call writefile(lines, 'XTest_startinsert', 'D')
let buf = RunVimInTerminal('-S XTest_startinsert', {})
call term_sendkeys(buf, "exit\r")
call WaitForAssert({-> assert_equal("some text", term_getline(buf, 1))})
call term_sendkeys(buf, "0l")
call term_sendkeys(buf, "A<\<Esc>")
call WaitForAssert({-> assert_equal("some text<", term_getline(buf, 1))})
call StopVimInTerminal(buf)
endfunc
" Test for passing invalid arguments to terminal functions
func Test_term_func_invalid_arg()
call assert_fails('let b = term_getaltscreen([])', 'E745:')
call assert_fails('let a = term_getattr(1, [])', 'E730:')
call assert_fails('let c = term_getcursor([])', 'E745:')
call assert_fails('let l = term_getline([], 1)', 'E745:')
call assert_fails('let l = term_getscrolled([])', 'E745:')
call assert_fails('let s = term_getsize([])', 'E745:')
call assert_fails('let s = term_getstatus([])', 'E745:')
call assert_fails('let s = term_scrape([], 1)', 'E745:')
call assert_fails('call term_sendkeys([], "a")', 'E745:')
call assert_fails('call term_setapi([], "")', 'E745:')
call assert_fails('call term_setrestore([], "")', 'E745:')
call assert_fails('call term_setkill([], "")', 'E745:')
if has('gui') || has('termguicolors')
call assert_fails('let p = term_getansicolors([])', 'E745:')
call assert_fails('call term_setansicolors([], [])', 'E745:')
endif
let buf = term_start('echo')
call assert_fails('call term_setapi(' .. buf .. ', {})', 'E731:')
call assert_fails('call term_setkill(' .. buf .. ', {})', 'E731:')
call assert_fails('call term_setrestore(' .. buf .. ', {})', 'E731:')
exe buf . "bwipe!"
endfunc
" Test for sending various special keycodes to a terminal
func Test_term_keycode_translation()
CheckRunVimInTerminal
let buf = RunVimInTerminal('', {})
call term_sendkeys(buf, ":set nocompatible\<CR>")
call term_sendkeys(buf, ":set timeoutlen=20\<CR>")
let keys = ["\<F1>", "\<F2>", "\<F3>", "\<F4>", "\<F5>", "\<F6>", "\<F7>",
\ "\<F8>", "\<F9>", "\<F10>", "\<F11>", "\<F12>", "\<Home>",
\ "\<S-Home>", "\<C-Home>", "\<End>", "\<S-End>", "\<C-End>",
\ "\<Ins>", "\<Del>", "\<Left>", "\<S-Left>", "\<C-Left>", "\<Right>",
\ "\<S-Right>", "\<C-Right>", "\<Up>", "\<S-Up>", "\<Down>",
\ "\<S-Down>"]
let output = ['<F1>', '<F2>', '<F3>', '<F4>', '<F5>', '<F6>', '<F7>',
\ '<F8>', '<F9>', '<F10>', '<F11>', '<F12>', '<Home>', '<S-Home>',
\ '<C-Home>', '<End>', '<S-End>', '<C-End>', '<Insert>', '<Del>',
\ '<Left>', '<S-Left>', '<C-Left>', '<Right>', '<S-Right>',
\ '<C-Right>', '<Up>', '<S-Up>', '<Down>', '<S-Down>']
call term_sendkeys(buf, "i")
for i in range(len(keys))
call term_sendkeys(buf, "\<C-U>\<C-K>" .. keys[i])
call WaitForAssert({-> assert_equal(output[i], term_getline(buf, 1))}, 200)
endfor
let keypad_keys = ["\<k0>", "\<k1>", "\<k2>", "\<k3>", "\<k4>", "\<k5>",
\ "\<k6>", "\<k7>", "\<k8>", "\<k9>", "\<kPoint>", "\<kPlus>",
\ "\<kMinus>", "\<kMultiply>", "\<kDivide>"]
let keypad_output = ['0', '1', '2', '3', '4', '5',
\ '6', '7', '8', '9', '.', '+',
\ '-', '*', '/']
for i in range(len(keypad_keys))
" TODO: Mysteriously keypad 3 and 9 do not work on some systems.
if keypad_output[i] == '3' || keypad_output[i] == '9'
continue
endif
call term_sendkeys(buf, "\<C-U>" .. keypad_keys[i])
call WaitForAssert({-> assert_equal(keypad_output[i], term_getline(buf, 1))}, 100)
endfor
call feedkeys("\<C-U>\<kEnter>\<BS>one\<C-W>.two", 'xt')
call WaitForAssert({-> assert_equal('two', term_getline(buf, 1))})
call StopVimInTerminal(buf)
endfunc
" Test for using the mouse in a terminal
func Test_term_mouse()
CheckNotGui
CheckRunVimInTerminal
let save_mouse = &mouse
let save_term = &term
let save_ttymouse = &ttymouse
let save_clipboard = &clipboard
set mouse=a term=xterm ttymouse=sgr mousetime=200 clipboard=
let lines =<< trim END
one two three four five
red green yellow red blue
vim emacs sublime nano
END
call writefile(lines, 'Xtest_mouse', 'D')
" Create a terminal window running Vim for the test with mouse enabled
let prev_win = win_getid()
let buf = RunVimInTerminal('Xtest_mouse -n', {})
call term_sendkeys(buf, ":set nocompatible\<CR>")
call term_sendkeys(buf, ":set mouse=a term=xterm ttymouse=sgr\<CR>")
call term_sendkeys(buf, ":set clipboard=\<CR>")
call term_sendkeys(buf, ":set mousemodel=extend\<CR>")
call TermWait(buf)
redraw!
" Funcref used in WaitFor() to check that the "Xbuf" file is readable and
" has some contents. This avoids a "List index out of range" error when the
" file hasn't been written yet.
let XbufNotEmpty = {-> filereadable('Xbuf') && len(readfile('Xbuf')) > 0}
" Use the mouse to enter the terminal window
call win_gotoid(prev_win)
call feedkeys(MouseLeftClickCode(1, 1), 'x')
call feedkeys(MouseLeftReleaseCode(1, 1), 'x')
call assert_equal(1, getwininfo(win_getid())[0].terminal)
" Test for <LeftMouse> click/release
call test_setmouse(2, 5)
call feedkeys("\<LeftMouse>\<LeftRelease>", 'xt')
call test_setmouse(3, 8)
call term_sendkeys(buf, "\<LeftMouse>\<LeftRelease>")
call TermWait(buf, 50)
call delete('Xbuf')
call term_sendkeys(buf, ":call writefile([json_encode(getpos('.'))], 'Xbuf')\<CR>")
call TermWait(buf, 50)
call WaitFor(XbufNotEmpty)
let pos = json_decode(readfile('Xbuf')[0])
call assert_equal([3, 8], pos[1:2])
call delete('Xbuf')
" Test for selecting text using mouse
call test_setmouse(2, 11)
call term_sendkeys(buf, "\<LeftMouse>")
call test_setmouse(2, 16)
call term_sendkeys(buf, "\<LeftRelease>y")
call TermWait(buf, 50)
call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\<CR>")
call WaitFor(XbufNotEmpty)
call WaitForAssert({-> assert_equal('yellow', readfile('Xbuf')[0])})
call delete('Xbuf')
" Test for selecting text using double click
call test_setmouse(1, 11)
call term_sendkeys(buf, "\<LeftMouse>\<LeftRelease>\<LeftMouse>")
call test_setmouse(1, 17)
call term_sendkeys(buf, "\<LeftRelease>y")
call TermWait(buf, 50)
call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\<CR>")
call WaitFor(XbufNotEmpty)
call assert_equal('three four', readfile('Xbuf')[0])
call delete('Xbuf')
" Test for selecting a line using triple click
call test_setmouse(3, 2)
call term_sendkeys(buf, "\<LeftMouse>\<LeftRelease>\<LeftMouse>\<LeftRelease>\<LeftMouse>\<LeftRelease>y")
call TermWait(buf, 50)
call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\<CR>")
call WaitFor(XbufNotEmpty)
call assert_equal("vim emacs sublime nano\n", readfile('Xbuf')[0])
call delete('Xbuf')
" Test for selecting a block using quadruple click
call test_setmouse(1, 11)
call term_sendkeys(buf, "\<LeftMouse>\<LeftRelease>\<LeftMouse>\<LeftRelease>\<LeftMouse>\<LeftRelease>\<LeftMouse>")
call test_setmouse(3, 13)
call term_sendkeys(buf, "\<LeftRelease>y")
call TermWait(buf, 50)
call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\<CR>")
call WaitFor(XbufNotEmpty)
call assert_equal("ree\nyel\nsub", readfile('Xbuf')[0])
call delete('Xbuf')
" Test for extending a selection using right click
call test_setmouse(2, 9)
call term_sendkeys(buf, "\<LeftMouse>\<LeftRelease>")
call test_setmouse(2, 16)
call term_sendkeys(buf, "\<RightMouse>\<RightRelease>y")
call TermWait(buf, 50)
call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\<CR>")
call WaitFor(XbufNotEmpty)
call assert_equal("n yellow", readfile('Xbuf')[0])
call delete('Xbuf')
" Test for pasting text using middle click
call term_sendkeys(buf, ":let @r='bright '\<CR>")
call test_setmouse(2, 22)
call term_sendkeys(buf, "\"r\<MiddleMouse>\<MiddleRelease>")
call TermWait(buf, 50)
call term_sendkeys(buf, ":call writefile([getline(2)], 'Xbuf')\<CR>")
call WaitFor(XbufNotEmpty)
call assert_equal("red bright blue", readfile('Xbuf')[0][-15:])
call delete('Xbuf')
" cleanup
call TermWait(buf)
call StopVimInTerminal(buf)
let &mouse = save_mouse
let &term = save_term
let &ttymouse = save_ttymouse
let &clipboard = save_clipboard
set mousetime&
call delete('Xbuf')
endfunc
" Test for sync buffer cwd with shell's pwd
func Test_terminal_sync_shell_dir()
CheckUnix
" The test always use sh (see src/testdir/util/unix.vim).
" BSD's sh doesn't seem to play well with the OSC 7 escape sequence.
CheckNotBSD
set asd
" , is
" 1. a valid character for directory names
" 2. a reserved character in url-encoding
let chars = ",a"
" "," is url-encoded as '%2C'
let chars_url = "%2Ca"
let tmpfolder = fnamemodify(tempname(),':h') .. '/' .. chars
let tmpfolder_url = fnamemodify(tempname(),':h') .. '/' .. chars_url
call mkdir(tmpfolder, "p")
let buf = Run_shell_in_terminal({})
call term_sendkeys(buf, "echo $'\\e\]7;file://" .. tmpfolder_url .. "\\a'\<CR>")
"call term_sendkeys(buf, "cd " .. tmpfolder .. "\<CR>")
call TermWait(buf)
if has("mac")
let expected = "/private" .. tmpfolder
else
let expected = tmpfolder
endif
call assert_equal(expected, getcwd(winnr()))
set noasd
endfunc
" Test for modeless selection in a terminal
func Test_term_modeless_selection()
CheckUnix
CheckNotGui
CheckRunVimInTerminal
CheckFeature clipboard_working
let save_mouse = &mouse
let save_term = &term
let save_ttymouse = &ttymouse
set mouse=a term=xterm ttymouse=sgr mousetime=200
set clipboard=autoselectml
let lines =<< trim END
one two three four five
red green yellow red blue
vim emacs sublime nano
END
call writefile(lines, 'Xtest_modeless', 'D')
" Create a terminal window running Vim for the test with mouse disabled
let prev_win = win_getid()
let buf = RunVimInTerminal('Xtest_modeless -n', {})
call term_sendkeys(buf, ":set nocompatible\<CR>")
call term_sendkeys(buf, ":set mouse=\<CR>")
call TermWait(buf)
redraw!
" Use the mouse to enter the terminal window
call win_gotoid(prev_win)
call feedkeys(MouseLeftClickCode(1, 1), 'x')
call feedkeys(MouseLeftReleaseCode(1, 1), 'x')
call TermWait(buf)
call assert_equal(1, getwininfo(win_getid())[0].terminal)
" Test for copying a modeless selection to clipboard
let @* = 'clean'
" communicating with X server may take a little time
sleep 100m
call feedkeys(MouseLeftClickCode(2, 3), 'x')
call feedkeys(MouseLeftDragCode(2, 11), 'x')
call feedkeys(MouseLeftReleaseCode(2, 11), 'x')
call assert_equal("d green y", @*)
" cleanup
call TermWait(buf)
call StopVimInTerminal(buf)
let &mouse = save_mouse
let &term = save_term
let &ttymouse = save_ttymouse
set mousetime& clipboard&
new | only!
endfunc
func Test_terminal_getwinpos()
CheckRunVimInTerminal
" split, go to the bottom-right window
split
wincmd j
set splitright
let buf = RunVimInTerminal('', {'cols': 60})
call TermWait(buf, 100)
call term_sendkeys(buf, ":echo getwinpos(500)\<CR>")
" Find the output of getwinpos() in the bottom line.
let rows = term_getsize(buf)[0]
call WaitForAssert({-> assert_match('\[\d\+, \d\+\]', term_getline(buf, rows))})
let line = term_getline(buf, rows)
let xpos = str2nr(substitute(line, '\[\(\d\+\), \d\+\]', '\1', ''))
let ypos = str2nr(substitute(line, '\[\d\+, \(\d\+\)\]', '\1', ''))
" Position must be bigger than the getwinpos() result of Vim itself.
" The calculation in the console assumes a 10 x 7 character cell.
" In the GUI it can be more, let's assume a 20 x 14 cell.
" And then add 100 / 200 tolerance.
let [xroot, yroot] = getwinpos()
let winpos = 50->getwinpos()
call assert_equal(xroot, winpos[0])
call assert_equal(yroot, winpos[1])
let [winrow, wincol] = win_screenpos(0)
let xoff = wincol * (has('gui_running') ? 14 : 7) + 100
let yoff = winrow * (has('gui_running') ? 20 : 10) + 200
call assert_inrange(xroot + 2, xroot + xoff, xpos)
call assert_inrange(yroot + 2, yroot + yoff, ypos)
call TermWait(buf)
call term_sendkeys(buf, ":q\<CR>")
call StopVimInTerminal(buf)
set splitright&
only!
endfunc
func Test_terminal_term_start_error()
func s:term_start_error() abort
try
return term_start([[]])
catch
return v:exception
finally
"
endtry
endfunc
autocmd WinEnter * call type(0)
" Must not crash in s:term_start_error, nor the exception thrown.
let result = s:term_start_error()
call assert_match('^Vim(return):E730:', result)
autocmd! WinEnter
delfunc s:term_start_error
endfunc
func Test_terminal_vt420()
CheckRunVimInTerminal
" For Termcap
CheckUnix
CheckExecutable infocmp
let a = system('infocmp vt420')
if v:shell_error
" reset v:shell_error
let a = system('true')
throw 'Skipped: vt420 terminfo not available'
endif
let rows = 15
call writefile([':set term=vt420'], 'Xterm420', 'D')
let buf = RunVimInTerminal('-S Xterm420', #{rows: rows})
call TermWait(buf, 100)
call term_sendkeys(buf, ":set t_xo?\<CR>")
call WaitForAssert({-> assert_match('t_xo=y', term_getline(buf, rows))})
call StopVimInTerminal(buf)
call writefile([''], 'Xterm420')
let buf = RunVimInTerminal('-S Xterm420', #{rows: rows})
call TermWait(buf, 100)
call term_sendkeys(buf, ":set t_xo?\<CR>")
call WaitForAssert({-> assert_match('t_xo=\s\+', term_getline(buf, rows))})
call StopVimInTerminal(buf)
endfunc
" Test for using 'vertical' with term_start(). If a following term_start(),
" doesn't have the 'vertical' attribute, then it should be split horizontally.
func Test_terminal_vertical()
let lines =<< trim END
call term_start("NONE", {'vertical': 1})
call term_start("NONE")
VAR layout = winlayout()
call assert_equal('row', layout[0], string(layout))
call assert_equal('col', layout[1][0][0], string(layout))
:%bw!
END
call v9.CheckLegacyAndVim9Success(lines)
endfunc
" Needs to come before Test_hidden_terminal(), why?
func Test_autocmd_buffilepost_with_hidden_term()
CheckExecutable true
new XTestFile
defer delete('XTestFile')
call setline(1, ['one', 'two', 'three'])
call cursor(3, 10)
augroup TestCursor
au!
autocmd BufFilePost * call setbufvar(3, '&tabstop', 4)
augroup END
let buf = term_start(['true'], #{hidden: 1, term_finish: 'close'})
call term_wait(buf)
redraw!
call assert_equal('XTestFile', bufname('%'))
call assert_equal([0, 3, 5, 0], getpos('.'))
augroup TestCursor
au!
augroup END
augroup! TestCursor
bw! XTestFile
endfunc
func Test_terminal_visual_empty_listchars()
CheckScreendump
CheckRunVimInTerminal
CheckUnix
let lines = [
\ 'set listchars=',
\ ':term sh -c "printf ''hello\\n\\nhello''"'
\ ]
call writefile(lines, 'XtermStart1', 'D')
let buf = RunVimInTerminal('-S XtermStart1', #{rows: 15})
call term_wait(buf)
call term_sendkeys(buf, "V2k")
call VerifyScreenDump(buf, 'Test_terminal_empty_listchars', {})
call term_sendkeys(buf, "\<esc>")
call term_sendkeys(buf, ":set nu\<cr>")
call term_sendkeys(buf, "ggV2j")
call VerifyScreenDump(buf, 'Test_terminal_empty_listchars2', {})
call StopVimInTerminal(buf)
endfunc
func Test_terminal_ansi_color_windows_cui()
if !has('win32') || has('gui_running')
throw 'Skipped: only for the Windows CUI'
endif
if exists('$APPVEYOR')
throw 'Skipped: this test cannot be performed because AppVeyor does not support ANSI escape sequences'
endif
call assert_equal('dark', &background)
" Outputs 16 ANSI colors as background colors
let ansi = ''
for i in range(16)
let ansi ..= printf("\e[%dm%X", i + (i < 8 ? 40 : 92), i)
endfor
let ansi ..= "\e[40m\n"
call writefile([ansi], 'XANSIcolor', 'D')
let expected_chars = '0123456789ABCDEF'
let expected_colors = [
\ '#000000', '#e00000', '#00e000', '#e0e000',
\ '#0000e0', '#e000e0', '#00e0e0', '#e0e0e0',
\ '#808080', '#ff4040', '#40ff40', '#ffff40',
\ '#4040ff', '#ff40ff', '#40ffff', '#ffffff',
\ ]
" Ideally, 16 colors should be checked. However, in the current CI
" environment, the 16th color is something other than white, and we don't
" know the cause nor solution. After discussion, we decided to check only 15
" colors for the time being.
" FIXME: Check all 16 colors in the future.
let len_to_check = 15
let expected_colors = expected_colors[:len_to_check-1]
" First, make sure vim can display correct ANSI color text in terminal.
let buf = term_start("cmd /C type XANSIcolor")
call WaitForAssert({-> assert_equal(expected_chars, term_scrape(buf, 1)[:15]->map({_, v -> v['chars']})->join(''))})
call assert_equal(expected_colors, term_scrape(buf, 1)[:len_to_check-1]->map({_, v -> v['bg']}))
bwipeout!
" Next, check if vim can do the same thing in the vim terminal in terminal.
let lines = [
\ 'call term_start("cmd /C type XANSIcolor")'
\ ]
call writefile(lines, 'XloadANSI', 'D')
let cmd = GetVimCommandCleanTerm()
let buf = term_start(cmd .. '-S XloadANSI')
call WaitForAssert({-> assert_equal(expected_chars, term_scrape(buf, 1)[:15]->map({_, v -> v['chars']})->join(''))})
call assert_equal(expected_colors, term_scrape(buf, 1)[:len_to_check-1]->map({_, v -> v['bg']}))
endfunc
func Test_terminal_backspace_on_windows()
if !has('win32')
throw 'Skipped: only for the Windows CUI'
endif
" Specify a simple prompt for easy comparison
let save_prompt = $PROMPT
let $PROMPT = '>'
" Return the prompt line before the cursor
func s:get_cmd_prompt(buf)
let cur = term_getcursor(a:buf)
return term_getline(a:buf, cur[0])[:cur[1]-2]
endfunc
let buf = term_start('cmd.exe')
call WaitForAssert({-> assert_equal('>', s:get_cmd_prompt(buf))}, 100)
" Verify sent characters are echoed back
call term_sendkeys(buf, 'foo bar')
call WaitForAssert({-> assert_equal('>foo bar', s:get_cmd_prompt(buf))}, 100)
" Backspace should delete a character in front of the cursor
call term_sendkeys(buf, "\<BS>")
call WaitForAssert({-> assert_equal('>foo ba', s:get_cmd_prompt(buf))}, 100)
" Ctrl+H behaves like Backspace
call term_sendkeys(buf, "\<C-H>")
call WaitForAssert({-> assert_equal('>foo b', s:get_cmd_prompt(buf))}, 100)
" Send a total of four BS and Ctrl+H to erase four characters.
call term_sendkeys(buf, "\<BS>\<BS>\<C-H>\<C-H>")
call WaitForAssert({-> assert_equal('>f', s:get_cmd_prompt(buf))}, 100)
delfunc s:get_cmd_prompt
let $PROMPT = save_prompt
endfunc
func Test_terminal_split_utf8()
CheckUnix
let buf = term_start('cat', {})
let chan = buf->term_getjob()->job_getchannel()
call ch_sendraw(chan, "1: \xc3")
call WaitForAssert({-> assert_equal('1: ', term_getline(buf, 1))})
call ch_sendraw(chan, "\xa5\xcc\xb2\n")
call WaitForAssert({-> assert_equal('1: å̲', term_getline(buf, 1))})
call WaitForAssert({-> assert_equal('1: å̲', term_getline(buf, 2))})
call ch_sendraw(chan, "2: \xc3\xa5")
call WaitForAssert({-> assert_equal('2: å', term_getline(buf, 3))})
call ch_sendraw(chan, "\xcc\xb2\n")
call WaitForAssert({-> assert_equal('2: å̲', term_getline(buf, 3))})
call WaitForAssert({-> assert_equal('2: å̲', term_getline(buf, 4))})
call ch_sendraw(chan, "3: \xc3\xa5\xcc")
call WaitForAssert({-> assert_equal('3: å', term_getline(buf, 5))})
call ch_sendraw(chan, "\xb2\n")
call WaitForAssert({-> assert_equal('3: å̲', term_getline(buf, 5))})
call WaitForAssert({-> assert_equal('3: å̲', term_getline(buf, 6))})
exe buf .. "bwipe!"
endfunc
func Test_terminal_max_combining_chars()
" somehow doesn't work on MS-Windows
CheckUnix
let cmd = "cat samples/terminal_max_combining_chars.txt\<CR>"
let buf = Run_shell_in_terminal({'term_rows': 15, 'term_cols': 35})
call TermWait(buf)
call term_sendkeys(buf, cmd)
" last char is a space with many combining chars
call WaitForAssert({-> assert_match("AAAAAAAAAAAAAAAAAAAAAAAAAAAA.", term_getline(buf, 14))})
call term_sendkeys(buf, "exit\r")
exe buf . "bwipe!"
endfunc
func Test_term_getpos()
CheckRunVimInTerminal
CheckUnix
CheckExecutable seq
defer delete('XTest_getpos_result')
let lines =<< trim EOL
term ++curwin sh
EOL
call writefile(lines, 'XTest_getpos', 'D')
let buf = RunVimInTerminal('-S XTest_getpos', {'rows': 15})
call term_sendkeys(buf, "for i in `seq 1 30`; do echo line$i; done\<cr>")
call WaitForAssert({-> assert_match("line18", term_getline(buf, 1))})
call WaitForAssert({-> assert_match("line30", term_getline(buf, 13))})
call term_sendkeys(buf, "\<c-w>:let g:job_w0 = line('w0')\<cr>")
call term_sendkeys(buf, "\<c-w>:let g:job_wdollar = line('w$')\<cr>")
call term_sendkeys(buf, "\<c-w>:call writefile([string(g:job_w0), string(g:job_wdollar)], 'XTest_getpos_result')\<cr>")
call WaitForAssert({-> assert_true(filereadable('XTest_getpos_result'))})
call WaitForAssert({-> assert_equal(2, len(readfile('XTest_getpos_result')))})
let job_result = readfile('XTest_getpos_result')
" 15 - 1: statusline - 1: prompt line
call assert_equal(13, str2nr(job_result[1]) - str2nr(job_result[0]))
call assert_true(str2nr(job_result[0]) > 1)
call delete('XTest_getpos_result')
" switch to Terminal-Normal mode and record w0/w$
call term_sendkeys(buf, "\<c-w>N")
call term_sendkeys(buf, ":let g:w0 = line('w0')\<cr>")
call term_sendkeys(buf, ":let g:wdollar = line('w$')\<cr>")
call term_sendkeys(buf, ":call writefile([string(g:w0), string(g:wdollar)], 'XTest_getpos_result')\<cr>")
call WaitForAssert({-> assert_true(filereadable('XTest_getpos_result'))})
call WaitForAssert({-> assert_equal(2, len(readfile('XTest_getpos_result')))})
let result = readfile('XTest_getpos_result')
" 15 - 1: statusline - 1: for prompt line
call assert_equal(13, str2nr(result[1]) - str2nr(result[0]))
call assert_true(str2nr(result[0]) > 1)
" Regression: line('w0') and line('w$') must not move cursor position
call term_sendkeys(buf, "gg")
call term_sendkeys(buf, ":call line('w0')\<cr>")
call term_sendkeys(buf, ":call line('w$')\<cr>")
call term_wait(buf)
call WaitForAssert({-> assert_match("for i in", term_getline(buf, 1))})
call WaitForAssert({-> assert_match("line12", term_getline(buf, 13))})
call StopVimInTerminal(buf)
" this crashed
new
setl buftype=terminal
call assert_equal(2, line('w0') + line('w$'))
bw
endfunc
func Test_term_autowrite()
set autowrite
new termautowritetestfile
call setline(1, 'test content')
term echo "test"
call assert_equal(['test content'], readfile('termautowritetestfile'))
call delete('termautowritetestfile')
bwipe!
set noautowrite
endfunc
" Test that CSI sequences with more than CSI_ARGS_MAX arguments do not crash
func Test_terminal_csi_args_overflow()
CheckExecutable printf
let seq = "\033[" .. repeat('1;', 49) .. '1m'
let seq ..= "\033[1111111111111111111m"
let buf = term_start([&shell, &shellcmdflag, 'printf "' .. seq .. '"'])
" If we get here without a crash, the fix works
call assert_equal('running', term_getstatus(buf))
call StopVimInTerminal(buf)
endfunc
func Test_terminal_output_combining_chars()
CheckUnix
new
let cmd = "cat samples/combining_chars.txt"
let buf = term_start(cmd, {'curwin': 1, 'term_finish': 'open', 'term_rows': 10, 'term_cols': 30})
call WaitForAssert({-> assert_match('finished', term_getstatus(buf))})
call TermWait(buf)
let lines = getbufline(buf, 1, '$')
" get byte lengths to confirm combining chars present
let lens = map(copy(lines), 'len(v:val)')
let expected = repeat([11], 190) + repeat([14], 10)
call assert_equal(expected, lens)
bw!
endfunc
" This caused a Crash
func Test_terminal_csi_resize_oob()
return
CheckUnix
CheckExecutable printf
" CSI 8 ; rows ; cols t with missing, zero or negative dimensions reached
" on_resize()/resize_buffer() unvalidated, causing a negative-size memmove()
" and out-of-bounds set_lineinfo()/DECALN accesses in libvterm. Rendering
" these must not crash Vim.
" Sequences:
" 1 resize_buffer negative-size memmove
" 2 set_lineinfo OOB after corrupt resize
" 3 DECALN putglyph OOB after corrupt resize
let seqs = ["\<ESC>[8;0;t",
\ "\<ESC>[8;;t\<ESC>[J",
\ "\<ESC>[8;;0t\<ESC>#8"]
for seq in seqs
let buf = term_start([&shell, &shellcmdflag, 'printf "%s" ' .. shellescape(seq)],
\ #{term_rows: 10, term_cols: 40})
call TermWait(buf)
" Getting here without a crash (and no ASAN report) is the test.
call assert_true(bufexists(buf))
exe 'bwipe! ' .. buf
endfor
endfunc
" This caused a Crash, but Vim builds libvterm using -DWCWIDTH_FUNCTION=utf_uint2cells
" which wouldn't return -1 and therefore does not reproduce here
func Test_terminal_negative_col_oob()
CheckUnix
CheckExecutable printf
" A wcwidth() == -1 codepoint (U+0087, \302\207 in UTF-8) drove
" state->pos.col negative, then used as an array index in the tabstop
" helpers and moverect_internal(). These are single-byte / single-bit
" out-of-bounds accesses that do not crash a normal build, so this test
" is only meaningful under a sanitizer build; otherwise it just confirms
" Vim does not crash.
" Sequences:
" 1. clear_col_tabstop OOB read
" 2. is_col_tabstop OOB read
" 3. set_col_tabstop OOB write (HTS)
" 4. moverect_internal memmove (insert mode)
let seqs = ["\302\207\<ESC>[g",
\ "\302\207\302\207\t",
\ "\302\207\<ESC>H",
\ "\<ESC>[4h\302\2070"]
for seq in seqs
let buf = term_start([&shell, &shellcmdflag, 'printf "%s" ' .. shellescape(seq)],
\ #{term_rows: 10, term_cols: 40})
call TermWait(buf)
call assert_true(bufexists(buf))
exe 'bwipe! ' .. buf
endfor
endfunc
" vim: shiftwidth=2 sts=2 expandtab