Skip to content

Commit 908b9ad

Browse files
committed
feat: dependencies are opt=true by default if they only appear as a dep
1 parent 5810635 commit 908b9ad

File tree

6 files changed

+128
-26
lines changed

6 files changed

+128
-26
lines changed

README.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -42,8 +42,9 @@
4242
- [x] move tasks etc to Plugin.state
4343
- [ ] allow setting up plugins through config
4444
- [ ] handlers imply opt
45-
- [ ] dependencies imply opt for deps
45+
- [x] dependencies imply opt for deps
4646
- [x] fix local plugin spec
47+
- [ ] investigate all opt=true. Simplifies logic (easily switch between opt/start afterwards)
4748

4849
## 📦 Differences with Packer
4950

lua/lazy/core/plugin.lua

+56-18
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@ M.dirty = false
3838
---@field name string display name and name used for plugin config files
3939
---@field uri string
4040
---@field dir string
41+
---@field dep? boolean True if this plugin is only in the spec as a dependency
4142
---@field enabled? boolean|(fun():boolean)
4243
---@field opt? boolean
4344
---@field dependencies? string[]
@@ -82,7 +83,8 @@ function Spec.load(modname, modpath)
8283
end
8384

8485
---@param plugin LazyPlugin
85-
function Spec:add(plugin)
86+
---@param is_dep? boolean
87+
function Spec:add(plugin, is_dep)
8688
local pkg = plugin[1]
8789
if type(pkg) ~= "string" then
8890
Util.error("Invalid plugin spec " .. vim.inspect(plugin))
@@ -108,27 +110,30 @@ function Spec:add(plugin)
108110
plugin.name = slash and name:sub(#name - slash + 2) or pkg:gsub("%W+", "_")
109111
end
110112

113+
plugin.dep = is_dep
114+
111115
M.process_local(plugin)
112116
local other = self.plugins[plugin.name]
113-
self.plugins[plugin.name] = other and vim.tbl_extend("force", self.plugins[plugin.name], plugin) or plugin
117+
self.plugins[plugin.name] = other and M.merge(other, plugin) or plugin
114118
return self.plugins[plugin.name]
115119
end
116120

117121
---@param spec LazySpec
118122
---@param results? string[]
119-
function Spec:normalize(spec, results)
123+
---@param is_dep? boolean
124+
function Spec:normalize(spec, results, is_dep)
120125
results = results or {}
121126
if type(spec) == "string" then
122-
table.insert(results, self:add({ spec }).name)
127+
table.insert(results, self:add({ spec }, is_dep).name)
123128
elseif #spec > 1 or Util.is_list(spec) then
124129
---@cast spec LazySpec[]
125130
for _, s in ipairs(spec) do
126-
self:normalize(s, results)
131+
self:normalize(s, results, is_dep)
127132
end
128133
elseif spec.enabled == nil or spec.enabled == true or (type(spec.enabled) == "function" and spec.enabled()) then
129134
---@cast spec LazyPlugin
130-
local plugin = self:add(spec)
131-
plugin.dependencies = plugin.dependencies and self:normalize(plugin.dependencies, {}) or nil
135+
local plugin = self:add(spec, is_dep)
136+
plugin.dependencies = plugin.dependencies and self:normalize(plugin.dependencies, {}, true) or nil
132137
table.insert(results, plugin.name)
133138
end
134139
return results
@@ -152,22 +157,55 @@ function Spec.revive(spec)
152157
return spec
153158
end
154159

155-
function M.update_state(check_clean)
160+
---@param old LazyPlugin
161+
---@param new LazyPlugin
162+
---@return LazyPlugin
163+
function M.merge(old, new)
164+
local is_dep = old.dep and new.dep
165+
166+
local Handler = require("lazy.core.handler")
167+
---@diagnostic disable-next-line: no-unknown
168+
for k, v in pairs(new) do
169+
if k == "dep" then
170+
elseif old[k] ~= nil and old[k] ~= v then
171+
if Handler.handlers[k] then
172+
local values = type(v) == "string" and { v } or v
173+
vim.list_extend(values, type(old[k]) == "string" and { old[k] } or old[k])
174+
---@diagnostic disable-next-line: no-unknown
175+
old[k] = values
176+
else
177+
error("Merging plugins is not supported for key `" .. k .. "`")
178+
end
179+
else
180+
---@diagnostic disable-next-line: no-unknown
181+
old[k] = v
182+
end
183+
end
184+
old.dep = is_dep
185+
return old
186+
end
187+
188+
---@param opts? {clean:boolean, installed:boolean, plugins?: LazyPlugin[]}
189+
function M.update_state(opts)
190+
opts = opts or {}
191+
156192
---@type table<"opt"|"start", table<string,FileType>>
157193
local installed = { opt = {}, start = {} }
158-
for opt, packs in pairs(installed) do
159-
Util.ls(Config.options.packpath .. "/" .. opt, function(_, name, type)
160-
if type == "directory" or type == "link" then
161-
packs[name] = type
162-
end
163-
end)
194+
if opts.installed ~= false then
195+
for opt, packs in pairs(installed) do
196+
Util.ls(Config.options.packpath .. "/" .. opt, function(_, name, type)
197+
if type == "directory" or type == "link" then
198+
packs[name] = type
199+
end
200+
end)
201+
end
164202
end
165203

166-
for _, plugin in pairs(Config.plugins) do
204+
for _, plugin in pairs(opts.plugins or Config.plugins) do
167205
plugin._ = plugin._ or {}
168206
plugin[1] = plugin["1"] or plugin[1]
169207
if plugin.opt == nil then
170-
plugin.opt = Config.options.opt
208+
plugin.opt = plugin.dep or Config.options.opt
171209
end
172210
local opt = plugin.opt and "opt" or "start"
173211
plugin.dir = Config.options.packpath .. "/" .. opt .. "/" .. plugin.name
@@ -179,7 +217,7 @@ function M.update_state(check_clean)
179217
end
180218
end
181219

182-
if check_clean then
220+
if opts.clean then
183221
Config.to_clean = {}
184222
for opt, packs in pairs(installed) do
185223
for pack in pairs(packs) do
@@ -244,7 +282,7 @@ function M.load()
244282
for _, spec in ipairs(specs) do
245283
for _, plugin in pairs(spec.plugins) do
246284
local other = Config.plugins[plugin.name]
247-
Config.plugins[plugin.name] = other and vim.tbl_extend("force", other, plugin) or plugin
285+
Config.plugins[plugin.name] = other and M.merge(other, plugin) or plugin
248286
end
249287
end
250288

lua/lazy/manage/init.lua

+2-2
Original file line numberDiff line numberDiff line change
@@ -115,15 +115,15 @@ end
115115

116116
---@param opts? ManagerOpts
117117
function M.clean(opts)
118-
Plugin.update_state(true)
118+
Plugin.update_state({ clean = true })
119119
M.run({
120120
pipeline = { "fs.clean" },
121121
plugins = Config.to_clean,
122122
}, opts)
123123
end
124124

125125
function M.clear()
126-
Plugin.update_state(true)
126+
Plugin.update_state({ clean = true })
127127
for _, plugin in pairs(Config.plugins) do
128128
plugin._.updated = nil
129129
plugin._.cloned = nil

lua/lazy/view/init.lua

+1-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ M.modes = {
1313
{ name = "log", key = "L", desc = "Show recent updates for all plugins" },
1414
{ name = "restore", key = "R", desc = "Updates all plugins to the state in the lockfile" },
1515
{ name = "profile", key = "P", desc = "Show detailed profiling", toggle = true },
16-
{ name = "help", key = "g?", hide = true, desc = "Toggle this help page", toggle = true },
16+
{ name = "help", key = "?", hide = true, desc = "Toggle this help page", toggle = true },
1717

1818
{ plugin = true, name = "update", key = "u", desc = "Update this plugin. This will also update the lockfile" },
1919
{

lua/lazy/view/render.lua

+1-2
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,6 @@ local Config = require("lazy.core.config")
22
local Util = require("lazy.util")
33
local Sections = require("lazy.view.sections")
44
local Handler = require("lazy.core.handler")
5-
local Plugin = require("lazy.core.plugin")
65
local Git = require("lazy.manage.git")
76

87
local Text = require("lazy.view.text")
@@ -95,7 +94,7 @@ end
9594

9695
function M:title()
9796
self:append(" lazy.nvim ", "LazyH1"):center():nl()
98-
self:append("press "):append("g?", "LazySpecial"):append(" for help"):center():nl()
97+
self:append("press "):append("<?>", "LazySpecial"):append(" for help"):center():nl()
9998
self:append("https://github.com/folke/lazy.nvim", "LazyMuted"):center():nl()
10099

101100
local View = require("lazy.view")

tests/core/plugin_spec.lua

+66-2
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ local assert = require("luassert")
55

66
Config.setup()
77

8-
describe("plugin spec", function()
8+
describe("plugin spec uri/name", function()
99
local tests = {
1010
{ { "~/foo" }, { [1] = "~/foo", name = "foo", uri = vim.fn.fnamemodify("~/foo", ":p") } },
1111
{ { "/tmp/foo" }, { [1] = "/tmp/foo", name = "foo", uri = "/tmp/foo" } },
@@ -19,11 +19,75 @@ describe("plugin spec", function()
1919
}
2020

2121
for _, test in ipairs(tests) do
22-
it("parses uri " .. vim.inspect(test[1]):gsub("%s+", " "), function()
22+
it("parses " .. vim.inspect(test[1]):gsub("%s+", " "), function()
2323
local spec = Plugin.Spec.new(test[1])
2424
local plugins = vim.tbl_values(spec.plugins)
2525
assert.equal(1, #plugins)
2626
assert.same(test[2], plugins[1])
2727
end)
2828
end
2929
end)
30+
31+
describe("plugin spec opt", function()
32+
it("handles dependencies", function()
33+
local tests = {
34+
{ "foo/bar", dependencies = { "foo/dep1", "foo/dep2" } },
35+
{ "foo/bar", dependencies = { { "foo/dep1" }, "foo/dep2" } },
36+
{ { { "foo/bar", dependencies = { { "foo/dep1" }, "foo/dep2" } } } },
37+
}
38+
Config.options.opt = false
39+
for _, test in ipairs(tests) do
40+
local spec = Plugin.Spec.new(test)
41+
Plugin.update_state({ plugins = spec.plugins })
42+
assert(vim.tbl_count(spec.plugins) == 3)
43+
assert(#spec.plugins.bar.dependencies == 2)
44+
assert(spec.plugins.bar.dep ~= true)
45+
assert(spec.plugins.bar.opt == false)
46+
assert(spec.plugins.dep1.dep == true)
47+
assert(spec.plugins.dep1.opt == true)
48+
assert(spec.plugins.dep2.dep == true)
49+
assert(spec.plugins.dep2.opt == true)
50+
end
51+
end)
52+
53+
it("handles opt from dep", function()
54+
Config.options.opt = false
55+
local spec = Plugin.Spec.new({ "foo/dep1", { "foo/bar", dependencies = { "foo/dep1", "foo/dep2" } } })
56+
Plugin.update_state({ plugins = spec.plugins })
57+
assert.same(3, vim.tbl_count(spec.plugins))
58+
assert(spec.plugins.bar.dep ~= true)
59+
assert(spec.plugins.bar.opt == false)
60+
assert(spec.plugins.dep2.dep == true)
61+
assert(spec.plugins.dep2.opt == true)
62+
assert(spec.plugins.dep1.dep ~= true)
63+
assert(spec.plugins.dep1.opt == false)
64+
end)
65+
66+
it("merges lazy loaders", function()
67+
local tests = {
68+
{ { "foo/bar", module = "mod1" }, { "foo/bar", module = "mod2" } },
69+
{ { "foo/bar", module = { "mod1" } }, { "foo/bar", module = { "mod2" } } },
70+
{ { "foo/bar", module = "mod1" }, { "foo/bar", module = { "mod2" } } },
71+
}
72+
for _, test in ipairs(tests) do
73+
local spec = Plugin.Spec.new(test)
74+
assert(vim.tbl_count(spec.plugins) == 1)
75+
assert(type(spec.plugins.bar.module) == "table")
76+
assert(#spec.plugins.bar.module == 2)
77+
assert(vim.tbl_contains(spec.plugins.bar.module, "mod1"))
78+
assert(vim.tbl_contains(spec.plugins.bar.module, "mod2"))
79+
end
80+
end)
81+
82+
it("refuses to merge", function()
83+
assert.has.errors(function()
84+
Plugin.Spec.new({
85+
{ "foo/dep1", config = 1 },
86+
{
87+
"foo/bar",
88+
dependencies = { { "foo/dep1", config = 2 }, "foo/dep2" },
89+
},
90+
})
91+
end)
92+
end)
93+
end)

0 commit comments

Comments
 (0)