mirror of
https://github.com/neovim/neovim.git
synced 2025-02-25 18:55:25 -06:00
This test exposes a bug (present on current master).
Steps to reproduce:
CC=clang make CMAKE_BUILD_TYPE=Debug CMAKE_EXTRA_FLAGS="-DCLANG_ASAN_UBSAN=ON"
export ASAN_SYMBOLIZER_PATH=/usr/lib/llvm-5.0/bin/llvm-symbolizer
TEST_FILE=test_autocmd.res make oldtest
==3668==ERROR: AddressSanitizer: heap-use-after-free on address 0x6260001411c8 at pc 0x000000a38a47 bp 0x7fff9982ee50 sp 0x7fff9982ee48
READ of size 4 at 0x6260001411c8 thread T0
0 0xa38a46 in _typval_encode_nothing_convert_one_value /home/vagrant/neovim/build/../src/nvim/eval/typval_encode.c.h:320:15
1 0xa340d5 in encode_vim_to_nothing /home/vagrant/neovim/build/../src/nvim/eval/typval_encode.c.h:830:9
2 0xa0ad63 in tv_clear /home/vagrant/neovim/build/../src/nvim/eval/typval.c:2189:25
3 0x85d584 in vars_clear_ext /home/vagrant/neovim/build/../src/nvim/eval.c:18894:9
4 0x7efac9 in vars_clear /home/vagrant/neovim/build/../src/nvim/eval.c:18871:3
5 0x7ef471 in eval_clear /home/vagrant/neovim/build/../src/nvim/eval.c:638:3
6 0xf80a7c in free_all_mem /home/vagrant/neovim/build/../src/nvim/memory.c:676:3
7 0x12789dc in mch_exit /home/vagrant/neovim/build/../src/nvim/os_unix.c:152:3
8 0xe85039 in getout /home/vagrant/neovim/build/../src/nvim/main.c:671:3
9 0xbb4496 in ex_quit_all /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:6051:5
10 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
11 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
12 0x845302 in call_user_func /home/vagrant/neovim/build/../src/nvim/eval.c:21332:3
13 0x81097b in call_func /home/vagrant/neovim/build/../src/nvim/eval.c:6358:11
14 0x825778 in get_func_tv /home/vagrant/neovim/build/../src/nvim/eval.c:6120:11
15 0x81e361 in ex_call /home/vagrant/neovim/build/../src/nvim/eval.c:2735:9
16 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
17 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
18 0xb17fdd in do_source /home/vagrant/neovim/build/../src/nvim/ex_cmds2.c:2973:3
19 0xb14630 in cmd_source /home/vagrant/neovim/build/../src/nvim/ex_cmds2.c:2718:14
20 0xb14727 in ex_source /home/vagrant/neovim/build/../src/nvim/ex_cmds2.c:2699:3
21 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
22 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
23 0xb34015 in do_cmdline_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:280:10
24 0xe829d3 in exe_commands /home/vagrant/neovim/build/../src/nvim/main.c:1702:5
25 0xe708bd in main /home/vagrant/neovim/build/../src/nvim/main.c:524:5
26 0x7f0cd5b7e82f in __libc_start_main /build/glibc-Cl5G7W/glibc-2.23/csu/../csu/libc-start.c:291
27 0x44d628 in _start (/home/vagrant/neovim/build/bin/nvim+0x44d628)
0x6260001411c8 is located 200 bytes inside of 10256-byte region [0x626000141100,0x626000143910)
freed by thread T0 here:
0 0x50df60 in __interceptor_cfree.localalias.0 (/home/vagrant/neovim/build/bin/nvim+0x50df60)
1 0xf7dbe4 in xfree /home/vagrant/neovim/build/../src/nvim/memory.c:133:3
2 0x676577 in free_buffer /home/vagrant/neovim/build/../src/nvim/buffer.c:749:5
3 0x66e15b in close_buffer /home/vagrant/neovim/build/../src/nvim/buffer.c:590:5
4 0x67a835 in do_buffer /home/vagrant/neovim/build/../src/nvim/buffer.c:1216:9
5 0x681a1d in do_bufdel /home/vagrant/neovim/build/../src/nvim/buffer.c:945:16
6 0xb9cd72 in ex_bunload /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:4585:17
7 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
8 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
9 0x845302 in call_user_func /home/vagrant/neovim/build/../src/nvim/eval.c:21332:3
10 0x81097b in call_func /home/vagrant/neovim/build/../src/nvim/eval.c:6358:11
11 0x825778 in get_func_tv /home/vagrant/neovim/build/../src/nvim/eval.c:6120:11
12 0x81e361 in ex_call /home/vagrant/neovim/build/../src/nvim/eval.c:2735:9
13 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
14 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
15 0x8646b1 in ex_execute /home/vagrant/neovim/build/../src/nvim/eval.c:19478:7
16 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
17 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
18 0x845302 in call_user_func /home/vagrant/neovim/build/../src/nvim/eval.c:21332:3
19 0x81097b in call_func /home/vagrant/neovim/build/../src/nvim/eval.c:6358:11
20 0x825778 in get_func_tv /home/vagrant/neovim/build/../src/nvim/eval.c:6120:11
21 0x81e361 in ex_call /home/vagrant/neovim/build/../src/nvim/eval.c:2735:9
22 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
23 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
24 0xb17fdd in do_source /home/vagrant/neovim/build/../src/nvim/ex_cmds2.c:2973:3
25 0xb14630 in cmd_source /home/vagrant/neovim/build/../src/nvim/ex_cmds2.c:2718:14
26 0xb14727 in ex_source /home/vagrant/neovim/build/../src/nvim/ex_cmds2.c:2699:3
27 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
28 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
29 0xb34015 in do_cmdline_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:280:10
previously allocated by thread T0 here:
0 0x50e340 in calloc (/home/vagrant/neovim/build/bin/nvim+0x50e340)
1 0xf7dc97 in xcalloc /home/vagrant/neovim/build/../src/nvim/memory.c:147:15
2 0x67eb65 in buflist_new /home/vagrant/neovim/build/../src/nvim/buffer.c:1641:11
3 0xa8fdcc in do_ecmd /home/vagrant/neovim/build/../src/nvim/ex_cmds.c:2221:13
4 0x683b21 in empty_curbuf /home/vagrant/neovim/build/../src/nvim/buffer.c:1031:12
5 0x67a098 in do_buffer /home/vagrant/neovim/build/../src/nvim/buffer.c:1196:14
6 0x68173f in do_bufdel /home/vagrant/neovim/build/../src/nvim/buffer.c:927:11
7 0xb9cd72 in ex_bunload /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:4585:17
8 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
9 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
10 0x845302 in call_user_func /home/vagrant/neovim/build/../src/nvim/eval.c:21332:3
11 0x81097b in call_func /home/vagrant/neovim/build/../src/nvim/eval.c:6358:11
12 0x825778 in get_func_tv /home/vagrant/neovim/build/../src/nvim/eval.c:6120:11
13 0x81e361 in ex_call /home/vagrant/neovim/build/../src/nvim/eval.c:2735:9
14 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
15 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
16 0x8646b1 in ex_execute /home/vagrant/neovim/build/../src/nvim/eval.c:19478:7
17 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
18 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
19 0x845302 in call_user_func /home/vagrant/neovim/build/../src/nvim/eval.c:21332:3
20 0x81097b in call_func /home/vagrant/neovim/build/../src/nvim/eval.c:6358:11
21 0x825778 in get_func_tv /home/vagrant/neovim/build/../src/nvim/eval.c:6120:11
22 0x81e361 in ex_call /home/vagrant/neovim/build/../src/nvim/eval.c:2735:9
23 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
24 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
25 0xb17fdd in do_source /home/vagrant/neovim/build/../src/nvim/ex_cmds2.c:2973:3
26 0xb14630 in cmd_source /home/vagrant/neovim/build/../src/nvim/ex_cmds2.c:2718:14
27 0xb14727 in ex_source /home/vagrant/neovim/build/../src/nvim/ex_cmds2.c:2699:3
28 0xb4ba9c in do_one_cmd /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:2242:5
29 0xb2dd03 in do_cmdline /home/vagrant/neovim/build/../src/nvim/ex_docmd.c:609:20
SUMMARY: AddressSanitizer: heap-use-after-free /home/vagrant/neovim/build/../src/nvim/eval/typval_encode.c.h:320:15 in _typval_encode_nothing_convert_one_va
lue
Shadow bytes around the buggy address:
0x0c4c800201e0: fa fa fa fa fa fa fa fa fa fa fa fa fa fa fa fa
0x0c4c800201f0: fa fa fa fa fa fa fa fa fa fa fa fa fa fa fa fa
0x0c4c80020200: fa fa fa fa fa fa fa fa fa fa fa fa fa fa fa fa
0x0c4c80020210: fa fa fa fa fa fa fa fa fa fa fa fa fa fa fa fa
0x0c4c80020220: fd fd fd fd fd fd fd fd fd fd fd fd fd fd fd fd
=>0x0c4c80020230: fd fd fd fd fd fd fd fd fd[fd]fd fd fd fd fd fd
0x0c4c80020240: fd fd fd fd fd fd fd fd fd fd fd fd fd fd fd fd
0x0c4c80020250: fd fd fd fd fd fd fd fd fd fd fd fd fd fd fd fd
0x0c4c80020260: fd fd fd fd fd fd fd fd fd fd fd fd fd fd fd fd
0x0c4c80020270: fd fd fd fd fd fd fd fd fd fd fd fd fd fd fd fd
0x0c4c80020280: fd fd fd fd fd fd fd fd fd fd fd fd fd fd fd fd
Shadow byte legend (one shadow byte represents 8 application bytes):
Addressable: 00
Partially addressable: 01 02 03 04 05 06 07
Heap left redzone: fa
Freed heap region: fd
Stack left redzone: f1
Stack mid redzone: f2
Stack right redzone: f3
Stack after return: f5
Stack use after scope: f8
Global redzone: f9
Global init order: f6
Poisoned by user: f7
Container overflow: fc
Array cookie: ac
Intra object redzone: bb
ASan internal: fe
Left alloca redzone: ca
Right alloca redzone: cb
==3668==ABORTING
1167 lines
30 KiB
VimL
1167 lines
30 KiB
VimL
" Tests for autocommands
|
|
|
|
set belloff=all
|
|
|
|
func! s:cleanup_buffers() abort
|
|
for bnr in range(1, bufnr('$'))
|
|
if bufloaded(bnr) && bufnr('%') != bnr
|
|
execute 'bd! ' . bnr
|
|
endif
|
|
endfor
|
|
endfunc
|
|
|
|
func Test_vim_did_enter()
|
|
call assert_false(v:vim_did_enter)
|
|
|
|
" This script will never reach the main loop, can't check if v:vim_did_enter
|
|
" becomes one.
|
|
endfunc
|
|
|
|
if has('timers')
|
|
func ExitInsertMode(id)
|
|
call feedkeys("\<Esc>")
|
|
endfunc
|
|
|
|
func Test_cursorhold_insert()
|
|
" Need to move the cursor.
|
|
call feedkeys("ggG", "xt")
|
|
|
|
let g:triggered = 0
|
|
au CursorHoldI * let g:triggered += 1
|
|
set updatetime=20
|
|
call timer_start(100, 'ExitInsertMode')
|
|
call feedkeys('a', 'x!')
|
|
call assert_equal(1, g:triggered)
|
|
au! CursorHoldI
|
|
set updatetime&
|
|
endfunc
|
|
|
|
func Test_cursorhold_insert_ctrl_x()
|
|
let g:triggered = 0
|
|
au CursorHoldI * let g:triggered += 1
|
|
set updatetime=20
|
|
call timer_start(100, 'ExitInsertMode')
|
|
" CursorHoldI does not trigger after CTRL-X
|
|
call feedkeys("a\<C-X>", 'x!')
|
|
call assert_equal(0, g:triggered)
|
|
au! CursorHoldI
|
|
set updatetime&
|
|
endfunc
|
|
endif
|
|
|
|
func Test_bufunload()
|
|
augroup test_bufunload_group
|
|
autocmd!
|
|
autocmd BufUnload * call add(s:li, "bufunload")
|
|
autocmd BufDelete * call add(s:li, "bufdelete")
|
|
autocmd BufWipeout * call add(s:li, "bufwipeout")
|
|
augroup END
|
|
|
|
let s:li=[]
|
|
new
|
|
setlocal bufhidden=
|
|
bunload
|
|
call assert_equal(["bufunload", "bufdelete"], s:li)
|
|
|
|
let s:li=[]
|
|
new
|
|
setlocal bufhidden=delete
|
|
bunload
|
|
call assert_equal(["bufunload", "bufdelete"], s:li)
|
|
|
|
let s:li=[]
|
|
new
|
|
setlocal bufhidden=unload
|
|
bwipeout
|
|
call assert_equal(["bufunload", "bufdelete", "bufwipeout"], s:li)
|
|
|
|
au! test_bufunload_group
|
|
augroup! test_bufunload_group
|
|
endfunc
|
|
|
|
" SEGV occurs in older versions. (At least 7.4.2005 or older)
|
|
func Test_autocmd_bufunload_with_tabnext()
|
|
tabedit
|
|
tabfirst
|
|
|
|
augroup test_autocmd_bufunload_with_tabnext_group
|
|
autocmd!
|
|
autocmd BufUnload <buffer> tabnext
|
|
augroup END
|
|
|
|
quit
|
|
call assert_equal(2, tabpagenr('$'))
|
|
|
|
autocmd! test_autocmd_bufunload_with_tabnext_group
|
|
augroup! test_autocmd_bufunload_with_tabnext_group
|
|
tablast
|
|
quit
|
|
endfunc
|
|
|
|
func Test_autocmd_bufwinleave_with_tabfirst()
|
|
tabedit
|
|
augroup sample
|
|
autocmd!
|
|
autocmd BufWinLeave <buffer> tabfirst
|
|
augroup END
|
|
call setline(1, ['a', 'b', 'c'])
|
|
edit! a.txt
|
|
tabclose
|
|
endfunc
|
|
|
|
" SEGV occurs in older versions. (At least 7.4.2321 or older)
|
|
func Test_autocmd_bufunload_avoiding_SEGV_01()
|
|
split aa.txt
|
|
let lastbuf = bufnr('$')
|
|
|
|
augroup test_autocmd_bufunload
|
|
autocmd!
|
|
exe 'autocmd BufUnload <buffer> ' . (lastbuf + 1) . 'bwipeout!'
|
|
augroup END
|
|
|
|
call assert_fails('edit bb.txt', 'E937:')
|
|
|
|
autocmd! test_autocmd_bufunload
|
|
augroup! test_autocmd_bufunload
|
|
bwipe! aa.txt
|
|
bwipe! bb.txt
|
|
endfunc
|
|
|
|
" SEGV occurs in older versions. (At least 7.4.2321 or older)
|
|
func Test_autocmd_bufunload_avoiding_SEGV_02()
|
|
setlocal buftype=nowrite
|
|
let lastbuf = bufnr('$')
|
|
|
|
augroup test_autocmd_bufunload
|
|
autocmd!
|
|
exe 'autocmd BufUnload <buffer> ' . (lastbuf + 1) . 'bwipeout!'
|
|
augroup END
|
|
|
|
normal! i1
|
|
call assert_fails('edit a.txt', 'E517:')
|
|
call feedkeys("\<CR>")
|
|
|
|
autocmd! test_autocmd_bufunload
|
|
augroup! test_autocmd_bufunload
|
|
bwipe! a.txt
|
|
endfunc
|
|
|
|
func Test_win_tab_autocmd()
|
|
let g:record = []
|
|
|
|
augroup testing
|
|
au WinNew * call add(g:record, 'WinNew')
|
|
au WinEnter * call add(g:record, 'WinEnter')
|
|
au WinLeave * call add(g:record, 'WinLeave')
|
|
au TabNew * call add(g:record, 'TabNew')
|
|
au TabClosed * call add(g:record, 'TabClosed')
|
|
au TabEnter * call add(g:record, 'TabEnter')
|
|
au TabLeave * call add(g:record, 'TabLeave')
|
|
augroup END
|
|
|
|
split
|
|
tabnew
|
|
close
|
|
close
|
|
|
|
call assert_equal([
|
|
\ 'WinLeave', 'WinNew', 'WinEnter',
|
|
\ 'WinLeave', 'TabLeave', 'WinNew', 'WinEnter', 'TabNew', 'TabEnter',
|
|
\ 'WinLeave', 'TabLeave', 'TabClosed', 'WinEnter', 'TabEnter',
|
|
\ 'WinLeave', 'WinEnter'
|
|
\ ], g:record)
|
|
|
|
let g:record = []
|
|
tabnew somefile
|
|
tabnext
|
|
bwipe somefile
|
|
|
|
call assert_equal([
|
|
\ 'WinLeave', 'TabLeave', 'WinNew', 'WinEnter', 'TabNew', 'TabEnter',
|
|
\ 'WinLeave', 'TabLeave', 'WinEnter', 'TabEnter',
|
|
\ 'TabClosed'
|
|
\ ], g:record)
|
|
|
|
augroup testing
|
|
au!
|
|
augroup END
|
|
unlet g:record
|
|
endfunc
|
|
|
|
func s:AddAnAutocmd()
|
|
augroup vimBarTest
|
|
au BufReadCmd * echo 'hello'
|
|
augroup END
|
|
call assert_equal(3, len(split(execute('au vimBarTest'), "\n")))
|
|
endfunc
|
|
|
|
func Test_early_bar()
|
|
" test that a bar is recognized before the {event}
|
|
call s:AddAnAutocmd()
|
|
augroup vimBarTest | au! | augroup END
|
|
call assert_equal(1, len(split(execute('au vimBarTest'), "\n")))
|
|
|
|
call s:AddAnAutocmd()
|
|
augroup vimBarTest| au!| augroup END
|
|
call assert_equal(1, len(split(execute('au vimBarTest'), "\n")))
|
|
|
|
" test that a bar is recognized after the {event}
|
|
call s:AddAnAutocmd()
|
|
augroup vimBarTest| au!BufReadCmd| augroup END
|
|
call assert_equal(1, len(split(execute('au vimBarTest'), "\n")))
|
|
|
|
" test that a bar is recognized after the {group}
|
|
call s:AddAnAutocmd()
|
|
au! vimBarTest|echo 'hello'
|
|
call assert_equal(1, len(split(execute('au vimBarTest'), "\n")))
|
|
endfunc
|
|
|
|
func RemoveGroup()
|
|
autocmd! StartOK
|
|
augroup! StartOK
|
|
endfunc
|
|
|
|
func Test_augroup_warning()
|
|
augroup TheWarning
|
|
au VimEnter * echo 'entering'
|
|
augroup END
|
|
call assert_true(match(execute('au VimEnter'), "TheWarning.*VimEnter") >= 0)
|
|
redir => res
|
|
augroup! TheWarning
|
|
redir END
|
|
call assert_true(match(res, "W19:") >= 0)
|
|
call assert_true(match(execute('au VimEnter'), "-Deleted-.*VimEnter") >= 0)
|
|
|
|
" check "Another" does not take the pace of the deleted entry
|
|
augroup Another
|
|
augroup END
|
|
call assert_true(match(execute('au VimEnter'), "-Deleted-.*VimEnter") >= 0)
|
|
augroup! Another
|
|
|
|
" no warning for postpone aucmd delete
|
|
augroup StartOK
|
|
au VimEnter * call RemoveGroup()
|
|
augroup END
|
|
call assert_true(match(execute('au VimEnter'), "StartOK.*VimEnter") >= 0)
|
|
redir => res
|
|
doautocmd VimEnter
|
|
redir END
|
|
call assert_true(match(res, "W19:") < 0)
|
|
au! VimEnter
|
|
endfunc
|
|
|
|
func Test_augroup_deleted()
|
|
" This caused a crash before E936 was introduced
|
|
augroup x
|
|
call assert_fails('augroup! x', 'E936:')
|
|
au VimEnter * echo
|
|
augroup end
|
|
augroup! x
|
|
call assert_true(match(execute('au VimEnter'), "-Deleted-.*VimEnter") >= 0)
|
|
au! VimEnter
|
|
endfunc
|
|
|
|
" Tests for autocommands on :close command.
|
|
" This used to be in test13.
|
|
func Test_three_windows()
|
|
" Clean up buffers, because in some cases this function fails.
|
|
call s:cleanup_buffers()
|
|
|
|
" Write three files and open them, each in a window.
|
|
" Then go to next window, with autocommand that deletes the previous one.
|
|
" Do this twice, writing the file.
|
|
e! Xtestje1
|
|
call setline(1, 'testje1')
|
|
w
|
|
sp Xtestje2
|
|
call setline(1, 'testje2')
|
|
w
|
|
sp Xtestje3
|
|
call setline(1, 'testje3')
|
|
w
|
|
wincmd w
|
|
au WinLeave Xtestje2 bwipe
|
|
wincmd w
|
|
call assert_equal('Xtestje1', expand('%'))
|
|
|
|
au WinLeave Xtestje1 bwipe Xtestje3
|
|
close
|
|
call assert_equal('Xtestje1', expand('%'))
|
|
|
|
" Test deleting the buffer on a Unload event. If this goes wrong there
|
|
" will be the ATTENTION prompt.
|
|
e Xtestje1
|
|
au!
|
|
au! BufUnload Xtestje1 bwipe
|
|
call assert_fails('e Xtestje3', 'E937:')
|
|
call assert_equal('Xtestje3', expand('%'))
|
|
|
|
e Xtestje2
|
|
sp Xtestje1
|
|
call assert_fails('e', 'E937:')
|
|
call assert_equal('Xtestje2', expand('%'))
|
|
|
|
" Test changing buffers in a BufWipeout autocommand. If this goes wrong
|
|
" there are ml_line errors and/or a Crash.
|
|
au!
|
|
only
|
|
e Xanother
|
|
e Xtestje1
|
|
bwipe Xtestje2
|
|
bwipe Xtestje3
|
|
au BufWipeout Xtestje1 buf Xtestje1
|
|
bwipe
|
|
call assert_equal('Xanother', expand('%'))
|
|
|
|
only
|
|
|
|
helptags ALL
|
|
help
|
|
wincmd w
|
|
1quit
|
|
call assert_equal('Xanother', expand('%'))
|
|
|
|
au!
|
|
enew
|
|
bwipe! Xtestje1
|
|
call delete('Xtestje1')
|
|
call delete('Xtestje2')
|
|
call delete('Xtestje3')
|
|
endfunc
|
|
|
|
func Test_BufEnter()
|
|
au! BufEnter
|
|
au Bufenter * let val = val . '+'
|
|
let g:val = ''
|
|
split NewFile
|
|
call assert_equal('+', g:val)
|
|
bwipe!
|
|
call assert_equal('++', g:val)
|
|
|
|
" Also get BufEnter when editing a directory
|
|
call mkdir('Xdir')
|
|
split Xdir
|
|
call assert_equal('+++', g:val)
|
|
|
|
" On MS-Windows we can't edit the directory, make sure we wipe the right
|
|
" buffer.
|
|
bwipe! Xdir
|
|
|
|
call delete('Xdir', 'd')
|
|
au! BufEnter
|
|
endfunc
|
|
|
|
" Closing a window might cause an endless loop
|
|
" E814 for older Vims
|
|
func Test_autocmd_bufwipe_in_SessLoadPost()
|
|
tabnew
|
|
set noswapfile
|
|
mksession!
|
|
|
|
let content = ['set nocp noswapfile',
|
|
\ 'let v:swapchoice="e"',
|
|
\ 'augroup test_autocmd_sessionload',
|
|
\ 'autocmd!',
|
|
\ 'autocmd SessionLoadPost * 4bw!',
|
|
\ 'augroup END',
|
|
\ '',
|
|
\ 'func WriteErrors()',
|
|
\ ' call writefile([execute("messages")], "Xerrors")',
|
|
\ 'endfunc',
|
|
\ 'au VimLeave * call WriteErrors()',
|
|
\ ]
|
|
call writefile(content, 'Xvimrc')
|
|
call system(v:progpath. ' --headless -i NONE -u Xvimrc --noplugins -S Session.vim -c cq')
|
|
let errors = join(readfile('Xerrors'))
|
|
call assert_match('E814', errors)
|
|
|
|
set swapfile
|
|
for file in ['Session.vim', 'Xvimrc', 'Xerrors']
|
|
call delete(file)
|
|
endfor
|
|
endfunc
|
|
|
|
" SEGV occurs in older versions.
|
|
func Test_autocmd_bufwipe_in_SessLoadPost2()
|
|
tabnew
|
|
set noswapfile
|
|
mksession!
|
|
|
|
let content = ['set nocp noswapfile',
|
|
\ 'function! DeleteInactiveBufs()',
|
|
\ ' tabfirst',
|
|
\ ' let tabblist = []',
|
|
\ ' for i in range(1, tabpagenr(''$''))',
|
|
\ ' call extend(tabblist, tabpagebuflist(i))',
|
|
\ ' endfor',
|
|
\ ' for b in range(1, bufnr(''$''))',
|
|
\ ' if bufexists(b) && buflisted(b) && (index(tabblist, b) == -1 || bufname(b) =~# ''^$'')',
|
|
\ ' exec ''bwipeout '' . b',
|
|
\ ' endif',
|
|
\ ' endfor',
|
|
\ ' echomsg "SessionLoadPost DONE"',
|
|
\ 'endfunction',
|
|
\ 'au SessionLoadPost * call DeleteInactiveBufs()',
|
|
\ '',
|
|
\ 'func WriteErrors()',
|
|
\ ' call writefile([execute("messages")], "Xerrors")',
|
|
\ 'endfunc',
|
|
\ 'au VimLeave * call WriteErrors()',
|
|
\ ]
|
|
call writefile(content, 'Xvimrc')
|
|
call system(v:progpath. ' --headless -i NONE -u Xvimrc --noplugins -S Session.vim -c cq')
|
|
let errors = join(readfile('Xerrors'))
|
|
" This probably only ever matches on unix.
|
|
call assert_notmatch('Caught deadly signal SEGV', errors)
|
|
call assert_match('SessionLoadPost DONE', errors)
|
|
|
|
set swapfile
|
|
for file in ['Session.vim', 'Xvimrc', 'Xerrors']
|
|
call delete(file)
|
|
endfor
|
|
endfunc
|
|
|
|
func Test_empty_doau()
|
|
doau \|
|
|
endfunc
|
|
|
|
func s:AutoCommandOptionSet(match)
|
|
let item = remove(g:options, 0)
|
|
let expected = printf("Option: <%s>, Oldval: <%s>, NewVal: <%s>, Scope: <%s>\n", item[0], item[1], item[2], item[3])
|
|
let actual = printf("Option: <%s>, Oldval: <%s>, NewVal: <%s>, Scope: <%s>\n", a:match, v:option_old, v:option_new, v:option_type)
|
|
let g:opt = [expected, actual]
|
|
"call assert_equal(expected, actual)
|
|
endfunc
|
|
|
|
func Test_OptionSet()
|
|
throw 'skipped: Nvim does not support test_override()'
|
|
if !has("eval") || !has("autocmd") || !exists("+autochdir")
|
|
return
|
|
endif
|
|
|
|
call test_override('starting', 1)
|
|
set nocp
|
|
au OptionSet * :call s:AutoCommandOptionSet(expand("<amatch>"))
|
|
|
|
" 1: Setting number option"
|
|
let g:options=[['number', 0, 1, 'global']]
|
|
set nu
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 2: Setting local number option"
|
|
let g:options=[['number', 1, 0, 'local']]
|
|
setlocal nonu
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 3: Setting global number option"
|
|
let g:options=[['number', 1, 0, 'global']]
|
|
setglobal nonu
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 4: Setting local autoindent option"
|
|
let g:options=[['autoindent', 0, 1, 'local']]
|
|
setlocal ai
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 5: Setting global autoindent option"
|
|
let g:options=[['autoindent', 0, 1, 'global']]
|
|
setglobal ai
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 6: Setting global autoindent option"
|
|
let g:options=[['autoindent', 1, 0, 'global']]
|
|
set ai!
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" Should not print anything, use :noa
|
|
" 7: don't trigger OptionSet"
|
|
let g:options=[['invalid', 1, 1, 'invalid']]
|
|
noa set nonu
|
|
call assert_equal([['invalid', 1, 1, 'invalid']], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 8: Setting several global list and number option"
|
|
let g:options=[['list', 0, 1, 'global'], ['number', 0, 1, 'global']]
|
|
set list nu
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 9: don't trigger OptionSet"
|
|
let g:options=[['invalid', 1, 1, 'invalid'], ['invalid', 1, 1, 'invalid']]
|
|
noa set nolist nonu
|
|
call assert_equal([['invalid', 1, 1, 'invalid'], ['invalid', 1, 1, 'invalid']], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 10: Setting global acd"
|
|
let g:options=[['autochdir', 0, 1, 'local']]
|
|
setlocal acd
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 11: Setting global autoread (also sets local value)"
|
|
let g:options=[['autoread', 0, 1, 'global']]
|
|
set ar
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 12: Setting local autoread"
|
|
let g:options=[['autoread', 1, 1, 'local']]
|
|
setlocal ar
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 13: Setting global autoread"
|
|
let g:options=[['autoread', 1, 0, 'global']]
|
|
setglobal invar
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 14: Setting option backspace through :let"
|
|
let g:options=[['backspace', '', 'eol,indent,start', 'global']]
|
|
let &bs="eol,indent,start"
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 15: Setting option backspace through setbufvar()"
|
|
let g:options=[['backup', 0, 1, 'local']]
|
|
" try twice, first time, shouldn't trigger because option name is invalid,
|
|
" second time, it should trigger
|
|
call assert_fails("call setbufvar(1, '&l:bk', 1)", "E355")
|
|
" should trigger, use correct option name
|
|
call setbufvar(1, '&backup', 1)
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 16: Setting number option using setwinvar"
|
|
let g:options=[['number', 0, 1, 'local']]
|
|
call setwinvar(0, '&number', 1)
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 17: Setting key option, shouldn't trigger"
|
|
let g:options=[['key', 'invalid', 'invalid1', 'invalid']]
|
|
setlocal key=blah
|
|
setlocal key=
|
|
call assert_equal([['key', 'invalid', 'invalid1', 'invalid']], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 18: Setting string option"
|
|
let oldval = &tags
|
|
let g:options=[['tags', oldval, 'tagpath', 'global']]
|
|
set tags=tagpath
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" 1l: Resetting string option"
|
|
let g:options=[['tags', 'tagpath', oldval, 'global']]
|
|
set tags&
|
|
call assert_equal([], g:options)
|
|
call assert_equal(g:opt[0], g:opt[1])
|
|
|
|
" Cleanup
|
|
au! OptionSet
|
|
for opt in ['nu', 'ai', 'acd', 'ar', 'bs', 'backup', 'cul', 'cp']
|
|
exe printf(":set %s&vi", opt)
|
|
endfor
|
|
call test_override('starting', 0)
|
|
delfunc! AutoCommandOptionSet
|
|
endfunc
|
|
|
|
func Test_OptionSet_diffmode()
|
|
throw 'skipped: Nvim does not support test_override()'
|
|
call test_override('starting', 1)
|
|
" 18: Changing an option when enetering diff mode
|
|
new
|
|
au OptionSet diff :let &l:cul=v:option_new
|
|
|
|
call setline(1, ['buffer 1', 'line2', 'line3', 'line4'])
|
|
call assert_equal(0, &l:cul)
|
|
diffthis
|
|
call assert_equal(1, &l:cul)
|
|
|
|
vnew
|
|
call setline(1, ['buffer 2', 'line 2', 'line 3', 'line4'])
|
|
call assert_equal(0, &l:cul)
|
|
diffthis
|
|
call assert_equal(1, &l:cul)
|
|
|
|
diffoff
|
|
call assert_equal(0, &l:cul)
|
|
call assert_equal(1, getwinvar(2, '&l:cul'))
|
|
bw!
|
|
|
|
call assert_equal(1, &l:cul)
|
|
diffoff!
|
|
call assert_equal(0, &l:cul)
|
|
call assert_equal(0, getwinvar(1, '&l:cul'))
|
|
bw!
|
|
|
|
" Cleanup
|
|
au! OptionSet
|
|
call test_override('starting', 0)
|
|
endfunc
|
|
|
|
func Test_OptionSet_diffmode_close()
|
|
throw 'skipped: Nvim does not support test_override()'
|
|
call test_override('starting', 1)
|
|
" 19: Try to close the current window when entering diff mode
|
|
" should not segfault
|
|
new
|
|
au OptionSet diff close
|
|
|
|
call setline(1, ['buffer 1', 'line2', 'line3', 'line4'])
|
|
call assert_fails(':diffthis', 'E788')
|
|
call assert_equal(1, &diff)
|
|
vnew
|
|
call setline(1, ['buffer 2', 'line 2', 'line 3', 'line4'])
|
|
call assert_fails(':diffthis', 'E788')
|
|
call assert_equal(1, &diff)
|
|
bw!
|
|
call assert_fails(':diffoff!', 'E788')
|
|
bw!
|
|
|
|
" Cleanup
|
|
au! OptionSet
|
|
call test_override('starting', 0)
|
|
"delfunc! AutoCommandOptionSet
|
|
endfunc
|
|
|
|
" Test for Bufleave autocommand that deletes the buffer we are about to edit.
|
|
func Test_BufleaveWithDelete()
|
|
new | edit Xfile1
|
|
|
|
augroup test_bufleavewithdelete
|
|
autocmd!
|
|
autocmd BufLeave Xfile1 bwipe Xfile2
|
|
augroup END
|
|
|
|
call assert_fails('edit Xfile2', 'E143:')
|
|
call assert_equal('Xfile1', bufname('%'))
|
|
|
|
autocmd! test_bufleavewithdelete BufLeave Xfile1
|
|
augroup! test_bufleavewithdelete
|
|
|
|
new
|
|
bwipe! Xfile1
|
|
endfunc
|
|
|
|
" Test for autocommand that changes the buffer list, when doing ":ball".
|
|
func Test_Acmd_BufAll()
|
|
enew!
|
|
%bwipe!
|
|
call writefile(['Test file Xxx1'], 'Xxx1')
|
|
call writefile(['Test file Xxx2'], 'Xxx2')
|
|
call writefile(['Test file Xxx3'], 'Xxx3')
|
|
|
|
" Add three files to the buffer list
|
|
split Xxx1
|
|
close
|
|
split Xxx2
|
|
close
|
|
split Xxx3
|
|
close
|
|
|
|
" Wipe the buffer when the buffer is opened
|
|
au BufReadPost Xxx2 bwipe
|
|
|
|
call append(0, 'Test file Xxx4')
|
|
ball
|
|
|
|
call assert_equal(2, winnr('$'))
|
|
call assert_equal('Xxx1', bufname(winbufnr(winnr('$'))))
|
|
wincmd t
|
|
|
|
au! BufReadPost
|
|
%bwipe!
|
|
call delete('Xxx1')
|
|
call delete('Xxx2')
|
|
call delete('Xxx3')
|
|
enew! | only
|
|
endfunc
|
|
|
|
" Test for autocommand that changes current buffer on BufEnter event.
|
|
" Check if modelines are interpreted for the correct buffer.
|
|
func Test_Acmd_BufEnter()
|
|
%bwipe!
|
|
call writefile(['start of test file Xxx1',
|
|
\ "\<Tab>this is a test",
|
|
\ 'end of test file Xxx1'], 'Xxx1')
|
|
call writefile(['start of test file Xxx2',
|
|
\ 'vim: set noai :',
|
|
\ "\<Tab>this is a test",
|
|
\ 'end of test file Xxx2'], 'Xxx2')
|
|
|
|
au BufEnter Xxx2 brew
|
|
set ai modeline modelines=3
|
|
edit Xxx1
|
|
" edit Xxx2, autocmd will do :brew
|
|
edit Xxx2
|
|
exe "normal G?this is a\<CR>"
|
|
" Append text with autoindent to this file
|
|
normal othis should be auto-indented
|
|
call assert_equal("\<Tab>this should be auto-indented", getline('.'))
|
|
call assert_equal(3, line('.'))
|
|
" Remove autocmd and edit Xxx2 again
|
|
au! BufEnter Xxx2
|
|
buf! Xxx2
|
|
exe "normal G?this is a\<CR>"
|
|
" append text without autoindent to Xxx
|
|
normal othis should be in column 1
|
|
call assert_equal("this should be in column 1", getline('.'))
|
|
call assert_equal(4, line('.'))
|
|
|
|
%bwipe!
|
|
call delete('Xxx1')
|
|
call delete('Xxx2')
|
|
set ai&vim modeline&vim modelines&vim
|
|
endfunc
|
|
|
|
" Test for issue #57
|
|
" do not move cursor on <c-o> when autoindent is set
|
|
func Test_ai_CTRL_O()
|
|
enew!
|
|
set ai
|
|
let save_fo = &fo
|
|
set fo+=r
|
|
exe "normal o# abcdef\<Esc>2hi\<CR>\<C-O>d0\<Esc>"
|
|
exe "normal o# abcdef\<Esc>2hi\<C-O>d0\<Esc>"
|
|
call assert_equal(['# abc', 'def', 'def'], getline(2, 4))
|
|
|
|
set ai&vim
|
|
let &fo = save_fo
|
|
enew!
|
|
endfunc
|
|
|
|
" Test for autocommand that deletes the current buffer on BufLeave event.
|
|
" Also test deleting the last buffer, should give a new, empty buffer.
|
|
func Test_BufLeave_Wipe()
|
|
throw 'skipped: TODO: '
|
|
%bwipe!
|
|
let content = ['start of test file Xxx',
|
|
\ 'this is a test',
|
|
\ 'end of test file Xxx']
|
|
call writefile(content, 'Xxx1')
|
|
call writefile(content, 'Xxx2')
|
|
|
|
au BufLeave Xxx2 bwipe
|
|
edit Xxx1
|
|
split Xxx2
|
|
" delete buffer Xxx2, we should be back to Xxx1
|
|
bwipe
|
|
call assert_equal('Xxx1', bufname('%'))
|
|
call assert_equal(1, winnr('$'))
|
|
|
|
" Create an alternate buffer
|
|
%write! test.out
|
|
call assert_equal('test.out', bufname('#'))
|
|
" delete alternate buffer
|
|
bwipe test.out
|
|
call assert_equal('Xxx1', bufname('%'))
|
|
call assert_equal('', bufname('#'))
|
|
|
|
au BufLeave Xxx1 bwipe
|
|
" delete current buffer, get an empty one
|
|
bwipe!
|
|
call assert_equal(1, line('$'))
|
|
call assert_equal('', bufname('%'))
|
|
let g:bufinfo = getbufinfo()
|
|
call assert_equal(1, len(g:bufinfo))
|
|
|
|
call delete('Xxx1')
|
|
call delete('Xxx2')
|
|
call delete('test.out')
|
|
%bwipe
|
|
au! BufLeave
|
|
|
|
" check that bufinfo doesn't contain a pointer to freed memory
|
|
call test_garbagecollect_now()
|
|
endfunc
|
|
|
|
func Test_QuitPre()
|
|
edit Xfoo
|
|
let winid = win_getid(winnr())
|
|
split Xbar
|
|
au! QuitPre * let g:afile = expand('<afile>')
|
|
" Close the other window, <afile> should be correct.
|
|
exe win_id2win(winid) . 'q'
|
|
call assert_equal('Xfoo', g:afile)
|
|
|
|
unlet g:afile
|
|
bwipe Xfoo
|
|
bwipe Xbar
|
|
endfunc
|
|
|
|
func Test_Cmdline()
|
|
au! CmdlineEnter : let g:entered = expand('<afile>')
|
|
au! CmdlineLeave : let g:left = expand('<afile>')
|
|
let g:entered = 0
|
|
let g:left = 0
|
|
call feedkeys(":echo 'hello'\<CR>", 'xt')
|
|
call assert_equal(':', g:entered)
|
|
call assert_equal(':', g:left)
|
|
au! CmdlineEnter
|
|
au! CmdlineLeave
|
|
|
|
au! CmdlineEnter / let g:entered = expand('<afile>')
|
|
au! CmdlineLeave / let g:left = expand('<afile>')
|
|
let g:entered = 0
|
|
let g:left = 0
|
|
new
|
|
call setline(1, 'hello')
|
|
call feedkeys("/hello\<CR>", 'xt')
|
|
call assert_equal('/', g:entered)
|
|
call assert_equal('/', g:left)
|
|
bwipe!
|
|
au! CmdlineEnter
|
|
au! CmdlineLeave
|
|
endfunc
|
|
|
|
" Test for BufWritePre autocommand that deletes or unloads the buffer.
|
|
func Test_BufWritePre()
|
|
%bwipe
|
|
au BufWritePre Xxx1 bunload
|
|
au BufWritePre Xxx2 bwipe
|
|
|
|
call writefile(['start of Xxx1', 'test', 'end of Xxx1'], 'Xxx1')
|
|
call writefile(['start of Xxx2', 'test', 'end of Xxx2'], 'Xxx2')
|
|
|
|
edit Xtest
|
|
e! Xxx2
|
|
bdel Xtest
|
|
e Xxx1
|
|
" write it, will unload it and give an error msg
|
|
call assert_fails('w', 'E203')
|
|
call assert_equal('Xxx2', bufname('%'))
|
|
edit Xtest
|
|
e! Xxx2
|
|
bwipe Xtest
|
|
" write it, will delete the buffer and give an error msg
|
|
call assert_fails('w', 'E203')
|
|
call assert_equal('Xxx1', bufname('%'))
|
|
au! BufWritePre
|
|
call delete('Xxx1')
|
|
call delete('Xxx2')
|
|
endfunc
|
|
|
|
" Test for BufUnload autocommand that unloads all the other buffers
|
|
func Test_bufunload_all()
|
|
call writefile(['Test file Xxx1'], 'Xxx1')"
|
|
call writefile(['Test file Xxx2'], 'Xxx2')"
|
|
|
|
let content = [
|
|
\ "func UnloadAllBufs()",
|
|
\ " let i = 1",
|
|
\ " while i <= bufnr('$')",
|
|
\ " if i != bufnr('%') && bufloaded(i)",
|
|
\ " exe i . 'bunload'",
|
|
\ " endif",
|
|
\ " let i += 1",
|
|
\ " endwhile",
|
|
\ "endfunc",
|
|
\ "au BufUnload * call UnloadAllBufs()",
|
|
\ "au VimLeave * call writefile(['Test Finished'], 'Xout')",
|
|
\ "edit Xxx1",
|
|
\ "split Xxx2",
|
|
\ "q"]
|
|
call writefile(content, 'Xtest')
|
|
|
|
call delete('Xout')
|
|
call system(v:progpath. ' -u NORC -i NONE -N -S Xtest')
|
|
call assert_true(filereadable('Xout'))
|
|
|
|
call delete('Xxx1')
|
|
call delete('Xxx2')
|
|
call delete('Xtest')
|
|
call delete('Xout')
|
|
endfunc
|
|
|
|
" Some tests for buffer-local autocommands
|
|
func Test_buflocal_autocmd()
|
|
let g:bname = ''
|
|
edit xx
|
|
au BufLeave <buffer> let g:bname = expand("%")
|
|
" here, autocommand for xx should trigger.
|
|
" but autocommand shall not apply to buffer named <buffer>.
|
|
edit somefile
|
|
call assert_equal('xx', g:bname)
|
|
let g:bname = ''
|
|
" here, autocommand shall be auto-deleted
|
|
bwipe xx
|
|
" autocmd should not trigger
|
|
edit xx
|
|
call assert_equal('', g:bname)
|
|
" autocmd should not trigger
|
|
edit somefile
|
|
call assert_equal('', g:bname)
|
|
enew
|
|
unlet g:bname
|
|
endfunc
|
|
|
|
" Test for "*Cmd" autocommands
|
|
func Test_Cmd_Autocmds()
|
|
call writefile(['start of Xxx', "\tabc2", 'end of Xxx'], 'Xxx')
|
|
|
|
enew!
|
|
au BufReadCmd XtestA 0r Xxx|$del
|
|
edit XtestA " will read text of Xxd instead
|
|
call assert_equal('start of Xxx', getline(1))
|
|
|
|
au BufWriteCmd XtestA call append(line("$"), "write")
|
|
write " will append a line to the file
|
|
call assert_equal('write', getline('$'))
|
|
call assert_fails('read XtestA', 'E484') " should not read anything
|
|
call assert_equal('write', getline(4))
|
|
|
|
" now we have:
|
|
" 1 start of Xxx
|
|
" 2 abc2
|
|
" 3 end of Xxx
|
|
" 4 write
|
|
|
|
au FileReadCmd XtestB '[r Xxx
|
|
2r XtestB " will read Xxx below line 2 instead
|
|
call assert_equal('start of Xxx', getline(3))
|
|
|
|
" now we have:
|
|
" 1 start of Xxx
|
|
" 2 abc2
|
|
" 3 start of Xxx
|
|
" 4 abc2
|
|
" 5 end of Xxx
|
|
" 6 end of Xxx
|
|
" 7 write
|
|
|
|
au FileWriteCmd XtestC '[,']copy $
|
|
normal 4GA1
|
|
4,5w XtestC " will copy lines 4 and 5 to the end
|
|
call assert_equal("\tabc21", getline(8))
|
|
call assert_fails('r XtestC', 'E484') " should not read anything
|
|
call assert_equal("end of Xxx", getline(9))
|
|
|
|
" now we have:
|
|
" 1 start of Xxx
|
|
" 2 abc2
|
|
" 3 start of Xxx
|
|
" 4 abc21
|
|
" 5 end of Xxx
|
|
" 6 end of Xxx
|
|
" 7 write
|
|
" 8 abc21
|
|
" 9 end of Xxx
|
|
|
|
let g:lines = []
|
|
au FileAppendCmd XtestD call extend(g:lines, getline(line("'["), line("']")))
|
|
w >>XtestD " will add lines to 'lines'
|
|
call assert_equal(9, len(g:lines))
|
|
call assert_fails('$r XtestD', 'E484') " should not read anything
|
|
call assert_equal(9, line('$'))
|
|
call assert_equal('end of Xxx', getline('$'))
|
|
|
|
au BufReadCmd XtestE 0r Xxx|$del
|
|
sp XtestE " split window with test.out
|
|
call assert_equal('end of Xxx', getline(3))
|
|
|
|
let g:lines = []
|
|
exe "normal 2Goasdf\<Esc>\<C-W>\<C-W>"
|
|
au BufWriteCmd XtestE call extend(g:lines, getline(0, '$'))
|
|
wall " will write other window to 'lines'
|
|
call assert_equal(4, len(g:lines), g:lines)
|
|
call assert_equal("\tasdf", g:lines[2])
|
|
|
|
au! BufReadCmd
|
|
au! BufWriteCmd
|
|
au! FileReadCmd
|
|
au! FileWriteCmd
|
|
au! FileAppendCmd
|
|
%bwipe!
|
|
call delete('Xxx')
|
|
enew!
|
|
endfunc
|
|
|
|
func SetChangeMarks(start, end)
|
|
exe a:start. 'mark ['
|
|
exe a:end. 'mark ]'
|
|
endfunc
|
|
|
|
" Verify the effects of autocmds on '[ and ']
|
|
func Test_change_mark_in_autocmds()
|
|
edit! Xtest
|
|
call feedkeys("ia\<CR>b\<CR>c\<CR>d\<C-g>u", 'xtn')
|
|
|
|
call SetChangeMarks(2, 3)
|
|
write
|
|
call assert_equal([1, 4], [line("'["), line("']")])
|
|
|
|
call SetChangeMarks(2, 3)
|
|
au BufWritePre * call assert_equal([1, 4], [line("'["), line("']")])
|
|
write
|
|
au! BufWritePre
|
|
|
|
if executable('cat')
|
|
write XtestFilter
|
|
write >> XtestFilter
|
|
|
|
call SetChangeMarks(2, 3)
|
|
" Marks are set to the entire range of the write
|
|
au FilterWritePre * call assert_equal([1, 4], [line("'["), line("']")])
|
|
" '[ is adjusted to just before the line that will receive the filtered
|
|
" data
|
|
au FilterReadPre * call assert_equal([4, 4], [line("'["), line("']")])
|
|
" The filtered data is read into the buffer, and the source lines are
|
|
" still present, so the range is after the source lines
|
|
au FilterReadPost * call assert_equal([5, 12], [line("'["), line("']")])
|
|
%!cat XtestFilter
|
|
" After the filtered data is read, the original lines are deleted
|
|
call assert_equal([1, 8], [line("'["), line("']")])
|
|
au! FilterWritePre,FilterReadPre,FilterReadPost
|
|
undo
|
|
|
|
call SetChangeMarks(1, 4)
|
|
au FilterWritePre * call assert_equal([2, 3], [line("'["), line("']")])
|
|
au FilterReadPre * call assert_equal([3, 3], [line("'["), line("']")])
|
|
au FilterReadPost * call assert_equal([4, 11], [line("'["), line("']")])
|
|
2,3!cat XtestFilter
|
|
call assert_equal([2, 9], [line("'["), line("']")])
|
|
au! FilterWritePre,FilterReadPre,FilterReadPost
|
|
undo
|
|
|
|
call delete('XtestFilter')
|
|
endif
|
|
|
|
call SetChangeMarks(1, 4)
|
|
au FileWritePre * call assert_equal([2, 3], [line("'["), line("']")])
|
|
2,3write Xtest2
|
|
au! FileWritePre
|
|
|
|
call SetChangeMarks(2, 3)
|
|
au FileAppendPre * call assert_equal([1, 4], [line("'["), line("']")])
|
|
write >> Xtest2
|
|
au! FileAppendPre
|
|
|
|
call SetChangeMarks(1, 4)
|
|
au FileAppendPre * call assert_equal([2, 3], [line("'["), line("']")])
|
|
2,3write >> Xtest2
|
|
au! FileAppendPre
|
|
|
|
call SetChangeMarks(1, 1)
|
|
au FileReadPre * call assert_equal([3, 1], [line("'["), line("']")])
|
|
au FileReadPost * call assert_equal([4, 11], [line("'["), line("']")])
|
|
3read Xtest2
|
|
au! FileReadPre,FileReadPost
|
|
undo
|
|
|
|
call SetChangeMarks(4, 4)
|
|
" When the line is 0, it's adjusted to 1
|
|
au FileReadPre * call assert_equal([1, 4], [line("'["), line("']")])
|
|
au FileReadPost * call assert_equal([1, 8], [line("'["), line("']")])
|
|
0read Xtest2
|
|
au! FileReadPre,FileReadPost
|
|
undo
|
|
|
|
call SetChangeMarks(4, 4)
|
|
" When the line is 0, it's adjusted to 1
|
|
au FileReadPre * call assert_equal([1, 4], [line("'["), line("']")])
|
|
au FileReadPost * call assert_equal([2, 9], [line("'["), line("']")])
|
|
1read Xtest2
|
|
au! FileReadPre,FileReadPost
|
|
undo
|
|
|
|
bwipe!
|
|
call delete('Xtest')
|
|
call delete('Xtest2')
|
|
endfunc
|
|
|
|
func Test_Filter_noshelltemp()
|
|
if !executable('cat')
|
|
return
|
|
endif
|
|
|
|
enew!
|
|
call setline(1, ['a', 'b', 'c', 'd'])
|
|
|
|
let shelltemp = &shelltemp
|
|
set shelltemp
|
|
|
|
let g:filter_au = 0
|
|
au FilterWritePre * let g:filter_au += 1
|
|
au FilterReadPre * let g:filter_au += 1
|
|
au FilterReadPost * let g:filter_au += 1
|
|
%!cat
|
|
call assert_equal(3, g:filter_au)
|
|
|
|
if has('filterpipe')
|
|
set noshelltemp
|
|
|
|
let g:filter_au = 0
|
|
au FilterWritePre * let g:filter_au += 1
|
|
au FilterReadPre * let g:filter_au += 1
|
|
au FilterReadPost * let g:filter_au += 1
|
|
%!cat
|
|
call assert_equal(0, g:filter_au)
|
|
endif
|
|
|
|
au! FilterWritePre,FilterReadPre,FilterReadPost
|
|
let &shelltemp = shelltemp
|
|
bwipe!
|
|
endfunc
|
|
|
|
func Test_TextYankPost()
|
|
enew!
|
|
call setline(1, ['foo'])
|
|
|
|
let g:event = []
|
|
au TextYankPost * let g:event = copy(v:event)
|
|
|
|
call assert_equal({}, v:event)
|
|
call assert_fails('let v:event = {}', 'E46:')
|
|
call assert_fails('let v:event.mykey = 0', 'E742:')
|
|
|
|
norm "ayiw
|
|
call assert_equal(
|
|
\{'regcontents': ['foo'], 'regname': 'a', 'operator': 'y', 'regtype': 'v'},
|
|
\g:event)
|
|
norm y_
|
|
call assert_equal(
|
|
\{'regcontents': ['foo'], 'regname': '', 'operator': 'y', 'regtype': 'V'},
|
|
\g:event)
|
|
call feedkeys("\<C-V>y", 'x')
|
|
call assert_equal(
|
|
\{'regcontents': ['f'], 'regname': '', 'operator': 'y', 'regtype': "\x161"},
|
|
\g:event)
|
|
norm "xciwbar
|
|
call assert_equal(
|
|
\{'regcontents': ['foo'], 'regname': 'x', 'operator': 'c', 'regtype': 'v'},
|
|
\g:event)
|
|
norm "bdiw
|
|
call assert_equal(
|
|
\{'regcontents': ['bar'], 'regname': 'b', 'operator': 'd', 'regtype': 'v'},
|
|
\g:event)
|
|
|
|
call assert_equal({}, v:event)
|
|
|
|
au! TextYankPost
|
|
unlet g:event
|
|
bwipe!
|
|
endfunc
|
|
|
|
func Test_nocatch_wipe_all_buffers()
|
|
" Real nasty autocommand: wipe all buffers on any event.
|
|
au * * bwipe *
|
|
call assert_fails('next x', 'E93')
|
|
bwipe
|
|
au!
|
|
endfunc
|
|
|
|
func Test_nocatch_wipe_dummy_buffer()
|
|
" Nasty autocommand: wipe buffer on any event.
|
|
au * x bwipe
|
|
call assert_fails('lv½ /x', 'E480')
|
|
au!
|
|
endfunc
|