mirror of
https://github.com/neovim/neovim
synced 2025-07-16 01:01:49 +00:00
Problem: Closes #31453 Solution: Introduce `vim.lsp.Capability`, which may serve as the base class for all LSP features that require caching data. it - was created if there is at least one client that supports the specific method; - was destroyed if all clients that support the method were detached. - Apply the refactor for `folding_range.lua` and `semantic_tokens.lua`. - Show active features in :checkhealth. Future: I found that these features that are expected to be refactored by `vim.lsp.Capability` have one characteristic in common: they all send LSP requests once the document is modified. The following code is different, but they are all for this purpose. - semantic tokens:fb8dba413f/runtime/lua/vim/lsp/semantic_tokens.lua (L192-L198)
- inlay hints, folding ranges, document colorfb8dba413f/runtime/lua/vim/lsp/inlay_hint.lua (L250-L266)
I think I can sum up this characteristic as the need to keep certain data synchronized with the latest version computed by the server. I believe we can handle this at the `vim.lsp.Capability` level, and I think it will be very useful. Therefore, my next step is to implement LSP request sending and data synchronization on `vim.lsp.Capability`, rather than limiting it to the current create/destroy data approach.
271 lines
8.5 KiB
Lua
271 lines
8.5 KiB
Lua
local M = {}
|
|
|
|
local report_info = vim.health.info
|
|
local report_warn = vim.health.warn
|
|
|
|
local function check_log()
|
|
local log = vim.lsp.log
|
|
local current_log_level = log.get_level()
|
|
local log_level_string = log.levels[current_log_level] ---@type string
|
|
report_info(string.format('LSP log level : %s', log_level_string))
|
|
|
|
if current_log_level < log.levels.WARN then
|
|
report_warn(
|
|
string.format(
|
|
'Log level %s will cause degraded performance and high disk usage',
|
|
log_level_string
|
|
)
|
|
)
|
|
end
|
|
|
|
local log_path = vim.lsp.get_log_path()
|
|
report_info(string.format('Log path: %s', log_path))
|
|
|
|
local log_file = vim.uv.fs_stat(log_path)
|
|
local log_size = log_file and log_file.size or 0
|
|
|
|
local report_fn = (log_size / 1000000 > 100 and report_warn or report_info)
|
|
report_fn(string.format('Log size: %d KB', log_size / 1000))
|
|
end
|
|
|
|
local function check_active_features()
|
|
vim.health.start('vim.lsp: Active Features')
|
|
---@type vim.lsp.Capability[]
|
|
local features = {
|
|
require('vim.lsp.semantic_tokens').__STHighlighter,
|
|
require('vim.lsp._folding_range').__FoldEvaluator,
|
|
}
|
|
for _, feature in ipairs(features) do
|
|
---@type string[]
|
|
local buf_infos = {}
|
|
for bufnr, instance in pairs(feature.active) do
|
|
local client_info = vim
|
|
.iter(pairs(instance.client_state))
|
|
:map(function(client_id)
|
|
local client = vim.lsp.get_client_by_id(client_id)
|
|
if client then
|
|
return string.format('%s (id: %d)', client.name, client.id)
|
|
else
|
|
return string.format('unknow (id: %d)', client_id)
|
|
end
|
|
end)
|
|
:join(', ')
|
|
if client_info == '' then
|
|
client_info = 'No supported client attached'
|
|
end
|
|
|
|
buf_infos[#buf_infos + 1] = string.format(' [%d]: %s', bufnr, client_info)
|
|
end
|
|
|
|
report_info(table.concat({
|
|
feature.name,
|
|
'- Active buffers:',
|
|
string.format(table.concat(buf_infos, '\n')),
|
|
}, '\n'))
|
|
end
|
|
end
|
|
|
|
--- @param f function
|
|
--- @return string
|
|
local function func_tostring(f)
|
|
local info = debug.getinfo(f, 'S')
|
|
return ('<function %s:%s>'):format(info.source, info.linedefined)
|
|
end
|
|
|
|
local function check_active_clients()
|
|
vim.health.start('vim.lsp: Active Clients')
|
|
local clients = vim.lsp.get_clients()
|
|
if next(clients) then
|
|
for _, client in pairs(clients) do
|
|
local server_version = vim.tbl_get(client, 'server_info', 'version')
|
|
or '? (no serverInfo.version response)'
|
|
local cmd ---@type string
|
|
local ccmd = client.config.cmd
|
|
if type(ccmd) == 'table' then
|
|
cmd = vim.inspect(ccmd)
|
|
elseif type(ccmd) == 'function' then
|
|
cmd = func_tostring(ccmd)
|
|
end
|
|
local dirs_info ---@type string
|
|
if client.workspace_folders and #client.workspace_folders > 1 then
|
|
local wfolders = {} --- @type string[]
|
|
for _, dir in ipairs(client.workspace_folders) do
|
|
wfolders[#wfolders + 1] = dir.name
|
|
end
|
|
dirs_info = ('- Workspace folders:\n %s'):format(table.concat(wfolders, '\n '))
|
|
else
|
|
dirs_info = string.format(
|
|
'- Root directory: %s',
|
|
client.root_dir and vim.fn.fnamemodify(client.root_dir, ':~')
|
|
) or nil
|
|
end
|
|
report_info(table.concat({
|
|
string.format('%s (id: %d)', client.name, client.id),
|
|
string.format('- Version: %s', server_version),
|
|
dirs_info,
|
|
string.format('- Command: %s', cmd),
|
|
string.format('- Settings: %s', vim.inspect(client.settings, { newline = '\n ' })),
|
|
string.format(
|
|
'- Attached buffers: %s',
|
|
vim.iter(pairs(client.attached_buffers)):map(tostring):join(', ')
|
|
),
|
|
}, '\n'))
|
|
end
|
|
else
|
|
report_info('No active clients')
|
|
end
|
|
end
|
|
|
|
local function check_watcher()
|
|
vim.health.start('vim.lsp: File Watcher')
|
|
|
|
-- Only run the check if file watching has been enabled by a client.
|
|
local clients = vim.lsp.get_clients()
|
|
if
|
|
--- @param client vim.lsp.Client
|
|
vim.iter(clients):all(function(client)
|
|
local has_capability = vim.tbl_get(
|
|
client.capabilities,
|
|
'workspace',
|
|
'didChangeWatchedFiles',
|
|
'dynamicRegistration'
|
|
)
|
|
local has_dynamic_capability =
|
|
client.dynamic_capabilities:get(vim.lsp.protocol.Methods.workspace_didChangeWatchedFiles)
|
|
return has_capability == nil
|
|
or has_dynamic_capability == nil
|
|
or client.workspace_folders == nil
|
|
end)
|
|
then
|
|
report_info('file watching "(workspace/didChangeWatchedFiles)" disabled on all clients')
|
|
return
|
|
end
|
|
|
|
local watchfunc = vim.lsp._watchfiles._watchfunc
|
|
assert(watchfunc)
|
|
local watchfunc_name --- @type string
|
|
if watchfunc == vim._watch.watch then
|
|
watchfunc_name = 'libuv-watch'
|
|
elseif watchfunc == vim._watch.watchdirs then
|
|
watchfunc_name = 'libuv-watchdirs'
|
|
elseif watchfunc == vim._watch.inotify then
|
|
watchfunc_name = 'inotify'
|
|
else
|
|
local nm = debug.getinfo(watchfunc, 'S').source
|
|
watchfunc_name = string.format('Custom (%s)', nm)
|
|
end
|
|
|
|
report_info('File watch backend: ' .. watchfunc_name)
|
|
if watchfunc_name == 'libuv-watchdirs' then
|
|
report_warn('libuv-watchdirs has known performance issues. Consider installing inotify-tools.')
|
|
end
|
|
end
|
|
|
|
local function check_position_encodings()
|
|
vim.health.start('vim.lsp: Position Encodings')
|
|
local clients = vim.lsp.get_clients()
|
|
if next(clients) then
|
|
local position_encodings = {} ---@type table<integer, table<string, integer[]>>
|
|
for _, client in pairs(clients) do
|
|
for bufnr in pairs(client.attached_buffers) do
|
|
if not position_encodings[bufnr] then
|
|
position_encodings[bufnr] = {}
|
|
end
|
|
if not position_encodings[bufnr][client.offset_encoding] then
|
|
position_encodings[bufnr][client.offset_encoding] = {}
|
|
end
|
|
table.insert(position_encodings[bufnr][client.offset_encoding], client.id)
|
|
end
|
|
end
|
|
|
|
-- Check if any buffers are attached to multiple clients with different position encodings
|
|
local buffers = {} ---@type integer[]
|
|
for bufnr, encodings in pairs(position_encodings) do
|
|
local list = {} ---@type string[]
|
|
for k in pairs(encodings) do
|
|
list[#list + 1] = k
|
|
end
|
|
|
|
if #list > 1 then
|
|
buffers[#buffers + 1] = bufnr
|
|
end
|
|
end
|
|
|
|
if #buffers > 0 then
|
|
local lines =
|
|
{ 'Found buffers attached to multiple clients with different position encodings.' }
|
|
for _, bufnr in ipairs(buffers) do
|
|
local encodings = position_encodings[bufnr]
|
|
local parts = {}
|
|
for encoding, client_ids in pairs(encodings) do
|
|
table.insert(
|
|
parts,
|
|
string.format('%s (client id(s): %s)', encoding:upper(), table.concat(client_ids, ', '))
|
|
)
|
|
end
|
|
table.insert(lines, string.format('- Buffer %d: %s', bufnr, table.concat(parts, ', ')))
|
|
end
|
|
report_warn(
|
|
table.concat(lines, '\n'),
|
|
'Use the positionEncodings client capability to ensure all clients use the same position encoding'
|
|
)
|
|
else
|
|
report_info('No buffers contain mixed position encodings')
|
|
end
|
|
else
|
|
report_info('No active clients')
|
|
end
|
|
end
|
|
|
|
local function check_enabled_configs()
|
|
vim.health.start('vim.lsp: Enabled Configurations')
|
|
|
|
for name in vim.spairs(vim.lsp._enabled_configs) do
|
|
local config = vim.lsp.config[name]
|
|
local text = {} --- @type string[]
|
|
text[#text + 1] = ('%s:'):format(name)
|
|
if not config then
|
|
report_warn(
|
|
("'%s' config not found. Ensure that vim.lsp.config('%s') was called."):format(name, name)
|
|
)
|
|
else
|
|
for k, v in
|
|
vim.spairs(config --[[@as table<string,any>]])
|
|
do
|
|
local v_str --- @type string?
|
|
if k == 'name' then
|
|
v_str = nil
|
|
elseif k == 'filetypes' then
|
|
v_str = table.concat(v, ', ')
|
|
elseif type(v) == 'function' then
|
|
v_str = func_tostring(v)
|
|
else
|
|
v_str = vim.inspect(v, { newline = '\n ' })
|
|
end
|
|
|
|
if k == 'cmd' and type(v) == 'table' and vim.fn.executable(v[1]) == 0 then
|
|
report_warn(("'%s' is not executable. Configuration will not be used."):format(v[1]))
|
|
end
|
|
|
|
if v_str then
|
|
text[#text + 1] = ('- %s: %s'):format(k, v_str)
|
|
end
|
|
end
|
|
end
|
|
text[#text + 1] = ''
|
|
report_info(table.concat(text, '\n'))
|
|
end
|
|
end
|
|
|
|
--- Performs a healthcheck for LSP
|
|
function M.check()
|
|
check_log()
|
|
check_active_features()
|
|
check_active_clients()
|
|
check_enabled_configs()
|
|
check_watcher()
|
|
check_position_encodings()
|
|
end
|
|
|
|
return M
|