Skip to content

Commit 4326d4b

Browse files
committed
fix(pkg): correctly pre-load package specs and remove them when needed during resolve
1 parent ee2ca39 commit 4326d4b

File tree

6 files changed

+87
-38
lines changed

6 files changed

+87
-38
lines changed

lua/lazy/core/meta.lua

+42
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
local Config = require("lazy.core.config")
2+
local Pkg = require("lazy.pkg")
23
local Util = require("lazy.core.util")
34

45
--- This class is used to manage the plugins.
@@ -10,6 +11,7 @@ local Util = require("lazy.core.util")
1011
---@field dirty table<string, boolean>
1112
---@field spec LazySpecLoader
1213
---@field fragments LazyFragments
14+
---@field pkgs table<string, number>
1315
local M = {}
1416

1517
---@param spec LazySpecLoader
@@ -22,9 +24,30 @@ function M.new(spec)
2224
self.frag_to_meta = {}
2325
self.str_to_meta = {}
2426
self.dirty = {}
27+
self.pkgs = {}
2528
return self
2629
end
2730

31+
-- import package specs
32+
function M:load_pkgs()
33+
if not Config.options.pkg.enabled then
34+
return
35+
end
36+
local specs = Pkg.spec()
37+
for dir, spec in pairs(specs) do
38+
local meta, fragment = self:add(spec)
39+
if meta and fragment then
40+
-- tag all package fragments as optional
41+
for _, fid in ipairs(meta._.frags) do
42+
local frag = self.fragments:get(fid)
43+
frag.spec.optional = true
44+
end
45+
-- keep track of the top-level package fragment
46+
self.pkgs[dir] = fragment.id
47+
end
48+
end
49+
end
50+
2851
--- Remove a plugin and all its fragments.
2952
---@param name string
3053
function M:del(name)
@@ -88,6 +111,7 @@ function M:add(plugin)
88111
self.plugins[meta.name] = meta
89112
self.frag_to_meta[fragment.id] = meta
90113
self.dirty[meta.name] = true
114+
return meta, fragment
91115
end
92116

