Skip to content

Commit 7b272b6

Browse files
committed
feat: automatically detect config module changes in or oustside Neovim and reload
1 parent fbfa790 commit 7b272b6

File tree

6 files changed

+94
-8
lines changed

6 files changed

+94
-8
lines changed

README.md

+4-4
Original file line numberDiff line numberDiff line change
@@ -29,19 +29,19 @@
2929
- [ ] other package manager artifacts still present? compiled etc
3030
- [x] rename `run` to `build`
3131
- [ ] delete lazy keymaps when a plugin loads
32-
- [ ] temp colorscheme
33-
- [x] allow setting up plugins through config
32+
- [x] temp colorscheme
33+
- [x] allow setting up plugins through config **fooo**
3434
- [x] task timeout
3535
- [ ] log file
3636
- [ ] deal with re-sourcing init.lua. Check a global?
3737
- [x] incorrect when switching TN from opt to start
3838
- [ ] git tests
3939
- [x] max concurrency
4040
- [x] ui border
41-
- [ ] make sure we can reload specs while keeping state
41+
- [x] make sure we can reload specs while keeping state
4242
- [ ] show disabled plugins (strikethrough?)
4343
- [ ] Import specs from other plugin managers
44-
- [ ] use uv file watcher (or stat) to check for config changes
44+
- [x] use uv file watcher (or stat) to check for config changes
4545
- [ ] [packspec](https://github.com/nvim-lua/nvim-package-specification)
4646
- [ ] add support to specify `engines`, `os` and `cpu` like in `package.json`
4747
- [ ] semver merging. Should check if two or more semver ranges are compatible and calculate the union range

lua/lazy/core/config.lua

+2-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,7 @@ M.defaults = {
3333
-- install missing plugins on startup. This doesn't increase startup time.
3434
missing = true,
3535
-- try to load one of the colorschemes in this list when starting an install during startup
36-
-- the first colorscheme that is found will be used
36+
-- the first colorscheme that is found will be loaded
3737
colorscheme = { "habamax" },
3838
},
3939
ui = {
@@ -100,6 +100,7 @@ function M.setup(spec, opts)
100100
callback = function()
101101
require("lazy.core.cache").autosave()
102102
require("lazy.view").setup()
103+
require("lazy.manage.reloader").enable()
103104
end,
104105
})
105106

lua/lazy/core/handler.lua

+3-1
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,9 @@ function M.trigger(groups, events, pattern)
6363
end
6464
Util.info(lines)
6565
end
66-
vim.api.nvim_exec_autocmds(autocmd.event, { group = autocmd.group, modeline = false })
66+
Util.try(function()
67+
vim.api.nvim_exec_autocmds(autocmd.event, { group = autocmd.group, modeline = false })
68+
end)
6769
end
6870
end
6971
end

lua/lazy/core/plugin.lua

+2
Original file line numberDiff line numberDiff line change
@@ -195,6 +195,8 @@ function M.spec()
195195
-- spec is a module
196196
local function _load(name)
197197
local modname = name and (Config.spec .. "." .. name) or Config.spec
198+
-- unload the module so we get a clean slate
199+
---@diagnostic disable-next-line: no-unknown
198200
package.loaded[modname] = nil
199201
Util.try(function()
200202
spec:normalize(require(modname))

lua/lazy/manage/init.lua

+1-2
Original file line numberDiff line numberDiff line change
@@ -115,15 +115,14 @@ end
115115

116116
---@param opts? ManagerOpts
117117
function M.clean(opts)
118-
Plugin.update_state()
119118
M.run({
120119
pipeline = { "fs.clean" },
121120
plugins = Config.to_clean,
122121
}, opts)
123122
end
124123

125124
function M.clear()
126-
Plugin.update_state()
125+
Plugin.load()
127126
for _, plugin in pairs(Config.plugins) do
128127
plugin._.updated = nil
129128
plugin._.cloned = nil

lua/lazy/manage/reloader.lua

+82
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,82 @@
1+
local Cache = require("lazy.core.cache")
2+
local Config = require("lazy.core.config")
3+
local Util = require("lazy.util")
4+
local Plugin = require("lazy.core.plugin")
5+
6+
local M = {}
7+
8+
---@type table<string, CacheHash>
9+
M.files = {}
10+
11+
---@type vim.loop.Timer
12+
M.timer = nil
13+
M.main = nil
14+
M.root = nil
15+
16+
function M.enable()
17+
if M.timer then
18+
M.timer:stop()
19+
end
20+
if type(Config.spec) == "string" then
21+
M.timer = vim.loop.new_timer()
22+
M.root = vim.fn.stdpath("config") .. "/lua/" .. Config.spec:gsub("%.", "/")
23+
M.main = vim.loop.fs_stat(M.root .. ".lua") and (M.root .. ".lua") or (M.root .. "/init.lua")
24+
M.check(true)
25+
M.timer:start(2000, 2000, M.check)
26+
end
27+
end
28+
29+
function M.disable()
30+
if M.timer then
31+
M.timer:stop()
32+
M.timer = nil
33+
end
34+
end
35+
36+
function M.check(start)
37+
---@type table<string,true>
38+
local checked = {}
39+
---@type {file:string, what:string}[]
40+
local changes = {}
41+
42+
-- spec is a module
43+
local function check(_, modpath)
44+
checked[modpath] = true
45+
local hash = Cache.hash(modpath)
46+
if hash then
47+
if M.files[modpath] then
48+
if not Cache.eq(M.files[modpath], hash) then
49+
M.files[modpath] = hash
50+
table.insert(changes, { file = modpath, what = "changed" })
51+
end
52+
else
53+
M.files[modpath] = hash
54+
table.insert(changes, { file = modpath, what = "added" })
55+
end
56+
end
57+
end
58+
59+
check(nil, M.main)
60+
Util.lsmod(M.root, check)
61+
62+
for file in pairs(M.files) do
63+
if not checked[file] then
64+
table.insert(changes, { file = file, what = "deleted" })
65+
M.files[file] = nil
66+
end
67+
end
68+
69+
if not (start or #changes == 0) then
70+
vim.schedule(function()
71+
local lines = { "# Config Change Detected. Reloading...", "" }
72+
for _, change in ipairs(changes) do
73+
table.insert(lines, "- **" .. change.what .. "**: `" .. vim.fn.fnamemodify(change.file, ":p:~:.") .. "`")
74+
end
75+
Util.warn(lines)
76+
Plugin.load()
77+
vim.cmd([[do User LazyRender]])
78+
end)
79+
end
80+
end
81+
82+
return M

0 commit comments

Comments
 (0)