Skip to content

Commit 6fe425c

Browse files
committed
perf: caching strategy is now configurable
1 parent ae379a6 commit 6fe425c

File tree

3 files changed

+89
-29
lines changed

3 files changed

+89
-29
lines changed

lua/lazy/core/module.lua lua/lazy/core/cache.lua

+69-26
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,19 @@ local uv = vim.loop
55
local M = {}
66
M.dirty = false
77

8-
local cache_path = vim.fn.stdpath("state") .. "/lazy.state"
8+
---@class LazyCacheConfig
9+
M.config = {
10+
enabled = true,
11+
path = vim.fn.stdpath("state") .. "/lazy.state",
12+
-- choose what should be cached
13+
-- * lazy: cache all lazy.nvim core modules and your config files
14+
-- * init: all of the above and any module needed to init your plugins
15+
-- * VimEnter: any module till VimEnter
16+
-- * VeryLazy: any module till VeryLazy
17+
-- * allthethings: all mdules. Not recommended
18+
strategy = "VimEnter", ---@type "lazy"|"init"|"VimEnter"|"allthethings"
19+
}
20+
921
---@type CacheHash
1022
local cache_hash
1123

@@ -27,13 +39,25 @@ function M.check_load(modname, modpath)
2739
require("lazy.core.loader").autoload(modname, modpath)
2840
end
2941

30-
---@param modname string
31-
---@return any
32-
function M.loader(modname)
42+
---@param step? string
43+
function M.disable(step)
3344
if not M.enabled then
34-
return "lazy loader is disabled"
45+
return
46+
end
47+
if step and M.config.strategy ~= step then
48+
return
3549
end
3650

51+
local idx = M.idx()
52+
if idx then
53+
table.remove(package.loaders, idx)
54+
end
55+
M.enabled = false
56+
end
57+
58+
---@param modname string
59+
---@return any
60+
function M.loader(modname)
3761
local entry = M.cache[modname]
3862

3963
local chunk, err
@@ -68,11 +92,10 @@ function M.loader(modname)
6892
return chunk or error(err)
6993
end
7094

71-
---@param modname string
72-
function M.find(modname)
95+
function M.idx()
7396
-- update our loader position if needed
7497
if package.loaders[M.loader_idx] ~= M.loader then
75-
M.loader_idx = 1
98+
M.loader_idx = nil
7699
---@diagnostic disable-next-line: no-unknown
77100
for i, loader in ipairs(package.loaders) do
78101
if loader == M.loader then
@@ -81,29 +104,49 @@ function M.find(modname)
81104
end
82105
end
83106
end
107+
return M.loader_idx
108+
end
84109

85-
-- find the module and its modpath
86-
for i = M.loader_idx + 1, #package.loaders do
87-
---@diagnostic disable-next-line: no-unknown
88-
local chunk = package.loaders[i](modname)
89-
if type(chunk) == "function" then
90-
local info = debug.getinfo(chunk, "S")
91-
return chunk, (info.what ~= "C" and info.source:sub(2))
110+
---@param modname string
111+
function M.find(modname)
112+
if M.idx() then
113+
-- find the module and its modpath
114+
for i = M.loader_idx + 1, #package.loaders do
115+
---@diagnostic disable-next-line: no-unknown
116+
local chunk = package.loaders[i](modname)
117+
if type(chunk) == "function" then
118+
local info = debug.getinfo(chunk, "S")
119+
return chunk, (info.what ~= "C" and info.source:sub(2))
120+
end
92121
end
93122
end
94123
end
95124

96-
function M.setup()
125+
---@param opts? LazyConfig
126+
function M.setup(opts)
127+
-- no fancy deep extend here. just set the options
128+
if opts and opts.performance and opts.performance.cache then
129+
for k, v in pairs(opts.performance.cache) do
130+
M.config[k] = v
131+
end
132+
end
133+
97134
M.load_cache()
98135
table.insert(package.loaders, M.loader_idx, M.loader)
99136

