Files
neovim/test/functional/lua/command_line_completion_spec.lua
Phạm Bình An 6b00c9acfd fix(lua): no omni/cmdline completion for vim.env (#33044)
Problem:
- `:lua vim.env.<Tab>` does not show completion of environment variables
- Meanwhile, `:let $<Tab>` does show completion of environment variables

Solution:
- Fix it
2025-03-26 07:35:12 +08:00

376 lines
9.5 KiB
Lua

local t = require('test.testutil')
local n = require('test.functional.testnvim')()
local clear = n.clear
local eq = t.eq
local exec_lua = n.exec_lua
--- @return { [1]: string[], [2]: integer }
local get_completions = function(input, env)
return exec_lua('return { vim._expand_pat(...) }', input, env)
end
--- @return { [1]: string[], [2]: integer }
local get_compl_parts = function(parts)
return exec_lua('return { vim._expand_pat_get_parts(...) }', parts)
end
before_each(clear)
describe('nlua_expand_pat', function()
it('completes exact matches', function()
eq({ { 'exact' }, 0 }, get_completions('exact', { exact = true }))
end)
it('returns empty table when nothing matches', function()
eq({ {}, 0 }, get_completions('foo', { bar = true }))
-- can access non-exist field
for _, m in ipairs {
'vim.',
'vim.lsp.',
'vim.treesitter.',
'vim.deepcopy.',
'vim.fn.',
'vim.api.',
'vim.o.',
'vim.b.',
} do
eq({ {}, m:len() }, get_completions(m .. 'foo'))
eq({ {}, 0 }, get_completions(m .. 'foo.'))
eq({ {}, 0 }, get_completions(m .. 'foo.bar'))
eq({ {}, 0 }, get_completions(m .. 'foo.bar.'))
end
end)
it('returns nice completions with function call prefix', function()
eq({ { 'FOO' }, 6 }, get_completions('print(F', { FOO = true, bawr = true }))
end)
it('returns keys for nested dicts', function()
eq(
{ {
'nvim_buf_set_lines',
}, 8 },
get_completions('vim.api.nvim_buf_', {
vim = {
api = {
nvim_buf_set_lines = true,
nvim_win_doesnt_match = true,
},
other_key = true,
},
})
)
end)
it('with colons', function()
eq(
{ {
'bawr',
'baz',
}, 8 },
get_completions('MyClass:b', {
MyClass = {
baz = true,
bawr = true,
foo = false,
},
})
)
end)
it('returns keys after string key', function()
eq(
{ {
'nvim_buf_set_lines',
}, 11 },
get_completions('vim["api"].nvim_buf_', {
vim = {
api = {
nvim_buf_set_lines = true,
nvim_win_doesnt_match = true,
},
other_key = true,
},
})
)
eq(
{ {
'nvim_buf_set_lines',
}, 21 },
get_completions('vim["nested"]["api"].nvim_buf_', {
vim = {
nested = {
api = {
nvim_buf_set_lines = true,
nvim_win_doesnt_match = true,
},
},
other_key = true,
},
})
)
end)
it('with lazy submodules of "vim" global', function()
eq({ { 'inspect', 'inspect_pos' }, 4 }, get_completions('vim.inspec'))
eq({ { 'treesitter' }, 4 }, get_completions('vim.treesi'))
eq({ { 'dev' }, 15 }, get_completions('vim.treesitter.de'))
eq({ { 'edit_query' }, 19 }, get_completions('vim.treesitter.dev.edit_'))
eq({ { 'set' }, 11 }, get_completions('vim.keymap.se'))
end)
it('include keys in mt.__index and ._submodules', function()
eq(
{ { 'bar1', 'bar2', 'bar3' }, 4 },
exec_lua(function() -- metatable cannot be serialized
return {
vim._expand_pat('foo.', {
foo = setmetatable(
{ bar1 = true, _submodules = { bar2 = true } },
{ __index = { bar3 = true } }
),
}),
}
end)
)
end)
it('excludes private fields after "."', function()
eq(
{ { 'bar' }, 4 },
get_completions('foo.', {
foo = {
_bar = true,
bar = true,
},
})
)
end)
it('includes private fields after "._"', function()
eq(
{ { '_bar' }, 4 },
get_completions('foo._', {
foo = {
_bar = true,
bar = true,
},
})
)
end)
it('can interpolate globals', function()
eq(
{ {
'nvim_buf_set_lines',
}, 12 },
get_completions('vim[MY_VAR].nvim_buf_', {
MY_VAR = 'api',
vim = {
api = {
nvim_buf_set_lines = true,
nvim_win_doesnt_match = true,
},
other_key = true,
},
})
)
end)
describe('vim.fn', function()
it('simple completion', function()
local actual = get_completions('vim.fn.did')
local expected = {
{ 'did_filetype' },
#'vim.fn.',
}
eq(expected, actual)
end)
it('does not suggest "#" items', function()
exec_lua [[
-- ensure remote#host#... functions exist
vim.cmd [=[
runtime! autoload/remote/host.vim
]=]
-- make a dummy call to ensure vim.fn contains an entry: remote#host#...
vim.fn['remote#host#IsRunning']('python3')
]]
local actual = get_completions('vim.fn.remo')
local expected = {
{ 'remove' }, -- there should be no completion "remote#host#..."
#'vim.fn.',
}
eq(expected, actual)
end)
end)
describe('completes', function()
it('vim.v', function()
local actual = get_completions('vim.v.t_')
local expected = {
{ 't_blob', 't_bool', 't_dict', 't_float', 't_func', 't_list', 't_number', 't_string' },
#'vim.v.',
}
eq(expected, actual)
end)
it('vim.g', function()
exec_lua [[
vim.cmd [=[
let g:nlua_foo = 'completion'
let g:nlua_foo_bar = 'completion'
let g:nlua_foo#bar = 'nocompletion' " should be excluded from lua completion
]=]
]]
local actual = get_completions('vim.g.nlua')
local expected = {
{ 'nlua_foo', 'nlua_foo_bar' },
#'vim.g.',
}
eq(expected, actual)
end)
it('vim.b', function()
exec_lua [[
vim.b.nlua_foo_buf = 'bar'
vim.b.some_other_vars = 'bar'
]]
local actual = get_completions('vim.b.nlua')
local expected = {
{ 'nlua_foo_buf' },
#'vim.b.',
}
eq(expected, actual)
end)
it('vim.w', function()
exec_lua [[
vim.w.nlua_win_var = 42
]]
local actual = get_completions('vim.w.nlua')
local expected = {
{ 'nlua_win_var' },
#'vim.w.',
}
eq(expected, actual)
end)
it('vim.t', function()
exec_lua [[
vim.t.nlua_tab_var = 42
]]
local actual = get_completions('vim.t.')
local expected = {
{ 'nlua_tab_var' },
#'vim.t.',
}
eq(expected, actual)
end)
it('vim.env', function()
exec_lua [[
vim.env.NLUA_ENV_VAR = 'foo'
]]
local actual = get_completions('vim.env.NLUA')
local expected = {
{ 'NLUA_ENV_VAR' },
#'vim.env.',
}
eq(expected, actual)
end)
end)
describe('completes', function()
-- for { vim.o, vim.go, vim.opt, vim.opt_local, vim.opt_global }
local test_opt = function(accessor)
do
local actual = get_completions(accessor .. '.file')
local expected = {
'fileencoding',
'fileencodings',
'fileformat',
'fileformats',
'fileignorecase',
'filetype',
}
eq({ expected, #accessor + 1 }, actual, accessor .. '.file')
end
do
local actual = get_completions(accessor .. '.winh')
local expected = {
'winheight',
'winhighlight',
}
eq({ expected, #accessor + 1 }, actual, accessor .. '.winh')
end
end
test_opt('vim.o')
test_opt('vim.go')
test_opt('vim.opt')
test_opt('vim.opt_local')
test_opt('vim.opt_global')
it('vim.o, suggesting all known options', function()
local completions = get_completions('vim.o.')[1] ---@type string[]
eq(
exec_lua [[
return vim.tbl_count(vim.api.nvim_get_all_options_info())
]],
#completions
)
end)
it('vim.bo', function()
do
local actual = get_completions('vim.bo.file')
local compls = {
-- should contain buffer options only
'fileencoding',
'fileformat',
'filetype',
}
eq({ compls, #'vim.bo.' }, actual)
end
do
local actual = get_completions('vim.bo.winh')
local compls = {}
eq({ compls, #'vim.bo.' }, actual)
end
end)
it('vim.wo', function()
do
local actual = get_completions('vim.wo.file')
local compls = {}
eq({ compls, #'vim.wo.' }, actual)
end
do
local actual = get_completions('vim.wo.winh')
-- should contain window options only
local compls = { 'winhighlight' }
eq({ compls, #'vim.wo.' }, actual)
end
end)
end)
it('returns everything if input is empty', function()
eq({ { 'other', 'vim' }, 0 }, get_completions('', { vim = true, other = true }))
end)
it('get_parts', function()
eq({ {}, 1 }, get_compl_parts('vim'))
eq({ { 'vim' }, 5 }, get_compl_parts('vim.ap'))
eq({ { 'vim', 'api' }, 9 }, get_compl_parts('vim.api.nvim_buf'))
eq({ { 'vim', 'api' }, 9 }, get_compl_parts('vim:api.nvim_buf'))
eq({ { 'vim', 'api' }, 12 }, get_compl_parts("vim['api'].nvim_buf"))
eq({ { 'vim', 'api' }, 12 }, get_compl_parts('vim["api"].nvim_buf'))
eq({ { 'vim', 'nested', 'api' }, 22 }, get_compl_parts('vim["nested"]["api"].nvim_buf'))
eq({ { 'vim', 'nested', 'api' }, 25 }, get_compl_parts('vim[ "nested" ]["api"].nvim_buf'))
eq({ { 'vim', { 'NESTED' }, 'api' }, 20 }, get_compl_parts('vim[NESTED]["api"].nvim_buf'))
end)
end)