Skip to content

Commit 4384d0e

Browse files
committed
feat: added lockfile support
1 parent bbad0cb commit 4384d0e

File tree

7 files changed

+127
-44
lines changed

7 files changed

+127
-44
lines changed

README.md

+20-34
Original file line numberDiff line numberDiff line change
@@ -2,51 +2,37 @@
22

33
## ✨ Features
44

5-
- Partial clones instead of shallow clones
6-
- Async
7-
- No need for compile
8-
- Fast
9-
- Correct sequencing of dependencies (deps should always be opt. Maybe make everything opt?)
5+
- [x] Partial clones instead of shallow clones
6+
- [x] waits till missing deps are installed (bootstrap Neovim and start using it right away)
7+
- [x] Async
8+
- [x] No need for compile
9+
- [x] Fast
10+
- [x] Correct sequencing of dependencies (deps should always be opt. Maybe make everything opt?)
1011
- [ ] Import specs from Packer
11-
- Config in multiple files
12-
- Patterns for local packages
12+
- [x] Config in multiple files
13+
- [x] Patterns for local packages
14+
- [x] Profiling
1315
- [ ] lockfile
16+
- [ ] check for updates
1417
- [ ] package.lua
1518
- [ ] package-lock.lua
16-
- [ ] tag/version support `git tag --sort version:refname`
17-
- [ ] auto-loading on completion for lazy-loaded commands
18-
- [ ] semver https://devhints.io/semver
19+
- [x] tag/version support `git tag --sort version:refname`
20+
- [x] auto-loading on completion for lazy-loaded commands
21+
- [x] semver https://devhints.io/semver
1922
https://semver.npmjs.com/
2023

2124
## ✅ TODO
2225

23-
- [ ] show time taken for op in view
26+
- [ ] view keybindings for update/clean/...
27+
- [ ] add profiler to view
28+
- [ ] add buttons for actions
29+
- [x] show time taken for op in view
2430
- [ ] package meta index (package.lua cache for all packages)
2531
- [ ] migrate from Packer
2632
- [ ] auto lazy-loading of lua modules
2733
- [ ] use uv file watcher to check for config changes
2834
- [x] clear errors
29-
- [ ] add support for versions `git tag --sort v:refname`
35+
- [x] add support for versions `git tag --sort v:refname`
3036
- [ ] rename requires to deps
31-
- [ ] move tasks etc to Plugin.state
32-
- loaded
33-
- installed
34-
- updated
35-
- changed: just installed or updated (dirty)
36-
- is_local
37-
https://github.blog/2020-12-21-get-up-to-speed-with-partial-clone-and-shallow-clone/
38-
39-
## 🖥️ Git Operations
40-
41-
1. **install**:
42-
43-
- run `git clone` with given `branch`,`--single-branch`, `filter=blob:none`
44-
and `--no-checkout`
45-
- run `git checkout` with correct `branch`, `tag` or `commit`
46-
47-
2. **update**:
48-
49-
- if branch is missing `git remote set-branches --add origin MISSING_BRANCH`
50-
- `git switch MISSING_BRANCH`
51-
- run `git fetch`
52-
- run `git checkout` with correct `branch`, `tag` or `commit`
37+
- [x] move tasks etc to Plugin.state
38+
- [ ] allow setting up plugins through config

lua/lazy/core/config.lua

+2
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ M.defaults = {
1313
},
1414
interactive = true,
1515
packpath = vim.fn.stdpath("data") .. "/site/pack/lazy",
16+
lockfile = vim.fn.stdpath("config") .. "/lazy-lock.json",
1617
view = {
1718
icons = {
1819
start = "",
@@ -23,6 +24,7 @@ M.defaults = {
2324
keys = "",
2425
cmd = "",
2526
ft = "",
27+
task = "",
2628
},
2729
},
2830
}

lua/lazy/core/plugin.lua

+2-2
Original file line numberDiff line numberDiff line change
@@ -18,11 +18,11 @@ M.dirty = false
1818

1919
---@class LazyPluginState
2020
---@field loaded? {[string]:string, time:number}
21-
---@field installed? boolean
21+
---@field installed boolean
2222
---@field tasks? LazyTask[]
2323
---@field dirty? boolean
2424
---@field updated? {from:string, to:string}
25-
---@field is_local? boolean
25+
---@field is_local boolean
2626
---@field is_symlink? boolean
2727
---@field cloned? boolean
2828

lua/lazy/manage/init.lua

+11-5
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,7 @@ function M.run(ropts, opts)
3939
if opts.wait then
4040
runner:wait()
4141
end
42+
return runner
4243
end
4344

4445
---@param opts? ManagerOpts
@@ -58,23 +59,26 @@ function M.install(opts)
5859
}, opts)
5960
end
6061

