Skip to content

Commit 1446f6c

Browse files
committed
perf: minimize meta rebuild when loading specs
1 parent 972baa6 commit 1446f6c

File tree

4 files changed

+67
-13
lines changed

4 files changed

+67
-13
lines changed

lua/lazy/core/loader.lua

+8-2
Original file line numberDiff line numberDiff line change
@@ -524,7 +524,7 @@ function M.colorscheme(name)
524524
end
525525

526526
function M.auto_load(modname, modpath)
527-
local plugin = Plugin.find(modpath)
527+
local plugin = Plugin.find(modpath, { fast = not M.did_handlers })
528528
if plugin and modpath:find(plugin.dir, 1, true) == 1 then
529529
plugin._.rtp_loaded = true
530530
-- don't load if:
@@ -545,8 +545,14 @@ end
545545

546546
---@param modname string
547547
function M.loader(modname)
548-
local paths = Util.get_unloaded_rtp(modname)
548+
local paths, cached = Util.get_unloaded_rtp(modname, { cache = true })
549549
local ret = Cache.find(modname, { rtp = false, paths = paths })[1]
550+
551+
if not ret and cached then
552+
paths = Util.get_unloaded_rtp(modname)
553+
ret = Cache.find(modname, { rtp = false, paths = paths })[1]
554+
end
555+
550556
if ret then
551557
M.auto_load(modname, ret.modpath)
552558
local mod = package.loaded[modname]

lua/lazy/core/meta.lua

+15-1
Original file line numberDiff line numberDiff line change
@@ -117,6 +117,14 @@ end
117117
--- Rebuild all plugins based on dirty fragments,
118118
--- or dirty plugins. Will remove plugins that no longer have fragments.
119119
function M:rebuild()
120+
local frag_count = vim.tbl_count(self.fragments.dirty)
121+
local plugin_count = vim.tbl_count(self.dirty)
122+
if frag_count == 0 and plugin_count == 0 then
123+
return
124+
end
125+
if Config.options.debug then
126+
Util.track("rebuild plugins frags=" .. frag_count .. " plugins=" .. plugin_count)
127+
end
120128
for fid in pairs(self.fragments.dirty) do
121129
local meta = self.frag_to_meta[fid]
122130
if meta then
@@ -143,13 +151,20 @@ function M:rebuild()
143151
for n, _ in pairs(self.dirty) do
144152
self:_rebuild(n)
145153
end
154+
if Config.options.debug then
155+
Util.track()
156+
end
146157
end
147158

148159
--- Rebuild a single plugin.
149160
--- This will resolve the plugin based on its fragments using metatables.
150161
--- This also resolves dependencies, dep, optional, dir, dev, and url.
151162
---@param name string
152163
function M:_rebuild(name)
164+
if not self.dirty[name] then
165+
return
166+
end
167+
self.dirty[name] = nil
153168
local plugin = self.plugins[name]
154169
if not plugin or #plugin._.frags == 0 then
155170
self.plugins[name] = nil
@@ -225,7 +240,6 @@ function M:_rebuild(name)
225240

226241
setmetatable(plugin, { __index = super })
227242

228-
self.dirty[plugin.name] = nil
229243
return plugin
230244
end
231245

lua/lazy/core/plugin.lua

+9-2
Original file line numberDiff line numberDiff line change
@@ -362,17 +362,24 @@ end
362362

363363
-- Finds the plugin that has this path
364364
---@param path string
365-
function M.find(path)
365+
---@param opts? {fast?:boolean}
366+
function M.find(path, opts)
366367
if not Config.spec then
367368
return
368369
end
370+
opts = opts or {}
369371
local lua = path:find("/lua/", 1, true)
370372
if lua then
371373
local name = path:sub(1, lua - 1)
372374
local slash = name:reverse():find("/", 1, true)
373375
if slash then
374376
name = name:sub(#name - slash + 2)
375-
return name and Config.plugins[name] or Config.spec.plugins[name] or nil
377+
if name then
378+
if opts.fast then
379+
return Config.spec.meta.plugins[name]
380+
end
381+
return Config.spec.plugins[name]
382+
end
376383
end
377384
end
378385
end

lua/lazy/core/util.lua

+35-8
Original file line numberDiff line numberDiff line change
@@ -238,34 +238,61 @@ function M.walkmods(root, fn, modname)
238238
end
239239

240240
---@param modname string
241-
function M.get_unloaded_rtp(modname)
242-
modname = modname:gsub("/", ".")
243-
local idx = modname:find(".", 1, true)
244-
local topmod = idx and modname:sub(1, idx - 1) or modname
245-
topmod = M.normname(topmod)
241+
---@return string
242+
function M.topmod(modname)
243+
return modname:match("^[^./]+") or modname
244+
end
245+
246+
---@type table<string, string[]>
247+
M.unloaded_cache = {}
248+
249+
---@param modname string
250+
---@param opts? {cache?:boolean}
251+
function M.get_unloaded_rtp(modname, opts)
252+
opts = opts or {}
246253

254+
local topmod = M.topmod(modname)
255+
if opts.cache and M.unloaded_cache[topmod] then
256+
return M.unloaded_cache[topmod], true
257+
end
258+
259+
local norm = M.normname(topmod)
260+
261+
---@type string[]
247262
local rtp = {}
248263
local Config = require("lazy.core.config")
249264
if Config.spec then
250265
for _, plugin in pairs(Config.spec.plugins) do
251266
if not (plugin._.loaded or plugin.module == false) then
252-
if topmod == M.normname(plugin.name) then
267+
if norm == M.normname(plugin.name) then
253268
table.insert(rtp, 1, plugin.dir)
254269
else
255270
table.insert(rtp, plugin.dir)
256271
end
257272
end
258273
end
259274
end
260-
return rtp
275+
M.unloaded_cache[topmod] = rtp
276+
return rtp, false
261277
end
262278

263279
function M.find_root(modname)
280+
local paths, cached = M.get_unloaded_rtp(modname, { cache = true })
281+
264282
local ret = require("lazy.core.cache").find(modname, {
265283
rtp = true,
266-
paths = M.get_unloaded_rtp(modname),
284+
paths = paths,
267285
patterns = { "", ".lua" },
268286
})[1]
287+
288+
if not ret and cached then
289+
paths = M.get_unloaded_rtp(modname)
290+
ret = require("lazy.core.cache").find(modname, {
291+
rtp = false,
292+
paths = paths,
293+
patterns = { "", ".lua" },
294+
})[1]
295+
end
269296
if ret then
270297
local root = ret.modpath:gsub("/init%.lua$", ""):gsub("%.lua$", "")
271298
return root

0 commit comments

Comments
 (0)