Skip to content

Commit fe33e4e

Browse files
committed
feat: lazy caching now works with functions that have upvalues
1 parent 48199f8 commit fe33e4e

File tree

9 files changed

+161
-187
lines changed

9 files changed

+161
-187
lines changed

lua/lazy/core/config.lua

+1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ M.defaults = {
1111
---@type string[]
1212
patterns = {},
1313
},
14+
interactive = true,
1415
package_path = vim.fn.stdpath("data") .. "/site/pack/lazy",
1516
view = {
1617
icons = {

lua/lazy/core/loader.lua

+4-1
Original file line numberDiff line numberDiff line change
@@ -149,6 +149,9 @@ function M.init_plugins()
149149
Util.track("plugin_init")
150150
for _, name in ipairs(M.loaders.init) do
151151
local plugin = Config.plugins[name]
152+
if not plugin then
153+
error(name)
154+
end
152155
if plugin.init then
153156
Util.track(plugin.name)
154157
plugin.init()
@@ -247,7 +250,7 @@ end
247250
---@param plugin LazyPlugin
248251
function M.packadd(plugin, load_start)
249252
if plugin.opt then
250-
vim.cmd.packadd(plugin.pack)
253+
vim.cmd.packadd(plugin.name)
251254
M.source_plugin_files(plugin, true)
252255
elseif load_start then
253256
vim.opt.runtimepath:append(plugin.dir)

lua/lazy/core/state.lua

+58-71
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,18 @@
11
local Cache = require("lazy.core.cache")
22
local Module = require("lazy.core.module")
3+
local Config = require("lazy.core.config")
34

45
local M = {}
56

67
M.dirty = true
78

9+
---@alias CachedPlugin LazyPlugin | {_funs: string[]}
10+
local skip = { installed = true, loaded = true, tasks = true, dirty = true, dir = true }
11+
local funs = { config = true, init = true, run = true }
12+
813
function M.update_state(check_clean)
914
local Util = require("lazy.core.util")
10-
local Config = require("lazy.core.config")
15+
1116
---@type table<"opt"|"start", table<string,boolean>>
1217
local installed = { opt = {}, start = {} }
1318
for opt, packs in pairs(installed) do
@@ -19,9 +24,11 @@ function M.update_state(check_clean)
1924
end
2025

2126
for _, plugin in pairs(Config.plugins) do
27+
plugin.opt = plugin.opt == nil and Config.options.opt or plugin.opt
2228
local opt = plugin.opt and "opt" or "start"
23-
plugin.installed = installed[opt][plugin.pack] == true
24-
installed[opt][plugin.pack] = nil
29+
plugin.dir = Config.options.package_path .. "/" .. opt .. "/" .. plugin.name
30+
plugin.installed = installed[opt][plugin.name] == true
31+
installed[opt][plugin.name] = nil
2532
end
2633

2734
if check_clean then
@@ -44,101 +51,81 @@ function M.save()
4451
if not M.dirty then
4552
return
4653
end
47-
local Config = require("lazy.core.config")
54+
local Plugin = require("lazy.plugin")
4855

4956
---@class LazyState
5057
local state = {
51-
---@type CachedPlugin[]
52-
plugins = {},
58+
---@type table<string, LazySpec>
59+
specs = {},
5360
loaders = require("lazy.core.loader").loaders,
5461
config = Config.options,
5562
}
5663

57-
---@alias CachedPlugin LazyPlugin | {_funcs: table<string, number|boolean>}
58-
local skip = { installed = true, loaded = true, tasks = true, dirty = true, dir = true }
59-
local funcount = 0
60-
61-
for _, plugin in pairs(Config.plugins) do
62-
---@type CachedPlugin
63-
local save = {}
64-
table.insert(state.plugins, save)
65-
---@diagnostic disable-next-line: no-unknown
66-
for k, v in pairs(plugin) do
67-
if type(v) == "function" then
68-
save._funcs = save._funcs or {}
69-
if plugin.modname then
70-
save._funcs[k] = true
71-
else
72-
funcount = funcount + 1
73-
Cache.set("cache.state.fun." .. funcount, string.dump(v))
74-
save._funcs[k] = funcount
64+
for _, spec in ipairs(Plugin.specs()) do
65+
state.specs[spec.modname] = spec
66+
for _, plugin in pairs(spec.plugins) do
67+
---@cast plugin CachedPlugin
68+
for k, v in pairs(plugin) do
69+
if type(v) == "function" then
70+
if funs[k] then
71+
plugin._funs = plugin._funs or {}
72+
table.insert(plugin._funs, k)
73+
end
74+
plugin[k] = nil
75+
elseif skip[k] then
76+
plugin[k] = nil
7577
end
76-
elseif not skip[k] then
77-
save[k] = v
7878
end
7979
end
8080
end
8181
Cache.set("cache.state", vim.json.encode(state))
8282
end
8383

8484
function M.load()
85-
---@type boolean, LazyState
86-
local ok, state = pcall(vim.json.decode, Cache.get("cache.state"))
87-
if not ok then
88-
Cache.dirty()
89-
return false
90-
end
85+
local Plugin = require("lazy.plugin")
86+
local dirty = false
9187

92-
local Config = require("lazy.core.config")
93-
94-
if not vim.deep_equal(Config.options, state.config) then
95-
Cache.dirty()
96-
return false
88+
---@type boolean, LazyState?
89+
local ok, state = pcall(vim.json.decode, Cache.get("cache.state"))
90+
if not (ok and state and vim.deep_equal(Config.options, state.config)) then
91+
dirty = true
92+
state = nil
9793
end
9894

99-
if Module.is_dirty(Config.options.plugins, Config.paths.main) then
100-
return false
101-
end
95+
local function _loader(modname, modpath)
96+
local spec = state and state.specs[modname]
97+
if (not spec) or Module.is_dirty(modname, modpath) then
98+
dirty = true
99+
vim.schedule(function()
100+
vim.notify("Reloading " .. modname)
101+
end)
102+
return Plugin.Spec.load(modname, modpath)
103+
end
104+
---@type LazySpec
105+
local loaded = nil
102106

103-
-- plugins
104-
for _, plugin in ipairs(state.plugins) do
105-
Config.plugins[plugin.name] = plugin
106-
plugin.loaded = nil
107-
plugin.dir = Config.options.package_path .. "/" .. (plugin.opt and "opt" or "start") .. "/" .. plugin.pack
108-
if plugin.modname then
109-
if Module.is_dirty(plugin.modname, plugin.modpath) then
110-
return false
111-
end
112-
for fun in pairs(plugin._funcs or {}) do
113-
---@diagnostic disable-next-line: assign-type-mismatch
114-
plugin[fun] = function(...)
115-
local mod = Module.load(plugin.modname, plugin.modpath)
116-
for k in pairs(plugin._funcs) do
117-
plugin[k] = mod[k]
118-
end
119-
return plugin[fun](...)
120-
end
121-
end
122-
elseif plugin._funcs then
123-
for fun, id in pairs(plugin._funcs) do
124-
local chunk = assert(Cache.get("cache.state.fun." .. id))
125-
---@diagnostic disable-next-line: assign-type-mismatch
107+
for name, plugin in pairs(spec.plugins) do
108+
---@cast plugin CachedPlugin
109+
for _, fun in ipairs(plugin._funs or {}) do
126110
plugin[fun] = function(...)
127-
---@diagnostic disable-next-line: assign-type-mismatch
128-
plugin[fun] = loadstring(chunk)
129-
return plugin[fun](...)
111+
loaded = loaded or Plugin.Spec.load(spec.modname, spec.modpath)
112+
return loaded.plugins[name][fun](...)
130113
end
131114
end
132115
end
116+
return spec
133117
end
134-
M.update_state()
135118

136-
-- loaders
137-
require("lazy.core.loader").loaders = state.loaders
119+
Plugin.load(_loader)
138120

139-
M.dirty = false
121+
if state and not dirty then
122+
require("lazy.core.loader").loaders = state.loaders
123+
else
124+
Cache.dirty()
125+
end
140126

141-
return true
127+
M.dirty = dirty
128+
return not dirty
142129
end
143130

144131
return M

lua/lazy/core/util.lua

+14
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,20 @@ function M.track(name, time)
2727
end
2828
end
2929

30+
-- Fast implementation to check if a table is a list
31+
---@param t table
32+
function M.is_list(t)
33+
local i = 0
34+
---@diagnostic disable-next-line: no-unknown
35+
for _ in pairs(t) do
36+
i = i + 1
37+
if t[i] == nil then
38+
return false
39+
end
40+
end
41+
return true
42+
end
43+
3044
function M.very_lazy()
3145
local function _load()
3246
vim.defer_fn(function()

lua/lazy/init.lua

+3-16
Original file line numberDiff line numberDiff line change
@@ -36,21 +36,7 @@ function M.setup(opts)
3636
Util.track()
3737

3838
Util.track("state")
39-
if not State.load() then
40-
local Plugin = require("lazy.plugin")
41-
vim.schedule(function()
42-
Util.info("Reloading...")
43-
end)
44-
Util.track("reload")
45-
Plugin.reload()
46-
Util.track()
47-
-- if not Config.plugins.lazy then
48-
-- Plugin.plugin({
49-
-- "folke/lazy.nvim",
50-
-- opt = false,
51-
-- })
52-
-- end
53-
end
39+
State.load()
5440
Util.track()
5541

5642
Util.track("install")
@@ -59,6 +45,7 @@ function M.setup(opts)
5945
vim.cmd("do User LazyInstallPre")
6046
require("lazy.manager").install({
6147
wait = true,
48+
show = Config.options.interactive,
6249
})
6350
break
6451
end
@@ -75,7 +62,7 @@ function M.setup(opts)
7562

7663
Loader.init_plugins()
7764

78-
Config.plugins.lazy.loaded.time = lazy_delta
65+
Config.plugins["lazy.nvim"].loaded.time = lazy_delta
7966
done = true
8067

8168
vim.cmd("do User LazyDone")

lua/lazy/manager.lua

-5
Original file line numberDiff line numberDiff line change
@@ -66,11 +66,6 @@ function M.run(operation, opts, filter)
6666
runner:wait(on_done)
6767
end)
6868

69-
-- auto show if there are tasks running
70-
if opts.show == nil then
71-
require("lazy.view").show()
72-
end
73-
7469
if opts.wait then
7570
runner:wait()
7671
end

0 commit comments

Comments
 (0)