93117
--- Rebuild all plugins based on dirty fragments,
@@ -101,6 +125,7 @@ function M:rebuild()
101125
self.dirty[meta.name] = true
102126
else
103127
-- fragment was deleted, so remove it from plugin
128+
self.frag_to_meta[fid] = nil
104129
---@param f number
105130
meta._.frags = vim.tbl_filter(function(f)
106131
return f ~= fid
@@ -260,11 +285,28 @@ function M:fix_disabled()
260285
return changes
261286
end
262287

288+
--- Removes package fragments for plugins that no longer use the same directory.
289+
function M:fix_pkgs()
290+
for dir, fid in pairs(self.pkgs) do
291+
local plugin = self.frag_to_meta[fid]
292+
plugin = plugin and self.plugins[plugin.name]
293+
if plugin then
294+
-- check if plugin is still in the same directory
295+
if plugin.dir ~= dir then
296+
self.fragments:del(fid)
297+
end
298+
end
299+
end
300+
self:rebuild()
301+
end
302+
263303
--- Resolve all plugins, based on cond, enabled and optional.
264304
function M:resolve()
265305
Util.track("resolve plugins")
266306
self:rebuild()
267307

308+
self:fix_pkgs()
309+
268310
self:fix_cond()
269311

270312
-- selene: allow(empty_loop)

lua/lazy/core/plugin.lua

+6-27
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
local Config = require("lazy.core.config")
22
local Meta = require("lazy.core.meta")
3-
local Pkg = require("lazy.pkg")
43
local Util = require("lazy.core.util")
54

65
---@class LazyCorePlugin
@@ -16,7 +15,6 @@ M.loading = false
1615
---@field notifs {msg:string, level:number, file?:string}[]
1716
---@field importing? string
1817
---@field optional? boolean
19-
---@field pkgs table<string, boolean>
2018
local Spec = {}
2119
M.Spec = Spec
2220
M.LOCAL_SPEC = ".lazy.lua"
@@ -30,8 +28,8 @@ function Spec.new(spec, opts)
3028
self.modules = {}
3129
self.notifs = {}
3230
self.ignore_installed = {}
33-
self.pkgs = {}
3431
self.optional = opts and opts.optional
32+
self.meta:load_pkgs()
3533
if spec then
3634
self:parse(spec)
3735
end
@@ -62,25 +60,6 @@ function Spec.get_name(pkg)
6260
return slash and name:sub(#name - slash + 2) or pkg:gsub("%W+", "_")
6361
end
6462

65-
---@param plugin LazyPluginSpec
66-
function Spec:add(plugin)
67-
self.meta:add(plugin)
68-
69-
---@diagnostic disable-next-line: cast-type-mismatch
70-
---@cast plugin LazyPlugin
71-
72-
-- import the plugin's spec
73-
if Config.options.pkg.enabled and plugin.dir and not self.pkgs[plugin.dir] then
74-
self.pkgs[plugin.dir] = true
75-
local pkg = Pkg.get_spec(plugin)
76-
if pkg then
77-
self:normalize(pkg)
78-
end
79-
end
80-
81-
return plugin
82-
end
83-
8463
function Spec:error(msg)
8564
self:log(msg, vim.log.levels.ERROR)
8665
end
@@ -110,19 +89,19 @@ end
11089
---@param spec LazySpec|LazySpecImport
11190
function Spec:normalize(spec)
11291
if type(spec) == "string" then
113-
self:add({ spec })
92+
self.meta:add({ spec })
11493
elseif #spec > 1 or Util.is_list(spec) then
11594
---@cast spec LazySpec[]
11695
for _, s in ipairs(spec) do
11796
self:normalize(s)
11897
end
11998
elseif spec[1] or spec.dir or spec.url then
12099
---@cast spec LazyPluginSpec
121-
local plugin = self:add(spec)
100+
self.meta:add(spec)
122101
---@diagnostic disable-next-line: cast-type-mismatch
123-
---@cast plugin LazySpecImport
124-
if plugin and plugin.import then
125-
self:import(plugin)
102+
---@cast spec LazySpecImport
103+
if spec and spec.import then
104+
self:import(spec)
126105
end
127106
elseif spec.import then
128107
---@cast spec LazySpecImport

lua/lazy/pkg/init.lua

+21
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ local M = {}
77

88
---@class LazyPkg
99
---@field source string
10+
---@field name string
1011
---@field file string
1112
---@field spec? LazySpec
1213
---@field chunk? string|fun():LazySpec
@@ -30,6 +31,7 @@ function M.update()
3031
for _, source in ipairs(sources) do
3132
local spec = source.get(plugin)
3233
if spec then
34+
spec.name = plugin.name
3335
if type(spec.chunk) == "function" then
3436
spec.chunk = string.dump(spec.chunk)
3537
end
@@ -81,6 +83,25 @@ function M.get(plugin)
8183
return ret
8284
end
8385

86+
function M.spec()
87+
if not M.cache then
88+
_load()
89+
end
90+
---@type table<string,LazyPluginSpec>
91+
local ret = {}
92+
93+
for dir in pairs(M.cache) do
94+
local pkg = M.get({ dir = dir })
95+
local spec = pkg and pkg.spec
96+
if pkg and spec then
97+
spec = type(spec) == "table" and vim.deepcopy(spec) or spec
98+
ret[dir] = { pkg.name, specs = spec }
99+
end
100+
end
101+
102+
return ret
103+
end
104+
84105
---@param plugin LazyPlugin
85106
---@return LazySpec?
86107
function M.get_spec(plugin)

lua/lazy/pkg/rockspec.lua

+14-11
Original file line numberDiff line numberDiff line change
@@ -29,24 +29,27 @@ function M.get(plugin)
2929
end
3030

3131
---@type RockSpec?
32+
---@diagnostic disable-next-line: missing-fields
3233
local rockspec = {}
33-
local ret, err = loadfile(rockspec_file, "t", rockspec)
34-
if not ret then
34+
local load, err = loadfile(rockspec_file, "t", rockspec)
35+
if not load then
3536
error(err)
3637
end
37-
ret()
38-
return rockspec
39-
and rockspec.package
38+
load()
39+
40+
---@param dep string
41+
local rocks = vim.tbl_filter(function(dep)
42+
local name = dep:gsub("%s.*", "")
43+
return not vim.tbl_contains(M.skip, name)
44+
end, rockspec and rockspec.dependencies or {})
45+
46+
return #rocks > 0
4047
and {
4148
source = "rockspec",
4249
file = rockspec_file,
4350
spec = {
44-
dir = plugin.dir,
45-
url = plugin.url,
46-
rocks = vim.tbl_filter(function(dep)
47-
local name = dep:gsub("%s.*", "")
48-
return not vim.tbl_contains(M.skip, name)
49-
end, rockspec.dependencies),
51+
plugin.name,
52+
rocks = rocks,
5053
},
5154
}
5255
or nil

lua/lazy/types.lua

+1
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,7 @@
8686

8787
---@class LazyFragment
8888
---@field id number
89+
---@field pkg? boolean
8990
---@field pid? number
9091
---@field deps? number[]
9192
---@field frags? number[]

selene.toml

+3
Original file line numberDiff line numberDiff line change
@@ -1 +1,4 @@
11
std="vim"
2+
3+
[lints]
4+
mixed_table="allow"

0 commit comments

Comments
 (0)