mirror of
				https://github.com/neovim/neovim.git
				synced 2025-11-04 01:34:25 +00:00 
			
		
		
		
	Problem: - `:lua vim.env.<Tab>` does not show completion of environment variables - Meanwhile, `:let $<Tab>` does show completion of environment variables Solution: - Fix it
		
			
				
	
	
		
			376 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			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)
 |