Skip to content

Commit 8790070

Browse files
committed
perf: further optims to loading and caching specs. dont cache specs with plugin that have init or in start with config
1 parent 413dd5b commit 8790070

File tree

1 file changed

+27
-40
lines changed

1 file changed

+27
-40
lines changed

lua/lazy/core/plugin.lua

+27-40
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ local funs = { config = true, init = true, run = true }
1515
---@field uri string
1616
---@field branch? string
1717
---@field dir string
18-
---@field enabled? boolean
18+
---@field enabled? boolean|(fun():boolean)
1919
---@field opt? boolean
2020
---@field init? fun(LazyPlugin) Will always be run
2121
---@field config? fun(LazyPlugin) Will be executed when loading the plugin
@@ -36,6 +36,7 @@ local funs = { config = true, init = true, run = true }
3636
---@field modname string
3737
---@field modpath string
3838
---@field plugins table<string, LazyPlugin>
39+
---@field funs? table<string, string[]>
3940
local Spec = {}
4041

4142
---@param modname string
@@ -58,8 +59,9 @@ function Spec:add(plugin)
5859
Util.error("Invalid plugin spec " .. vim.inspect(plugin))
5960
end
6061
plugin.uri = plugin.uri or ("https://github.com/" .. plugin[1] .. ".git")
62+
63+
-- PERF: optimized code to get package name without using lua patterns
6164
if not plugin.name then
62-
-- PERF: optimized code to get package name without using lua patterns
6365
local name = plugin[1]:sub(-4) == ".git" and plugin[1]:sub(1, -5) or plugin[1]
6466
local slash = name:reverse():find("/", 1, true) --[[@as number?]]
6567
plugin.name = slash and name:sub(#name - slash + 2) or plugin[1]:gsub("%W+", "_")
@@ -82,21 +84,22 @@ function Spec:normalize(spec, results)
8284
for _, s in ipairs(spec) do
8385
self:normalize(s, results)
8486
end
85-
elseif spec.enabled ~= false then
87+
elseif spec.enabled == nil or spec.enabled == true or (type(spec.enabled) == "function" and spec.enabled()) then
8688
local plugin = self:add(spec)
8789
plugin.requires = plugin.requires and self:normalize(plugin.requires, {}) or nil
8890
table.insert(results, plugin.name)
8991
end
9092
return results
9193
end
9294

93-
---@param spec CachedSpec
95+
---@param spec LazySpec
9496
function Spec.revive(spec)
9597
if spec.funs then
9698
---@type LazySpec
9799
local loaded = nil
98100
for fun, plugins in pairs(spec.funs) do
99101
for _, name in pairs(plugins) do
102+
---@diagnostic disable-next-line: no-unknown
100103
spec.plugins[name][fun] = function(...)
101104
loaded = loaded or Spec.load(spec.modname, spec.modpath)
102105
return loaded.plugins[name][fun](...)
@@ -111,7 +114,7 @@ function M.update_state(check_clean)
111114
---@type table<"opt"|"start", table<string,boolean>>
112115
local installed = { opt = {}, start = {} }
113116
for opt, packs in pairs(installed) do
114-
Util.scandir(Config.options.package_path .. "/" .. opt, function(_, name, type)
117+
Util.ls(Config.options.package_path .. "/" .. opt, function(_, name, type)
115118
if type == "directory" or type == "link" then
116119
packs[name] = true
117120
end
@@ -153,26 +156,22 @@ function M.process_local(plugin)
153156
end
154157
end
155158

156-
---@alias LazySpecLoader fun(modname:string, modpath:string):LazySpec
157-
---@param loader? LazySpecLoader
158-
function M.specs(loader)
159-
loader = loader or Spec.load
159+
---@param cache? table<string,LazySpec>
160+
function M.specs(cache)
160161
---@type LazySpec[]
161162
local specs = {}
162163

163-
local function _load(modname, modpath)
164-
local spec = Util.try(function()
165-
return loader(modname, modpath)
166-
end, "Failed to load **" .. modname .. "**")
167-
if spec then
164+
local function _load(name, modpath)
165+
local modname = Config.options.plugins .. (name and ("." .. name) or "")
166+
Util.try(function()
167+
local spec = cache and cache[modname]
168+
spec = spec and not Module.is_dirty(modname, modpath) and Spec.revive(spec) or Spec.load(modname, modpath)
168169
table.insert(specs, spec)
169-
end
170+
end, "Failed to load **" .. modname .. "**")
170171
end
171172

172-
_load(Config.options.plugins, Config.paths.main)
173-
Util.lsmod(Config.paths.plugins, function(name, modpath)
174-
_load(Config.options.plugins .. "." .. name, modpath)
175-
end)
173+
_load(nil, Config.paths.main)
174+
Util.lsmod(Config.paths.plugins, _load)
176175
return specs
177176
end
178177

@@ -186,22 +185,9 @@ function M.load()
186185
state = nil
187186
end
188187

189-
local loader = state
190-
and function(modname, modpath)
191-
local spec = state and state.specs[modname]
192-
if (not spec) or Module.is_dirty(modname, modpath) then
193-
dirty = true
194-
vim.schedule(function()
195-
vim.notify("Reloading " .. modname)
196-
end)
197-
return Spec.load(modname, modpath)
198-
end
199-
return Spec.revive(spec)
200-
end
201-
202188
-- load specs
203189
Util.track("specs")
204-
local specs = M.specs(loader)
190+
local specs = M.specs(state and state.specs)
205191
Util.track()
206192

207193
-- merge
@@ -225,24 +211,25 @@ function M.load()
225211
end
226212

227213
function M.save()
228-
if not M.dirty then
229-
return
230-
end
231-
232-
---@alias CachedSpec LazySpec|{funs:table<string, string[]>}
233214
---@class LazyState
234215
local state = {
235-
---@type table<string, CachedSpec>
216+
---@type table<string, LazySpec>
236217
specs = {},
237218
loaders = require("lazy.core.loader").loaders,
238219
config = Config.options,
239220
}
240221

241222
for _, spec in ipairs(M.specs()) do
242-
---@cast spec CachedSpec
243223
spec.funs = {}
244224
state.specs[spec.modname] = spec
245225
for _, plugin in pairs(spec.plugins) do
226+
if plugin.init or (plugin.opt == false and plugin.config) then
227+
-- no use in caching specs that need init,
228+
-- or specs that are in start and have a config,
229+
-- since we'll load the real spec during startup anyway
230+
state.specs[spec.modname] = nil
231+
break
232+
end
246233
---@cast plugin CachedPlugin
247234
for k, v in pairs(plugin) do
248235
if type(v) == "function" then

0 commit comments

Comments
 (0)