61-
---@param opts? ManagerOpts
62+
---@param opts? ManagerOpts|{lockfile?:boolean}
6263
function M.update(opts)
64+
opts = opts or {}
6365
M.run({
6466
pipeline = {
6567
"fs.symlink",
6668
"git.branch",
6769
"git.fetch",
68-
"git.checkout",
70+
{ "git.checkout", lockfile = opts.lockfile },
6971
"plugin.docs",
70-
"plugin.run",
7172
"wait",
73+
"plugin.run",
7274
{ "git.log", updated = true },
7375
},
7476
plugins = function(plugin)
7577
return plugin.uri and plugin._.installed
7678
end,
77-
}, opts)
79+
}, opts):wait(function()
80+
require("lazy.manage.lock").update()
81+
end)
7882
end
7983

8084
---@param opts? ManagerOpts
@@ -97,9 +101,11 @@ function M.clean(opts)
97101
end
98102

99103
function M.clear()
104+
Plugin.update_state(true)
100105
for _, plugin in pairs(Config.plugins) do
101-
-- clear updated status
102106
plugin._.updated = nil
107+
plugin._.cloned = nil
108+
plugin._.dirty = nil
103109
-- clear finished tasks
104110
if plugin._.tasks then
105111
---@param task LazyTask

lua/lazy/manage/lock.lua

+73
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
local Config = require("lazy.core.config")
2+
local Git = require("lazy.manage.git")
3+
4+
local M = {}
5+
6+
---@type table<string, {commit:string, branch:string}>
7+
M.lock = {}
8+
M._loaded = false
9+
10+
function M.update()
11+
local f = assert(io.open(Config.options.lockfile, "w"))
12+
f:write("{\n")
13+
M.lock = {}
14+
15+
---@param plugin LazyPlugin
16+
local plugins = vim.tbl_filter(function(plugin)
17+
return not plugin._.is_local and plugin._.installed
18+
end, Config.plugins)
19+
20+
---@param plugin LazyPlugin
21+
---@type string[]
22+
local names = vim.tbl_map(function(plugin)
23+
return plugin.name
24+
end, plugins)
25+
table.sort(names)
26+
27+
for n, name in ipairs(names) do
28+
local plugin = Config.plugins[name]
29+
if not plugin._.is_local and plugin._.installed then
30+
local info = assert(Git.info(plugin.dir))
31+
if not info.branch then
32+
local branch = assert(Git.get_branch(plugin))
33+
info.branch = branch.branch
34+
end
35+
info.commit = info.commit
36+
-- f:write(([[ [%q] = { branch = %q, commit = %q },]]):format(name, info.branch, info.commit) .. "\n")
37+
f:write(([[ %q: { "branch": %q, "commit": %q }]]):format(name, info.branch, info.commit))
38+
if n ~= #names then
39+
f:write(",\n")
40+
end
41+
---@diagnostic disable-next-line: assign-type-mismatch
42+
M.lock[plugin.name] = info
43+
end
44+
end
45+
f:write("\n}")
46+
f:close()
47+
end
48+
49+
function M.load()
50+
M.lock = {}
51+
M._loaded = true
52+
local f = io.open(Config.options.lockfile, "r")
53+
if f then
54+
---@type string
55+
local data = f:read("*a")
56+
local ok, lock = pcall(vim.json.decode, data)
57+
if ok then
58+
M.lock = lock
59+
end
60+
f:close()
61+
end
62+
end
63+
64+
---@param plugin LazyPlugin
65+
---@return {commit:string, branch:string}
66+
function M.get(plugin)
67+
if not M._loaded then
68+
M.load()
69+
end
70+
return M.lock[plugin.name]
71+
end
72+
73+
return M

lua/lazy/manage/task/git.lua

+16-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
local Util = require("lazy.util")
22
local Git = require("lazy.manage.git")
3+
local Lock = require("lazy.manage.lock")
34

45
---@type table<string, LazyTaskDef>
56
local M = {}
@@ -118,11 +119,21 @@ M.checkout = {
118119
skip = function(plugin)
119120
return not plugin._.installed or plugin._.is_local
120121
end,
121-
run = function(self)
122+
---@param opts {lockfile?:boolean}
123+
run = function(self, opts)
122124
local info = assert(Git.info(self.plugin.dir))
123125
local target = assert(Git.get_target(self.plugin))
124126

125-
if not self.plugin._.cloned and info.commit == target.commit then
127+
local lock
128+
if opts.lockfile then
129+
lock = Lock.get(self.plugin)
130+
if lock then
131+
---@diagnostic disable-next-line: cast-local-type
132+
target = lock
133+
end
134+
end
135+
136+
if not self.plugin._.cloned and info.commit == target.commit and info.branch == target.branch then
126137
return
127138
end
128139

@@ -131,7 +142,9 @@ M.checkout = {
131142
"--progress",
132143
}
133144

134-
if target.tag then
145+
if lock then
146+
table.insert(args, lock.commit)
147+
elseif target.tag then
135148
table.insert(args, "tags/" .. target.tag)
136149
elseif self.plugin.commit then
137150
table.insert(args, self.plugin.commit)

lua/lazy/view/commands.lua

+3
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,9 @@ M.commands = {
4040
update = function()
4141
Manage.update({ clear = true, interactive = true })
4242
end,
43+
reset = function()
44+
Manage.update({ clear = true, interactive = true, lockfile = true })
45+
end,
4346
}
4447

4548
function M.setup()

0 commit comments

Comments
 (0)