mirror of
				https://github.com/neovim/neovim.git
				synced 2025-10-26 12:27:24 +00:00 
			
		
		
		
	 ff470bb853
			
		
	
	ff470bb853
	
	
	
		
			
			It is otherwise impossible to determine which test failed sanitizer/valgrind check. test/functional/helpers.lua module return was changed so that tests which do not provide after_each function to get new check will automatically fail.
		
			
				
	
	
		
			296 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			296 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
| -- Sanity checks for buffer_* API calls via msgpack-rpc
 | |
| local helpers = require('test.functional.helpers')(after_each)
 | |
| local clear, nvim, buffer = helpers.clear, helpers.nvim, helpers.buffer
 | |
| local curbuf, curwin, eq = helpers.curbuf, helpers.curwin, helpers.eq
 | |
| local curbufmeths, ok = helpers.curbufmeths, helpers.ok
 | |
| local funcs = helpers.funcs
 | |
| 
 | |
| describe('buffer_* functions', function()
 | |
|   before_each(clear)
 | |
| 
 | |
|   describe('line_count, insert and del_line', function()
 | |
|     it('works', function()
 | |
|       eq(1, curbuf('line_count'))
 | |
|       curbuf('insert', -1, {'line'})
 | |
|       eq(2, curbuf('line_count'))
 | |
|       curbuf('insert', -1, {'line'})
 | |
|       eq(3, curbuf('line_count'))
 | |
|       curbuf('del_line', -1)
 | |
|       eq(2, curbuf('line_count'))
 | |
|       curbuf('del_line', -1)
 | |
|       curbuf('del_line', -1)
 | |
|       -- There's always at least one line
 | |
|       eq(1, curbuf('line_count'))
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
| 
 | |
|   describe('{get,set,del}_line', function()
 | |
|     it('works', function()
 | |
|       eq('', curbuf('get_line', 0))
 | |
|       curbuf('set_line', 0, 'line1')
 | |
|       eq('line1', curbuf('get_line', 0))
 | |
|       curbuf('set_line', 0, 'line2')
 | |
|       eq('line2', curbuf('get_line', 0))
 | |
|       curbuf('del_line', 0)
 | |
|       eq('', curbuf('get_line', 0))
 | |
|     end)
 | |
| 
 | |
|     it('get_line: out-of-bounds is an error', function()
 | |
|       curbuf('set_line', 0, 'line1.a')
 | |
|       eq(1, curbuf('line_count')) -- sanity
 | |
|       eq(false, pcall(curbuf, 'get_line', 1))
 | |
|       eq(false, pcall(curbuf, 'get_line', -2))
 | |
|     end)
 | |
| 
 | |
|     it('set_line, del_line: out-of-bounds is an error', function()
 | |
|       curbuf('set_line', 0, 'line1.a')
 | |
|       eq(false, pcall(curbuf, 'set_line', 1, 'line1.b'))
 | |
|       eq(false, pcall(curbuf, 'set_line', -2, 'line1.b'))
 | |
|       eq(false, pcall(curbuf, 'del_line', 2))
 | |
|       eq(false, pcall(curbuf, 'del_line', -3))
 | |
|     end)
 | |
| 
 | |
|     it('can handle NULs', function()
 | |
|       curbuf('set_line', 0, 'ab\0cd')
 | |
|       eq('ab\0cd', curbuf('get_line', 0))
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
| 
 | |
|   describe('{get,set}_line_slice', function()
 | |
|     it('get_line_slice: out-of-bounds returns empty array', function()
 | |
|       curbuf('set_line_slice', 0, 0, true, true, {'a', 'b', 'c'})
 | |
|       eq({'a', 'b', 'c'}, curbuf('get_line_slice', 0, 2, true, true)) --sanity
 | |
| 
 | |
|       eq({}, curbuf('get_line_slice', 2, 3, false, true))
 | |
|       eq({}, curbuf('get_line_slice', 3, 9, true, true))
 | |
|       eq({}, curbuf('get_line_slice', 3, -1, true, true))
 | |
|       eq({}, curbuf('get_line_slice', -3, -4, false, true))
 | |
|       eq({}, curbuf('get_line_slice', -4, -5, true, true))
 | |
|     end)
 | |
| 
 | |
|     it('set_line_slice: out-of-bounds extends past end', function()
 | |
|       curbuf('set_line_slice', 0, 0, true, true, {'a', 'b', 'c'})
 | |
|       eq({'a', 'b', 'c'}, curbuf('get_line_slice', 0, 2, true, true)) --sanity
 | |
| 
 | |
|       eq({'c'}, curbuf('get_line_slice', -1, 4, true, true))
 | |
|       eq({'a', 'b', 'c'}, curbuf('get_line_slice', 0, 5, true, true))
 | |
|       curbuf('set_line_slice', 4, 5, true, true, {'d'})
 | |
|       eq({'a', 'b', 'c', 'd'}, curbuf('get_line_slice', 0, 5, true, true))
 | |
|       curbuf('set_line_slice', -4, -5, true, true, {'e'})
 | |
|       eq({'e', 'a', 'b', 'c', 'd'}, curbuf('get_line_slice', 0, 5, true, true))
 | |
|     end)
 | |
| 
 | |
|     it('works', function()
 | |
|       eq({''}, curbuf('get_line_slice', 0, -1, true, true))
 | |
|       -- Replace buffer
 | |
|       curbuf('set_line_slice', 0, -1, true, true, {'a', 'b', 'c'})
 | |
|       eq({'a', 'b', 'c'}, curbuf('get_line_slice', 0, -1, true, true))
 | |
|       eq({'b', 'c'}, curbuf('get_line_slice', 1, -1, true, true))
 | |
|       eq({'b'}, curbuf('get_line_slice', 1, 2, true, false))
 | |
|       eq({}, curbuf('get_line_slice', 1, 1, true, false))
 | |
|       eq({'a', 'b'}, curbuf('get_line_slice', 0, -1, true, false))
 | |
|       eq({'b'}, curbuf('get_line_slice', 1, -1, true, false))
 | |
|       eq({'b', 'c'}, curbuf('get_line_slice', -2, -1, true, true))
 | |
|       curbuf('set_line_slice', 1, 2, true, false, {'a', 'b', 'c'})
 | |
|       eq({'a', 'a', 'b', 'c', 'c'}, curbuf('get_line_slice', 0, -1, true, true))
 | |
|       curbuf('set_line_slice', -1, -1, true, true, {'a', 'b', 'c'})
 | |
|       eq({'a', 'a', 'b', 'c', 'a', 'b', 'c'},
 | |
|         curbuf('get_line_slice', 0, -1, true, true))
 | |
|       curbuf('set_line_slice', 0, -3, true, false, {})
 | |
|       eq({'a', 'b', 'c'}, curbuf('get_line_slice', 0, -1, true, true))
 | |
|       curbuf('set_line_slice', 0, -1, true, true, {})
 | |
|       eq({''}, curbuf('get_line_slice', 0, -1, true, true))
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
|   describe('{get,set}_lines', function()
 | |
|     local get_lines, set_lines = curbufmeths.get_lines, curbufmeths.set_lines
 | |
|     local line_count = curbufmeths.line_count
 | |
| 
 | |
|     it('has correct line_count when inserting and deleting', function()
 | |
|       eq(1, line_count())
 | |
|       set_lines(-1, -1, true, {'line'})
 | |
|       eq(2, line_count())
 | |
|       set_lines(-1, -1, true, {'line'})
 | |
|       eq(3, line_count())
 | |
|       set_lines(-2, -1, true, {})
 | |
|       eq(2, line_count())
 | |
|       set_lines(-2, -1, true, {})
 | |
|       set_lines(-2, -1, true, {})
 | |
|       -- There's always at least one line
 | |
|       eq(1, line_count())
 | |
|     end)
 | |
| 
 | |
|     it('can get, set and delete a single line', function()
 | |
|       eq({''}, get_lines(0, 1, true))
 | |
|       set_lines(0, 1, true, {'line1'})
 | |
|       eq({'line1'}, get_lines(0, 1, true))
 | |
|       set_lines(0, 1, true, {'line2'})
 | |
|       eq({'line2'}, get_lines(0, 1, true))
 | |
|       set_lines(0, 1, true, {})
 | |
|       eq({''}, get_lines(0, 1, true))
 | |
|     end)
 | |
| 
 | |
|     it('can get a single line with strict indexing', function()
 | |
|       set_lines(0, 1, true, {'line1.a'})
 | |
|       eq(1, line_count()) -- sanity
 | |
|       eq(false, pcall(get_lines, 1, 2, true))
 | |
|       eq(false, pcall(get_lines, -3, -2, true))
 | |
|     end)
 | |
| 
 | |
|     it('can get a single line with non-strict indexing', function()
 | |
|       set_lines(0, 1, true, {'line1.a'})
 | |
|       eq(1, line_count()) -- sanity
 | |
|       eq({}, get_lines(1, 2, false))
 | |
|       eq({}, get_lines(-3, -2, false))
 | |
|     end)
 | |
| 
 | |
|     it('can set and delete a single line with strict indexing', function()
 | |
|       set_lines(0, 1, true, {'line1.a'})
 | |
|       eq(false, pcall(set_lines, 1, 2, true, {'line1.b'}))
 | |
|       eq(false, pcall(set_lines, -3, -2, true, {'line1.c'}))
 | |
|       eq({'line1.a'}, get_lines(0, -1, true))
 | |
|       eq(false, pcall(set_lines, 1, 2, true, {}))
 | |
|       eq(false, pcall(set_lines, -3, -2, true, {}))
 | |
|       eq({'line1.a'}, get_lines(0, -1, true))
 | |
|     end)
 | |
| 
 | |
|     it('can set and delete a single line with non-strict indexing', function()
 | |
|       set_lines(0, 1, true, {'line1.a'})
 | |
|       set_lines(1, 2, false, {'line1.b'})
 | |
|       set_lines(-4, -3, false, {'line1.c'})
 | |
|       eq({'line1.c', 'line1.a', 'line1.b'}, get_lines(0, -1, true))
 | |
|       set_lines(3, 4, false, {})
 | |
|       set_lines(-5, -4, false, {})
 | |
|       eq({'line1.c', 'line1.a', 'line1.b'}, get_lines(0, -1, true))
 | |
|     end)
 | |
| 
 | |
|     it('can handle NULs', function()
 | |
|       set_lines(0, 1, true, {'ab\0cd'})
 | |
|       eq({'ab\0cd'}, get_lines(0, -1, true))
 | |
|     end)
 | |
| 
 | |
|     it('works with multiple lines', function()
 | |
|       eq({''}, get_lines(0, -1, true))
 | |
|       -- Replace buffer
 | |
|       for _, mode in pairs({false, true}) do
 | |
|         set_lines(0, -1, mode, {'a', 'b', 'c'})
 | |
|         eq({'a', 'b', 'c'}, get_lines(0, -1, mode))
 | |
|         eq({'b', 'c'}, get_lines(1, -1, mode))
 | |
|         eq({'b'}, get_lines(1, 2, mode))
 | |
|         eq({}, get_lines(1, 1, mode))
 | |
|         eq({'a', 'b'}, get_lines(0, -2, mode))
 | |
|         eq({'b'}, get_lines(1, -2, mode))
 | |
|         eq({'b', 'c'}, get_lines(-3, -1, mode))
 | |
|         set_lines(1, 2, mode, {'a', 'b', 'c'})
 | |
|         eq({'a', 'a', 'b', 'c', 'c'}, get_lines(0, -1, mode))
 | |
|         set_lines(-2, -1, mode, {'a', 'b', 'c'})
 | |
|         eq({'a', 'a', 'b', 'c', 'a', 'b', 'c'},
 | |
|           get_lines(0, -1, mode))
 | |
|         set_lines(0, -4, mode, {})
 | |
|         eq({'a', 'b', 'c'}, get_lines(0, -1, mode))
 | |
|         set_lines(0, -1, mode, {})
 | |
|         eq({''}, get_lines(0, -1, mode))
 | |
|       end
 | |
|     end)
 | |
| 
 | |
|     it('can get line ranges with non-strict indexing', function()
 | |
|       set_lines(0, -1, true, {'a', 'b', 'c'})
 | |
|       eq({'a', 'b', 'c'}, get_lines(0, -1, true)) --sanity
 | |
| 
 | |
|       eq({}, get_lines(3, 4, false))
 | |
|       eq({}, get_lines(3, 10, false))
 | |
|       eq({}, get_lines(-5, -5, false))
 | |
|       eq({}, get_lines(3, -1, false))
 | |
|       eq({}, get_lines(-3, -4, false))
 | |
|     end)
 | |
| 
 | |
|     it('can get line ranges with strict indexing', function()
 | |
|       set_lines(0, -1, true, {'a', 'b', 'c'})
 | |
|       eq({'a', 'b', 'c'}, get_lines(0, -1, true)) --sanity
 | |
| 
 | |
|       eq(false, pcall(get_lines, 3, 4, true))
 | |
|       eq(false, pcall(get_lines, 3, 10, true))
 | |
|       eq(false, pcall(get_lines, -5, -5, true))
 | |
|       -- empty or inverted ranges are not errors
 | |
|       eq({}, get_lines(3, -1, true))
 | |
|       eq({}, get_lines(-3, -4, true))
 | |
|     end)
 | |
| 
 | |
|     it('set_line_slice: out-of-bounds can extend past end', function()
 | |
|       set_lines(0, -1, true, {'a', 'b', 'c'})
 | |
|       eq({'a', 'b', 'c'}, get_lines(0, -1, true)) --sanity
 | |
| 
 | |
|       eq({'c'}, get_lines(-2, 5, false))
 | |
|       eq({'a', 'b', 'c'}, get_lines(0, 6, false))
 | |
|       eq(false, pcall(set_lines, 4, 6, true, {'d'}))
 | |
|       set_lines(4, 6, false, {'d'})
 | |
|       eq({'a', 'b', 'c', 'd'}, get_lines(0, -1, true))
 | |
|       eq(false, pcall(set_lines, -6, -6, true, {'e'}))
 | |
|       set_lines(-6, -6, false, {'e'})
 | |
|       eq({'e', 'a', 'b', 'c', 'd'}, get_lines(0, -1, true))
 | |
|     end)
 | |
| 
 | |
|   end)
 | |
| 
 | |
|   describe('{get,set,del}_var', function()
 | |
|     it('works', function()
 | |
|       curbuf('set_var', 'lua', {1, 2, {['3'] = 1}})
 | |
|       eq({1, 2, {['3'] = 1}}, curbuf('get_var', 'lua'))
 | |
|       eq({1, 2, {['3'] = 1}}, nvim('eval', 'b:lua'))
 | |
|       eq(1, funcs.exists('b:lua'))
 | |
|       curbufmeths.del_var('lua')
 | |
|       eq(0, funcs.exists('b:lua'))
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
|   describe('{get,set}_option', function()
 | |
|     it('works', function()
 | |
|       eq(8, curbuf('get_option', 'shiftwidth'))
 | |
|       curbuf('set_option', 'shiftwidth', 4)
 | |
|       eq(4, curbuf('get_option', 'shiftwidth'))
 | |
|       -- global-local option
 | |
|       curbuf('set_option', 'define', 'test')
 | |
|       eq('test', curbuf('get_option', 'define'))
 | |
|       -- Doesn't change the global value
 | |
|       eq([[^\s*#\s*define]], nvim('get_option', 'define'))
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
|   describe('{get,set}_name', function()
 | |
|     it('works', function()
 | |
|       nvim('command', 'new')
 | |
|       eq('', curbuf('get_name'))
 | |
|       local new_name = nvim('eval', 'resolve(tempname())')
 | |
|       curbuf('set_name', new_name)
 | |
|       eq(new_name, curbuf('get_name'))
 | |
|       nvim('command', 'w!')
 | |
|       local f = io.open(new_name)
 | |
|       ok(f ~= nil)
 | |
|       f:close()
 | |
|       os.remove(new_name)
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
|   describe('is_valid', function()
 | |
|     it('works', function()
 | |
|       nvim('command', 'new')
 | |
|       local b = nvim('get_current_buffer')
 | |
|       ok(buffer('is_valid', b))
 | |
|       nvim('command', 'bw!')
 | |
|       ok(not buffer('is_valid', b))
 | |
|     end)
 | |
|   end)
 | |
| 
 | |
|   describe('get_mark', function()
 | |
|     it('works', function()
 | |
|       curbuf('insert', -1, {'a', 'bit of', 'text'})
 | |
|       curwin('set_cursor', {3, 4})
 | |
|       nvim('command', 'mark V')
 | |
|       eq({3, 0}, curbuf('get_mark', 'V'))
 | |
|     end)
 | |
|   end)
 | |
| end)
 |