nvim-config/lua/plugins/language-manager.lua

371 lines
8.5 KiB
Lua

local M = {
lsp = {},
ts = {},
lint = {},
format = {},
}
M.group = vim.api.nvim_create_augroup('language-manager', { clear = true })
-- ======== Helpers ========
local cache_path = vim.fn.stdpath('cache') .. '/language-manager.json'
local function wrap(item)
if type(item) == 'string' then
return { item }
elseif type(item) == 'table' then
return item
else
return {}
end
end
local function create_set()
local seen = {}
return function(ref, item)
if not seen[ref] then
seen[ref] = {}
end
if item and not seen[ref][item] then
seen[ref][item] = true
return true
end
return false
end
end
local function read_file(path)
local f = io.open(path, 'r')
if not f then
return nil
end
local content = f:read('*a')
f:close()
return content
end
local function write_file(path, content)
local f = io.open(path, 'w')
if not f then
return false
end
f:write(content)
f:close()
return true
end
local function to_json(tbl)
local ok, result = pcall(vim.json.encode, tbl)
return ok and result or nil
end
local function from_json(str)
local ok, result = pcall(vim.json.decode, str)
return ok and result or nil
end
local function hash_spec(tbl)
local encoded = to_json(tbl) or ''
if vim.fn.exists('*sha256') == 1 then
return vim.fn.sha256(encoded)
else
local tmp = vim.fn.tempname()
write_file(tmp, encoded)
local handle = io.popen('openssl dgst -sha256 ' .. tmp)
local result = handle and handle:read('*a') or ''
if handle then
handle:close()
end
return result:match('([a-f0-9]+)') or ''
end
end
local function save_cache(data)
local encoded = to_json(data)
if encoded then
write_file(cache_path, encoded)
end
end
local function load_cache()
local raw = read_file(cache_path)
if not raw then
return nil
end
return from_json(raw)
end
-- ======== Spec Builder ========
local function create_spec()
local S = {}
local unique = create_set()
local specs = {}
function S.set(item, group)
specs[group] = item
end
function S.get()
return specs
end
function S.add(list, ...)
local groups = { ... }
local ref = table.concat(groups, '.')
local pointer = specs
local last_i = #groups
for i = 1, last_i do
local group = groups[i]
if i == last_i then
for _, item in ipairs(wrap(list)) do
if unique(ref, item) then
pointer[group] = pointer[group] or {}
table.insert(pointer[group], item)
end
end
else
pointer[group] = pointer[group] or {}
pointer = pointer[group]
end
end
end
return S
end
-- ======== Spec Generation ========
function M.generate_specs(specs_raw)
local install_spec = create_spec()
local specs = create_spec()
-- ensure Mason is available
vim.cmd.packadd('mason.nvim')
require('mason').setup()
local registry = require('mason-registry')
local lsp_map = {}
for _, spec in ipairs(specs_raw) do
if type(spec) == 'string' then
spec = { ft = spec }
end
spec.ts = spec.ts or spec.ft
specs.add(spec.ts, 'ts_parsers')
-- resolve LSP name using Mason registry
install_spec.add(spec.lsp, 'language_servers')
local resolved_lsps = {}
for _, lsp in ipairs(wrap(spec.lsp)) do
if registry.has_package(lsp) then
local pkg = registry.get_package(lsp)
local lspconfig = pkg.spec and pkg.spec.neovim and pkg.spec.neovim.lspconfig or lsp
table.insert(resolved_lsps, lspconfig)
lsp_map[lspconfig] = lsp
else
vim.notify('Unknown LSP: ' .. lsp, vim.log.levels.WARN)
end
end
specs.add(resolved_lsps, 'language_servers')
install_spec.add(spec.lint, 'code_tools')
install_spec.add(spec.format, 'code_tools')
for _, ft in ipairs(wrap(spec.ft)) do
specs.add(spec.lint, 'linters_by_ft', ft)
specs.add(spec.format, 'formatters_by_ft', ft)
end
end
local result = specs.get()
result.lsp_map = lsp_map
return result, install_spec.get()
end
-- ======== Cache Interface ========
function M.load_or_generate(specs_raw)
local hash = hash_spec(specs_raw)
local cache = load_cache()
if cache and cache.hash == hash then
M.general = cache.spec
else
M.general, M.mason = M.generate_specs(specs_raw)
save_cache({ hash = hash, spec = M.general })
end
return M.general, M.mason
end
function M.load_specs()
local cache = load_cache()
if cache then
M.general = cache.spec
else
local specs_raw = require('modules.language-specs').get()
local hash = hash_spec(specs_raw)
M.general, M.mason = M.generate_specs(specs_raw)
save_cache({ hash = hash, spec = M.general })
end
return M.general, M.mason
end
function M.invalidate_cache()
local ok = os.remove(cache_path)
if ok then
vim.notify('Language manager cache invalidated', vim.log.levels.INFO)
else
vim.notify('No cache to invalidate or failed to delete', vim.log.levels.WARN)
end
M.general = nil
M.mason = nil
end
-- ======== Setup Functions ========
local function on_ts_installed(parsers, cb)
local info = require('nvim-treesitter.info')
local timer = vim.loop.new_timer()
timer:start(0, 500, vim.schedule_wrap(function()
for _, lang in ipairs(parsers) do
if not vim.tbl_contains(info.installed_parsers(), lang) then
return
end
end
timer:stop()
timer:close()
cb()
end))
end
function M.ts.install()
local install = require('nvim-treesitter.install')
local parsers = M.general.ts_parsers or {}
install.ensure_installed(parsers) -- async
return parsers
end
-- ======== Orchestration ========
function M.install()
local parsers = M.ts.install()
on_ts_installed(parsers, function()
M.mason_install()
end)
end
local function ensure_registry_ready()
local registry = require('mason-registry')
if not registry.is_installed() then
registry.update()
vim.wait(10000, function()
return registry.is_installed()
end, 200)
end
return registry
end
function M.mason_install()
vim.cmd.packadd('mason.nvim')
require('mason').setup()
local registry = ensure_registry_ready()
local list = vim.list_extend(M.mason.language_servers, M.mason.code_tools)
print(vim.inspect(list))
local function install_and_wait(name)
local pkg = registry.get_package(name)
if pkg:is_installed() then
vim.notify('Already installed ' .. name, vim.log.levels.INFO)
return
end
vim.notify('Installing ' .. name, vim.log.levels.INFO)
local done = false
pkg:install():once('closed', function()
done = true
end)
local ok = vim.wait(60 * 60 * 1000, function()
return done or pkg:is_installed()
end, 200)
if not ok or not pkg:is_installed() then
vim.notify('Install failed: ' .. name, vim.log.levels.ERROR)
else
vim.notify('Installed ' .. name, vim.log.levels.INFO)
end
end
for _, name in ipairs(list) do
if registry.has_package(name) then
install_and_wait(name)
else
vim.notify('Package not found in registry: ' .. name, vim.log.levels.WARN)
end
end
-- force event loop to process any pending Mason async teardown
local settled = false
vim.defer_fn(function()
settled = true
end, 3000)
vim.wait(10 * 1000, function()
return settled
end, 100)
end
function M.lsp.setup()
for _, lsp_name in ipairs(M.general.language_servers or {}) do
vim.lsp.enable(lsp_name)
end
end
function M.ts.setup()
require('nvim-treesitter.configs').setup({
highlight = { enable = true },
incremental_selection = { enable = true },
})
end
function M.lint.setup()
vim.api.nvim_create_autocmd({ 'BufReadPre', 'BufNewFile' }, {
group = M.group,
callback = function()
local lint = require('lint')
lint.linters_by_ft = M.general.linters_by_ft or {}
vim.api.nvim_create_autocmd({ 'BufEnter', 'BufWritePost', 'InsertLeave' }, {
group = vim.api.nvim_create_augroup('language-manager.lint', { clear = true }),
callback = function()
lint.try_lint()
end,
})
end,
})
end
function M.format.setup()
vim.api.nvim_create_autocmd('BufWritePre', {
group = M.group,
once = true,
callback = function()
require('conform').setup({
format_on_save = { timeout_ms = 500, lsp_format = 'fallback' },
default_format_opts = { stop_after_first = true },
formatters_by_ft = M.general.formatters_by_ft or {},
})
end,
})
end
function M.setup()
M.load_specs()
M.ts.setup()
M.lsp.setup()
M.lint.setup()
M.format.setup()
end
return M