371 lines
8.5 KiB
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
|