Skip to content

Commit dbffad6

Browse files
committed
fix(fragments): prevent adding the same spec instance more than once
1 parent fd04bc6 commit dbffad6

File tree

5 files changed

+65
-37
lines changed

5 files changed

+65
-37
lines changed

lua/lazy/core/fragments.lua

+24-16
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,5 @@
11
local Config = require("lazy.core.config")
2-
3-
local M = {}
4-
5-
M._fid = 0
6-
7-
local function next_id()
8-
M._fid = M._fid + 1
9-
return M._fid
10-
end
2+
local Util = require("lazy.core.util")
113

124
--- This class is used to manage the fragments of a plugin spec.
135
--- It keeps track of the fragments and their relations to other fragments.
@@ -17,30 +9,39 @@ end
179
---@field frag_stack number[]
1810
---@field dep_stack number[]
1911
---@field dirty table<number, boolean>
12+
---@field plugins table<LazyPlugin, number>
2013
---@field spec LazySpecLoader
21-
local F = {}
14+
local M = {}
15+
16+
M._fid = 0
17+
18+
local function next_id()
19+
M._fid = M._fid + 1
20+
return M._fid
21+
end
2222

2323
---@param spec LazySpecLoader
2424
---@return LazyFragments
2525
function M.new(spec)
26-
local self = setmetatable({}, { __index = F })
26+
local self = setmetatable({}, { __index = M })
2727
self.fragments = {}
2828
self.frag_stack = {}
2929
self.dep_stack = {}
3030
self.spec = spec
3131
self.dirty = {}
32+
self.plugins = {}
3233
return self
3334
end
3435

3536
---@param id number
36-
function F:get(id)
37+
function M:get(id)
3738
return self.fragments[id]
3839
end
3940

