local helpers = require("test.unit.helpers")(after_each) local itp = helpers.gen_itp(it) local to_cstr = helpers.to_cstr local get_str = helpers.ffi.string local eq = helpers.eq local NULL = helpers.NULL local globals = helpers.cimport("./src/nvim/globals.h") local buffer = helpers.cimport("./src/nvim/buffer.h") local stl = helpers.cimport("./src/nvim/statusline.h") describe('buffer functions', function() local buflist_new = function(file, flags) local c_file = to_cstr(file) return buffer.buflist_new(c_file, c_file, 1, flags) end local close_buffer = function(win, buf, action, abort_if_last, ignore_abort) return buffer.close_buffer(win, buf, action, abort_if_last, ignore_abort) end local path1 = 'test_file_path' local path2 = 'file_path_test' local path3 = 'path_test_file' setup(function() -- create the files io.open(path1, 'w'):close() io.open(path2, 'w'):close() io.open(path3, 'w'):close() end) teardown(function() os.remove(path1) os.remove(path2) os.remove(path3) end) describe('buf_valid', function() itp('should view NULL as an invalid buffer', function() eq(false, buffer.buf_valid(NULL)) end) itp('should view an open buffer as valid', function() local buf = buflist_new(path1, buffer.BLN_LISTED) eq(true, buffer.buf_valid(buf)) end) itp('should view a closed and hidden buffer as valid', function() local buf = buflist_new(path1, buffer.BLN_LISTED) close_buffer(NULL, buf, 0, 0, 0) eq(true, buffer.buf_valid(buf)) end) itp('should view a closed and unloaded buffer as valid', function() local buf = buflist_new(path1, buffer.BLN_LISTED) close_buffer(NULL, buf, buffer.DOBUF_UNLOAD, 0, 0) eq(true, buffer.buf_valid(buf)) end) itp('should view a closed and wiped buffer as invalid', function() local buf = buflist_new(path1, buffer.BLN_LISTED) close_buffer(NULL, buf, buffer.DOBUF_WIPE, 0, 0) eq(false, buffer.buf_valid(buf)) end) end) describe('buflist_findpat', function() local ALLOW_UNLISTED = 1 local ONLY_LISTED = 0 local buflist_findpat = function(pat, allow_unlisted) return buffer.buflist_findpat(to_cstr(pat), NULL, allow_unlisted, 0, 0) end itp('should find exact matches', function() local buf = buflist_new(path1, buffer.BLN_LISTED) eq(buf.handle, buflist_findpat(path1, ONLY_LISTED)) close_buffer(NULL, buf, buffer.DOBUF_WIPE, 0, 0) end) itp('should prefer to match the start of a file path', function() local buf1 = buflist_new(path1, buffer.BLN_LISTED) local buf2 = buflist_new(path2, buffer.BLN_LISTED) local buf3 = buflist_new(path3, buffer.BLN_LISTED) eq(buf1.handle, buflist_findpat("test", ONLY_LISTED)) eq(buf2.handle, buflist_findpat("file", ONLY_LISTED)) eq(buf3.handle, buflist_findpat("path", ONLY_LISTED)) close_buffer(NULL, buf1, buffer.DOBUF_WIPE, 0, 0) close_buffer(NULL, buf2, buffer.DOBUF_WIPE, 0, 0) close_buffer(NULL, buf3, buffer.DOBUF_WIPE, 0, 0) end) itp('should prefer to match the end of a file over the middle', function() --{ Given: Two buffers, where 'test' appears in both -- And: 'test' appears at the end of buf3 but in the middle of buf2 local buf2 = buflist_new(path2, buffer.BLN_LISTED) local buf3 = buflist_new(path3, buffer.BLN_LISTED) -- Then: buf2 is the buffer that is found eq(buf2.handle, buflist_findpat("test", ONLY_LISTED)) --} --{ When: We close buf2 close_buffer(NULL, buf2, buffer.DOBUF_WIPE, 0, 0) -- And: Open buf1, which has 'file' in the middle of its name local buf1 = buflist_new(path1, buffer.BLN_LISTED) -- Then: buf3 is found since 'file' appears at the end of the name eq(buf3.handle, buflist_findpat("file", ONLY_LISTED)) --} close_buffer(NULL, buf1, buffer.DOBUF_WIPE, 0, 0) close_buffer(NULL, buf3, buffer.DOBUF_WIPE, 0, 0) end) itp('should match a unique fragment of a file path', function() local buf1 = buflist_new(path1, buffer.BLN_LISTED) local buf2 = buflist_new(path2, buffer.BLN_LISTED) local buf3 = buflist_new(path3, buffer.BLN_LISTED) eq(buf3.handle, buflist_findpat("_test_", ONLY_LISTED)) close_buffer(NULL, buf1, buffer.DOBUF_WIPE, 0, 0) close_buffer(NULL, buf2, buffer.DOBUF_WIPE, 0, 0) close_buffer(NULL, buf3, buffer.DOBUF_WIPE, 0, 0) end) itp('should include / ignore unlisted buffers based on the flag.', function() --{ Given: A buffer local buf3 = buflist_new(path3, buffer.BLN_LISTED) -- Then: We should find the buffer when it is given a unique pattern eq(buf3.handle, buflist_findpat("_test_", ONLY_LISTED)) --} --{ When: We unlist the buffer close_buffer(NULL, buf3, buffer.DOBUF_DEL, 0, 0) -- Then: It should not find the buffer when searching only listed buffers eq(-1, buflist_findpat("_test_", ONLY_LISTED)) -- And: It should find the buffer when including unlisted buffers eq(buf3.handle, buflist_findpat("_test_", ALLOW_UNLISTED)) --} --{ When: We wipe the buffer close_buffer(NULL, buf3, buffer.DOBUF_WIPE, 0, 0) -- Then: It should not find the buffer at all eq(-1, buflist_findpat("_test_", ONLY_LISTED)) eq(-1, buflist_findpat("_test_", ALLOW_UNLISTED)) --} end) itp('should prefer listed buffers to unlisted buffers.', function() --{ Given: Two buffers that match a pattern local buf1 = buflist_new(path1, buffer.BLN_LISTED) local buf2 = buflist_new(path2, buffer.BLN_LISTED) -- Then: The first buffer is preferred when both are listed eq(buf1.handle, buflist_findpat("test", ONLY_LISTED)) --} --{ When: The first buffer is unlisted close_buffer(NULL, buf1, buffer.DOBUF_DEL, 0, 0) -- Then: The second buffer is preferred because -- unlisted buffers are not allowed eq(buf2.handle, buflist_findpat("test", ONLY_LISTED)) --} --{ When: We allow unlisted buffers -- Then: The second buffer is still preferred -- because listed buffers are preferred to unlisted eq(buf2.handle, buflist_findpat("test", ALLOW_UNLISTED)) --} --{ When: We unlist the second buffer close_buffer(NULL, buf2, buffer.DOBUF_DEL, 0, 0) -- Then: The first buffer is preferred again -- because buf1 matches better which takes precedence -- when both buffers have the same listing status. eq(buf1.handle, buflist_findpat("test", ALLOW_UNLISTED)) -- And: Neither buffer is returned when ignoring unlisted eq(-1, buflist_findpat("test", ONLY_LISTED)) --} close_buffer(NULL, buf1, buffer.DOBUF_WIPE, 0, 0) close_buffer(NULL, buf2, buffer.DOBUF_WIPE, 0, 0) end) end) describe('build_stl_str_hl', function() local buffer_byte_size = 100 local STL_INITIAL_ITEMS = 20 local output_buffer = '' -- This function builds the statusline -- -- @param arg Optional arguments are: -- .pat The statusline format string -- .fillchar The fill character used in the statusline -- .maximum_cell_count The number of cells available in the statusline local function build_stl_str_hl(arg) output_buffer = to_cstr(string.rep(" ", buffer_byte_size)) local pat = arg.pat or '' local fillchar = arg.fillchar or (' '):byte() local maximum_cell_count = arg.maximum_cell_count or buffer_byte_size return stl.build_stl_str_hl(globals.curwin, output_buffer, buffer_byte_size, to_cstr(pat), NULL, 0, fillchar, maximum_cell_count, NULL, NULL, NULL) end -- Use this function to simplify testing the comparison between -- the format string and the resulting statusline. -- -- @param description The description of what the test should be doing -- @param statusline_cell_count The number of cells available in the statusline -- @param input_stl The format string for the statusline -- @param expected_stl The expected result string for the statusline -- -- @param arg Options can be placed in an optional dictionary as the last parameter -- .expected_cell_count The expected number of cells build_stl_str_hl will return -- .expected_byte_length The expected byte length of the string (defaults to byte length of expected_stl) -- .file_name The name of the file to be tested (useful in %f type tests) -- .fillchar The character that will be used to fill any 'extra' space in the stl local function statusline_test (description, statusline_cell_count, input_stl, expected_stl, arg) -- arg is the optional parameter -- so we either fill in option with arg or an empty dictionary local option = arg or {} local fillchar = option.fillchar or (' '):byte() local expected_cell_count = option.expected_cell_count or statusline_cell_count local expected_byte_length = option.expected_byte_length or #expected_stl itp(description, function() if option.file_name then buffer.setfname(globals.curbuf, to_cstr(option.file_name), NULL, 1) else buffer.setfname(globals.curbuf, nil, NULL, 1) end local result_cell_count = build_stl_str_hl{pat=input_stl, maximum_cell_count=statusline_cell_count, fillchar=fillchar} eq(expected_stl, get_str(output_buffer, expected_byte_length)) eq(expected_cell_count, result_cell_count) end) end -- expression testing statusline_test('Should expand expression', 2, '%!expand(20+1)', '21') statusline_test('Should expand broken expression to itself', 11, '%!expand(20+1', 'expand(20+1') -- file name testing statusline_test('should print no file name', 10, '%f', '[No Name]', {expected_cell_count=9}) statusline_test('should print the relative file name', 30, '%f', 'test/unit/buffer_spec.lua', {file_name='test/unit/buffer_spec.lua', expected_cell_count=25}) statusline_test('should print the full file name', 40, '%F', '/test/unit/buffer_spec.lua', {file_name='/test/unit/buffer_spec.lua', expected_cell_count=26}) -- fillchar testing statusline_test('should handle `!` as a fillchar', 10, 'abcde%=', 'abcde!!!!!', {fillchar=('!'):byte()}) statusline_test('should handle `~` as a fillchar', 10, '%=abcde', '~~~~~abcde', {fillchar=('~'):byte()}) statusline_test('should put fillchar `!` in between text', 10, 'abc%=def', 'abc!!!!def', {fillchar=('!'):byte()}) statusline_test('should put fillchar `~` in between text', 10, 'abc%=def', 'abc~~~~def', {fillchar=('~'):byte()}) statusline_test('should put fillchar `━` in between text', 10, 'abc%=def', 'abc━━━━def', {fillchar=0x2501}) statusline_test('should handle zero-fillchar as a space', 10, 'abcde%=', 'abcde ', {fillchar=0}) statusline_test('should print the tail file name', 80, '%t', 'buffer_spec.lua', {file_name='test/unit/buffer_spec.lua', expected_cell_count=15}) -- standard text testing statusline_test('should copy plain text', 80, 'this is a test', 'this is a test', {expected_cell_count=14}) -- line number testing statusline_test('should print the buffer number', 80, '%n', '1', {expected_cell_count=1}) statusline_test('should print the current line number in the buffer', 80, '%l', '0', {expected_cell_count=1}) statusline_test('should print the number of lines in the buffer', 80, '%L', '1', {expected_cell_count=1}) -- truncation testing statusline_test('should truncate when standard text pattern is too long', 10, '0123456789abcde', '<6789abcde') statusline_test('should truncate when using =', 10, 'abcdef%=ghijkl', 'abcdef') statusline_test('should truncate at `<` with one `=`, test 3', 10, 'abc%