100-
vim.api.nvim_create_autocmd("VimEnter", {
101-
once = true,
102-
callback = function()
103-
-- startup done, so stop caching
104-
M.enabled = false
105-
end,
106-
})
137+
if M.config.strategy == "VimEnter" then
138+
vim.api.nvim_create_autocmd("VimEnter", {
139+
once = true,
140+
callback = function()
141+
-- use schedule so all other VimEnter handlers will have run
142+
vim.schedule(function()
143+
-- startup done, so stop caching
144+
M.disable()
145+
end)
146+
end,
147+
})
148+
end
149+
return M
107150
end
108151

109152
---@return CacheHash?
@@ -118,7 +161,7 @@ function M.eq(h1, h2)
118161
end
119162

120163
function M.save_cache()
121-
local f = assert(uv.fs_open(cache_path, "w", 438))
164+
local f = assert(uv.fs_open(M.config.path, "w", 438))
122165
for modname, entry in pairs(M.cache) do
123166
if entry.used > os.time() - M.ttl then
124167
entry.modname = modname
@@ -142,7 +185,7 @@ end
142185

143186
function M.load_cache()
144187
M.cache = {}
145-
local f = uv.fs_open(cache_path, "r", 438)
188+
local f = uv.fs_open(M.config.path, "r", 438)
146189
if f then
147190
cache_hash = uv.fs_fstat(f) --[[@as CacheHash]]
148191
local data = uv.fs_read(f, cache_hash.size, 0) --[[@as string]]
@@ -172,7 +215,7 @@ function M.autosave()
172215
vim.api.nvim_create_autocmd("VimLeavePre", {
173216
callback = function()
174217
if M.dirty then
175-
local hash = M.hash(cache_path)
218+
local hash = M.hash(M.config.path)
176219
-- abort when the file was changed in the meantime
177220
if hash == nil or M.eq(cache_hash, hash) then
178221
M.save_cache()

lua/lazy/core/config.lua

+6-1
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,10 @@ M.defaults = {
4848
},
4949
throttle = 20, -- how frequently should the ui process render events
5050
},
51+
performance = {
52+
---@type LazyCacheConfig
53+
cache = nil,
54+
},
5155
}
5256

5357
M.ns = vim.api.nvim_create_namespace("lazy")
@@ -72,6 +76,7 @@ M.options = {}
7276
function M.setup(spec, opts)
7377
M.spec = spec
7478
M.options = vim.tbl_deep_extend("force", M.defaults, opts or {})
79+
M.options.performance.cache = require("lazy.core.cache")
7580
M.root = M.options.package.path .. "/pack/" .. M.options.package.name .. "/opt"
7681

7782
if M.options.package.reset then
@@ -84,7 +89,7 @@ function M.setup(spec, opts)
8489
pattern = "VeryLazy",
8590
once = true,
8691
callback = function()
87-
require("lazy.core.module").autosave()
92+
require("lazy.core.cache").autosave()
8893
require("lazy.view").setup()
8994
end,
9095
})

lua/lazy/init.lua

+14-2
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,12 @@ function M.setup(spec, opts)
88
end
99
local start = vim.loop.hrtime()
1010

11-
-- load module cache before anything else
12-
require("lazy.core.module").setup()
11+
local Cache
12+
if not (opts and opts.performance and opts.performance.cache and opts.performance.cache.enabled == false) then
13+
-- load module cache before anything else
14+
Cache = require("lazy.core.cache").setup(opts)
15+
end
16+
1317
local Util = require("lazy.core.util")
1418
local Config = require("lazy.core.config")
1519
local Loader = require("lazy.core.loader")
@@ -36,9 +40,17 @@ function M.setup(spec, opts)
3640
Config.plugins["lazy.nvim"]._.loaded = { time = delta, source = "init.lua" }
3741
end
3842

43+
if Cache then
44+
Cache.disable("lazy")
45+
end
46+
3947
-- load plugins with lazy=false or Plugin.init
4048
Loader.init_plugins()
4149

50+
if Cache then
51+
Cache.disable("init")
52+
end
53+
4254
-- all done!
4355
vim.cmd("do User LazyDone")
4456
end

0 commit comments

Comments
 (0)