4041
--- Remove a fragment and all its children.
4142
--- This will also remove the fragment from its parent's children list.
4243
---@param id number
43-
function F:del(id)
44+
function M:del(id)
4445
-- del fragment
4546
local fragment = self.fragments[id]
4647
if not fragment then
@@ -55,13 +56,13 @@ function F:del(id)
5556
local parent = self.fragments[pid]
5657
if parent.frags then
5758
---@param fid number
58-
parent.frags = vim.tbl_filter(function(fid)
59+
parent.frags = Util.filter(function(fid)
5960
return fid ~= id
6061
end, parent.frags)
6162
end
6263
if parent.deps then
6364
---@param fid number
64-
parent.deps = vim.tbl_filter(function(fid)
65+
parent.deps = Util.filter(function(fid)
6566
return fid ~= id
6667
end, parent.deps)
6768
end
@@ -81,8 +82,15 @@ end
8182
--- Add a fragment to the fragments list.
8283
--- This also resolves its name, url, dir, dependencies and child specs.
8384
---@param plugin LazyPluginSpec
84-
function F:add(plugin)
85+
function M:add(plugin)
86+
if self.plugins[plugin] then
87+
return self.fragments[self.plugins[plugin]]
88+
end
89+
8590
local id = next_id()
91+
setmetatable(plugin, nil)
92+
93+
self.plugins[plugin] = id
8694

8795
local pid = self.frag_stack[#self.frag_stack]
8896

lua/lazy/core/meta.lua

+6-7
Original file line numberDiff line numberDiff line change
@@ -33,8 +33,7 @@ function M:load_pkgs()
3333
if not Config.options.pkg.enabled then
3434
return
3535
end
36-
local specs = Pkg.get()
37-
for dir, pkg in pairs(specs) do
36+
for _, pkg in ipairs(Pkg.get()) do
3837
local meta, fragment = self:add(pkg.spec)
3938
if meta and fragment then
4039
meta._.pkg = pkg
@@ -44,7 +43,7 @@ function M:load_pkgs()
4443
frag.spec.optional = true
4544
end
4645
-- keep track of the top-level package fragment
47-
self.pkgs[dir] = fragment.id
46+
self.pkgs[pkg.dir] = fragment.id
4847
end
4948
end
5049
end
@@ -128,7 +127,7 @@ function M:rebuild()
128127
-- fragment was deleted, so remove it from plugin
129128
self.frag_to_meta[fid] = nil
130129
---@param f number
131-
meta._.frags = vim.tbl_filter(function(f)
130+
meta._.frags = Util.filter(function(f)
132131
return f ~= fid
133132
end, meta._.frags)
134133
-- if no fragments left, delete plugin
@@ -167,10 +166,10 @@ function M:_rebuild(name)
167166
assert(#plugin._.frags > 0, "no fragments found for plugin " .. name)
168167

169168
---@type table<number, boolean>
170-
local done = {}
169+
local added = {}
171170
for _, fid in ipairs(plugin._.frags) do
172-
if not done[fid] then
173-
done[fid] = true
171+
if not added[fid] then
172+
added[fid] = true
174173
local fragment = self.fragments:get(fid)
175174
assert(fragment, "fragment " .. fid .. " not found, for plugin " .. name)
176175
---@diagnostic disable-next-line: no-unknown

lua/lazy/core/plugin.lua

+9-10
Original file line numberDiff line numberDiff line change
@@ -170,10 +170,10 @@ function Spec:import(spec)
170170
self.importing = nil
171171
return self:error(
172172
"Invalid spec module: `"
173-
.. modname
174-
.. "`\nExpected a `table` of specs, but a `"
175-
.. type(mod)
176-
.. "` was returned instead"
173+
.. modname
174+
.. "`\nExpected a `table` of specs, but a `"
175+
.. type(mod)
176+
.. "` was returned instead"
177177
)
178178
end
179179
self:normalize(mod)
@@ -216,11 +216,11 @@ function M.update_state()
216216
plugin._ = plugin._ or {}
217217
if plugin.lazy == nil then
218218
local lazy = plugin._.dep
219-
or Config.options.defaults.lazy
220-
or plugin.event
221-
or plugin.keys
222-
or plugin.ft
223-
or plugin.cmd
219+
or Config.options.defaults.lazy
220+
or plugin.event
221+
or plugin.keys
222+
or plugin.ft
223+
or plugin.cmd
224224
plugin.lazy = lazy and true or false
225225
end
226226
if plugin.dir:find(Config.options.root, 1, true) == 1 then
@@ -342,7 +342,6 @@ function M.load()
342342
Config.plugins[name]._ = plugin._
343343
Config.plugins[name]._.dep = new_state.dep
344344
Config.plugins[name]._.frags = new_state.frags
345-
-- Config.plugins[name]._.tasks = new_state.tasks
346345
end
347346
end
348347
Util.track()

lua/lazy/core/util.lua

+14
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,20 @@ function M.track(data, time)
2828
end
2929
end
3030

31+
---@generic T
32+
---@param list T[]
33+
---@param fn fun(v: T):boolean?
34+
---@return T[]
35+
function M.filter(fn, list)
36+
local ret = {}
37+
for _, v in ipairs(list) do
38+
if fn(v) then
39+
table.insert(ret, v)
40+
end
41+
end
42+
return ret
43+
end
44+
3145
---@generic F: fun()
3246
---@param data (string|{[string]:string})?
3347
---@param fn F

lua/lazy/pkg/init.lua

+12-4
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ M.dirty = false
2525
---@field pkgs LazyPkg[]
2626
---@field version number
2727

28-
---@type table<string, LazyPkg>?
28+
---@type LazyPkg[]?
2929
M.cache = nil
3030

3131
function M.update()
@@ -65,6 +65,9 @@ function M.update()
6565
end
6666
end
6767
end
68+
table.sort(ret.pkgs, function(a, b)
69+
return a.name < b.name
70+
end)
6871
local code = "return " .. Util.dump(ret)
6972
vim.fn.mkdir(vim.fn.fnamemodify(Config.options.pkg.cache, ":h"), "p")
7073
Util.write_file(Config.options.pkg.cache, code)
@@ -91,8 +94,8 @@ local function _load()
9194
end
9295
-- wrap in the scope of the plugin
9396
pkg.spec = { pkg.name, specs = pkg.spec }
94-
M.cache[pkg.dir] = pkg
9597
end
98+
M.cache = ret.pkgs
9699
end
97100
end, "Error loading pkg:")
98101
end
@@ -107,10 +110,15 @@ end
107110

108111
---@param dir string
109112
---@return LazyPkg?
110-
---@overload fun():table<string, LazyPkg>
113+
---@overload fun():LazyPkg[]
111114
function M.get(dir)
112115
if dir then
113-
return M.cache[dir]
116+
for _, pkg in ipairs(M.cache) do
117+
if pkg.dir == dir then
118+
return pkg
119+
end
120+
end
121+
return
114122
end
115123
return M.cache
116124
end

0 commit comments

Comments
 (0)