mirror of
https://github.com/neovim/neovim
synced 2025-07-16 01:01:49 +00:00
feat(lsp): support function for client root_dir (#31630)
If root_dir is a function it is evaluated when the client is created to determine the root directory. This enables dynamically determining the root directory based on e.g. project or directory structure (example: finding a parent Cargo.toml file that contains "[workspace]" in a Rust project).
This commit is contained in:
@ -683,6 +683,13 @@ Lua module: vim.lsp *lsp-core*
|
|||||||
the LSP server will base its workspaceFolders,
|
the LSP server will base its workspaceFolders,
|
||||||
rootUri, and rootPath on initialization. Unused if
|
rootUri, and rootPath on initialization. Unused if
|
||||||
`root_dir` is provided.
|
`root_dir` is provided.
|
||||||
|
• {root_dir}? (`string|fun(cb:fun(string))`) Directory where the
|
||||||
|
LSP server will base its workspaceFolders, rootUri,
|
||||||
|
and rootPath on initialization. If a function, it
|
||||||
|
accepts a single callback argument which must be
|
||||||
|
called with the value of root_dir to use. The LSP
|
||||||
|
server will not be started until the callback is
|
||||||
|
called.
|
||||||
• {reuse_client}? (`fun(client: vim.lsp.Client, config: vim.lsp.ClientConfig): boolean`)
|
• {reuse_client}? (`fun(client: vim.lsp.Client, config: vim.lsp.ClientConfig): boolean`)
|
||||||
Predicate used to decide if a client should be
|
Predicate used to decide if a client should be
|
||||||
re-used. Used on all running clients. The default
|
re-used. Used on all running clients. The default
|
||||||
|
@ -334,6 +334,11 @@ end
|
|||||||
--- rootUri, and rootPath on initialization. Unused if `root_dir` is provided.
|
--- rootUri, and rootPath on initialization. Unused if `root_dir` is provided.
|
||||||
--- @field root_markers? string[]
|
--- @field root_markers? string[]
|
||||||
---
|
---
|
||||||
|
--- Directory where the LSP server will base its workspaceFolders, rootUri, and rootPath on
|
||||||
|
--- initialization. If a function, it accepts a single callback argument which must be called with
|
||||||
|
--- the value of root_dir to use. The LSP server will not be started until the callback is called.
|
||||||
|
--- @field root_dir? string|fun(cb:fun(string))
|
||||||
|
---
|
||||||
--- Predicate used to decide if a client should be re-used. Used on all
|
--- Predicate used to decide if a client should be re-used. Used on all
|
||||||
--- running clients. The default implementation re-uses a client if name and
|
--- running clients. The default implementation re-uses a client if name and
|
||||||
--- root_dir matches.
|
--- root_dir matches.
|
||||||
@ -499,6 +504,15 @@ local function lsp_enable_callback(bufnr)
|
|||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
|
|
||||||
|
--- @param config vim.lsp.Config
|
||||||
|
local function start(config)
|
||||||
|
return vim.lsp.start(config, {
|
||||||
|
bufnr = bufnr,
|
||||||
|
reuse_client = config.reuse_client,
|
||||||
|
_root_markers = config.root_markers,
|
||||||
|
})
|
||||||
|
end
|
||||||
|
|
||||||
for name in vim.spairs(lsp._enabled_configs) do
|
for name in vim.spairs(lsp._enabled_configs) do
|
||||||
local config = lsp._resolve_config(name)
|
local config = lsp._resolve_config(name)
|
||||||
|
|
||||||
@ -507,11 +521,14 @@ local function lsp_enable_callback(bufnr)
|
|||||||
-- do not propagate back to the enabled configs.
|
-- do not propagate back to the enabled configs.
|
||||||
config = vim.deepcopy(config)
|
config = vim.deepcopy(config)
|
||||||
|
|
||||||
vim.lsp.start(config, {
|
if type(config.root_dir) == 'function' then
|
||||||
bufnr = bufnr,
|
config.root_dir(function(root_dir)
|
||||||
reuse_client = config.reuse_client,
|
config.root_dir = root_dir
|
||||||
_root_markers = config.root_markers,
|
start(config)
|
||||||
})
|
end)
|
||||||
|
else
|
||||||
|
start(config)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -6245,5 +6245,38 @@ describe('LSP', function()
|
|||||||
end)
|
end)
|
||||||
)
|
)
|
||||||
end)
|
end)
|
||||||
|
|
||||||
|
it('supports a function for root_dir', function()
|
||||||
|
exec_lua(create_server_definition)
|
||||||
|
|
||||||
|
local tmp1 = t.tmpname(true)
|
||||||
|
|
||||||
|
eq(
|
||||||
|
'some_dir',
|
||||||
|
exec_lua(function()
|
||||||
|
local server = _G._create_server({
|
||||||
|
handlers = {
|
||||||
|
initialize = function(_, _, callback)
|
||||||
|
callback(nil, { capabilities = {} })
|
||||||
|
end,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
vim.lsp.config('foo', {
|
||||||
|
cmd = server.cmd,
|
||||||
|
filetypes = { 'foo' },
|
||||||
|
root_dir = function(cb)
|
||||||
|
cb('some_dir')
|
||||||
|
end,
|
||||||
|
})
|
||||||
|
vim.lsp.enable('foo')
|
||||||
|
|
||||||
|
vim.cmd.edit(assert(tmp1))
|
||||||
|
vim.bo.filetype = 'foo'
|
||||||
|
|
||||||
|
return vim.lsp.get_clients({ bufnr = vim.api.nvim_get_current_buf() })[1].root_dir
|
||||||
|
end)
|
||||||
|
)
|
||||||
|
end)
|
||||||
end)
|
end)
|
||||||
end)
|
end)
|
||||||
|
Reference in New Issue
Block a user