commit c2ecc9ca41a95b1644cae183fd19ce49ce10f980
parent dfa5d79d05d82aeedc7c19f9b670f31e01e3aea0
Author: Justin M. Keyes <justinkz@gmail.com>
Date: Tue, 27 Jan 2026 09:19:46 -0500
Merge #37309 :checkhealth vim.pack
Diffstat:
2 files changed, 263 insertions(+), 1 deletion(-)
diff --git a/runtime/lua/vim/pack.lua b/runtime/lua/vim/pack.lua
@@ -842,9 +842,13 @@ local function lock_sync(confirm)
end
-- Compute installed plugins
+ local plug_dir = get_plug_dir()
+ if vim.uv.fs_stat(plug_dir) == nil then
+ vim.fn.mkdir(plug_dir, 'p')
+ end
+
-- NOTE: The directory traversal is done on every startup, but it is very fast.
-- Also, single `vim.fs.dir()` scales better than on demand `uv.fs_stat()` checks.
- local plug_dir = get_plug_dir()
local installed = {} --- @type table<string,string>
for name, fs_type in vim.fs.dir(plug_dir) do
installed[name] = fs_type
diff --git a/runtime/lua/vim/pack/health.lua b/runtime/lua/vim/pack/health.lua
@@ -0,0 +1,258 @@
+local M = {}
+
+local health = vim.health
+
+local function get_lockfile_path()
+ return vim.fs.joinpath(vim.fn.stdpath('config'), 'nvim-pack-lock.json')
+end
+
+local function get_plug_dir()
+ return vim.fs.joinpath(vim.fn.stdpath('data'), 'site', 'pack', 'core', 'opt')
+end
+
+local function git_cmd(cmd, cwd)
+ cmd = vim.list_extend({ 'git', '-c', 'gc.auto=0' }, cmd)
+ local env = vim.fn.environ() --- @type table<string,string>
+ env.GIT_DIR, env.GIT_WORK_TREE = nil, nil
+ local sys_opts = { cwd = cwd, text = true, env = env, clear_env = true }
+ local out = vim.system(cmd, sys_opts):wait() --- @type vim.SystemCompleted
+ if out.code ~= 0 then
+ return false, ((out.stderr or ''):gsub('\n+$', ''))
+ end
+ return true, ((out.stdout or ''):gsub('\n+$', ''))
+end
+
+local function check_basics()
+ health.start('vim.pack: basics')
+
+ -- Requirements
+ if vim.fn.executable('git') == 0 then
+ health.warn('`git` executable is required. Install it using your package manager')
+ return false, false
+ end
+
+ -- Detect if not used
+ local lockfile_path = get_lockfile_path()
+ local has_lockfile = vim.fn.filereadable(lockfile_path) == 1
+ local plug_dir = get_plug_dir()
+ local has_plug_dir = vim.fn.isdirectory(plug_dir) == 1
+ if not has_lockfile and not has_plug_dir then
+ health.ok('`vim.pack` is not used')
+ return false, false
+ end
+
+ -- General info
+ local git = vim.fn.exepath('git')
+ local _, version = git_cmd({ 'version' }, vim.uv.cwd())
+ health.info(('Git: %s (%s)'):format(version:gsub('^git%s*', ''), git))
+ health.info('Lockfile: ' .. lockfile_path)
+ health.info('Plugin directory: ' .. plug_dir)
+
+ if has_lockfile and has_plug_dir then
+ health.ok('')
+ else
+ local lockfile_absent = has_lockfile and 'present' or 'absent'
+ local plug_dir_absent = has_plug_dir and 'present' or 'absent'
+ local msg = ('Lockfile is %s, plugin directory is %s.'):format(lockfile_absent, plug_dir_absent)
+ .. ' Restart Nvim and run `vim.pack.add({})` to '
+ .. (has_lockfile and 'install plugins from the lockfile' or 'regenerate the lockfile')
+ health.warn(msg)
+ end
+
+ return has_lockfile, has_plug_dir
+end
+
+local function is_version(x)
+ return type(x) == 'string' or (type(x) == 'table' and pcall(x.has, x, '1'))
+end
+
+local function failed_git_cmd(plug_name, plug_path)
+ local msg = ('Failed Git command inside plugin %s.'):format(vim.inspect(plug_name))
+ .. ' This is unexpected and should not happen.'
+ .. (' Manually delete directory %s and reinstall plugin'):format(plug_path)
+ health.error(msg)
+ return false
+end
+
+--- @return boolean Whether a check is successful
+local function check_plugin_lock_data(plug_name, lock_data)
+ local name_str = vim.inspect(plug_name)
+ local error_with_del_advice = function(reason)
+ local msg = ('%s %s.'):format(name_str, reason)
+ .. (' Delete %s entry (do not create trailing comma) and '):format(name_str)
+ .. 'restart Nvim to regenerate lockfile data'
+ health.error(msg)
+ return false
+ end
+
+ -- Types
+ if type(plug_name) ~= 'string' then
+ return error_with_del_advice('is not a valid plugin name')
+ end
+ if type(lock_data) ~= 'table' then
+ return error_with_del_advice('entry is not a valid type')
+ end
+ if type(lock_data.rev) ~= 'string' then
+ local reason = '`rev` entry is ' .. (lock_data.rev and 'not a valid type' or 'missing')
+ return error_with_del_advice(reason)
+ end
+ if type(lock_data.src) ~= 'string' then
+ local reason = '`src` entry is ' .. (lock_data.src and 'not a valid type' or 'missing')
+ return error_with_del_advice(reason)
+ end
+ if lock_data.version and not is_version(lock_data.version) then
+ return error_with_del_advice('`version` entry is not a valid type')
+ end
+
+ -- Alignment with what is actually present on disk
+ local plug_path = vim.fs.joinpath(get_plug_dir(), plug_name)
+ if vim.fn.isdirectory(plug_path) ~= 1 then
+ health.warn(
+ ('Plugin %s is not installed but present in the lockfile.'):format(name_str)
+ .. ' Restart Nvim and run `vim.pack.add({})` to autoinstall.'
+ .. (' To fully delete, run `vim.pack.del({ %s }, { force = true })`'):format(name_str)
+ )
+ return false
+ end
+
+ -- NOTE: `vim.pack` currently only supports Git repos as plugins
+ if not git_cmd({ 'rev-parse', '--git-dir' }, plug_path) then
+ return true
+ end
+
+ local has_head, head = git_cmd({ 'rev-list', '-1', 'HEAD' }, plug_path)
+ if not has_head then
+ return failed_git_cmd(plug_name, plug_path)
+ elseif lock_data.rev ~= head then
+ health.error(
+ ('Plugin %s is not at expected revision\n'):format(name_str)
+ .. ('Expected: %s\nActual: %s\n'):format(lock_data.rev, head)
+ .. 'To synchronize, restart Nvim and run '
+ .. ('`vim.pack.update({ %s }, { offline = true })`'):format(name_str)
+ )
+ return false
+ end
+
+ local has_origin, origin = git_cmd({ 'remote', 'get-url', 'origin' }, plug_path)
+ if not has_origin then
+ return failed_git_cmd(plug_name, plug_path)
+ elseif lock_data.src ~= origin then
+ health.error(
+ ('Plugin %s has not expected source\n'):format(name_str)
+ .. ('Expected: %s\nActual: %s\n'):format(lock_data.src, origin)
+ .. 'Delete `src` lockfile entry (do not create trailing comma) and '
+ .. 'restart Nvim to regenerate lockfile data'
+ )
+ return false
+ end
+
+ return true
+end
+
+local function check_lockfile()
+ health.start('vim.pack: lockfile')
+
+ local can_read, text = pcall(vim.fn.readblob, get_lockfile_path())
+ if not can_read then
+ health.error('Could not read lockfile. Delete it and restart Nvim.')
+ return
+ end
+
+ local can_parse, data = pcall(vim.json.decode, text)
+ if not can_parse then
+ health.error(('Could not parse lockfile: %s\nDelete it and restart Nvim'):format(data))
+ return
+ end
+
+ if type(data.plugins) ~= 'table' then
+ health.error('Field `plugins` is not proper type. Delete lockfile and restart Nvim')
+ return
+ end
+
+ local is_good = true
+ --- @cast data { plugins: table<string,table> }
+ for plug_name, lock_data in pairs(data.plugins) do
+ is_good = check_plugin_lock_data(plug_name, lock_data) and is_good
+ end
+
+ if is_good then
+ health.ok('')
+ end
+end
+
+--- @return boolean Whether a check is successful
+local function check_installed_plugin(plug_name)
+ local name_str = vim.inspect(plug_name)
+ local plug_path = vim.fs.joinpath(get_plug_dir(), plug_name)
+
+ if vim.fn.isdirectory(plug_path) ~= 1 then
+ health.error(('%s is not a directory. Delete it'):format(plug_name))
+ return false
+ end
+
+ if not git_cmd({ 'rev-parse', '--git-dir' }, plug_path) then
+ health.error(
+ ('%s is not a Git repository.'):format(name_str)
+ .. ' It was not installed by `vim.pack` and should not be present in the plugin directory.'
+ .. ' If installed manually, use dedicated `:h packages`'
+ )
+ return false
+ end
+
+ -- Detached HEAD is a sign that plugin is managed by `vim.pack`
+ local has_head_ref, head_ref = git_cmd({ 'rev-parse', '--abbrev-ref', 'HEAD' }, plug_path)
+ if not has_head_ref then
+ return failed_git_cmd(plug_name, plug_path)
+ elseif head_ref ~= 'HEAD' then
+ health.warn(
+ ('Plugin %s is not at state which is a result of `vim.pack` operation.\n'):format(name_str)
+ .. 'If it was intentional, make sure you know what you are doing.\n'
+ .. 'Otherwise, restart Nvim and run '
+ .. ('`vim.pack.update({ %s }, { offline = true })`.\n'):format(name_str)
+ .. 'If nothing is updated, plugin is at correct revision and will be managed as expected'
+ )
+ return false
+ end
+
+ -- Usage data
+ local has_pack_info, info = pcall(vim.pack.get, { plug_name })
+ if not has_pack_info then
+ health.error('Could not get `vim.pack` usage information for plugin ' .. name_str)
+ return false
+ end
+
+ if not info[1].active then
+ health.info(
+ ('Plugin %s is not active.'):format(name_str)
+ .. ' Is it lazy loaded or did you forget to run `vim.pack.del()`?'
+ )
+ end
+
+ return true
+end
+
+local function check_plug_dir()
+ health.start('vim.pack: plugin directory')
+
+ local is_good = true
+ local plug_dir = get_plug_dir()
+ for plug_name, _ in vim.fs.dir(plug_dir) do
+ is_good = check_installed_plugin(plug_name) and is_good
+ end
+
+ if is_good then
+ health.ok('')
+ end
+end
+
+function M.check()
+ local has_lockfile, has_plug_dir = check_basics()
+ if has_lockfile then
+ check_lockfile()
+ end
+ if has_plug_dir then
+ check_plug_dir()
+ end
+end
+
+return M