@@ -5,7 +5,19 @@ local uv = vim.loop
5
5
local M = {}
6
6
M .dirty = false
7
7
8
- local cache_path = vim .fn .stdpath (" state" ) .. " /lazy.state"
8
+ --- @class LazyCacheConfig
9
+ M .config = {
10
+ enabled = true ,
11
+ path = vim .fn .stdpath (" state" ) .. " /lazy.state" ,
12
+ -- choose what should be cached
13
+ -- * lazy: cache all lazy.nvim core modules and your config files
14
+ -- * init: all of the above and any module needed to init your plugins
15
+ -- * VimEnter: any module till VimEnter
16
+ -- * VeryLazy: any module till VeryLazy
17
+ -- * allthethings: all mdules. Not recommended
18
+ strategy = " VimEnter" , --- @type " lazy" | " init" | " VimEnter" | " allthethings"
19
+ }
20
+
9
21
--- @type CacheHash
10
22
local cache_hash
11
23
@@ -27,13 +39,25 @@ function M.check_load(modname, modpath)
27
39
require (" lazy.core.loader" ).autoload (modname , modpath )
28
40
end
29
41
30
- --- @param modname string
31
- --- @return any
32
- function M .loader (modname )
42
+ --- @param step ? string
43
+ function M .disable (step )
33
44
if not M .enabled then
34
- return " lazy loader is disabled"
45
+ return
46
+ end
47
+ if step and M .config .strategy ~= step then
48
+ return
35
49
end
36
50
51
+ local idx = M .idx ()
52
+ if idx then
53
+ table.remove (package.loaders , idx )
54
+ end
55
+ M .enabled = false
56
+ end
57
+
58
+ --- @param modname string
59
+ --- @return any
60
+ function M .loader (modname )
37
61
local entry = M .cache [modname ]
38
62
39
63
local chunk , err
@@ -68,11 +92,10 @@ function M.loader(modname)
68
92
return chunk or error (err )
69
93
end
70
94
71
- --- @param modname string
72
- function M .find (modname )
95
+ function M .idx ()
73
96
-- update our loader position if needed
74
97
if package.loaders [M .loader_idx ] ~= M .loader then
75
- M .loader_idx = 1
98
+ M .loader_idx = nil
76
99
--- @diagnostic disable-next-line : no-unknown
77
100
for i , loader in ipairs (package.loaders ) do
78
101
if loader == M .loader then
@@ -81,29 +104,49 @@ function M.find(modname)
81
104
end
82
105
end
83
106
end
107
+ return M .loader_idx
108
+ end
84
109
85
- -- find the module and its modpath
86
- for i = M .loader_idx + 1 , # package.loaders do
87
- --- @diagnostic disable-next-line : no-unknown
88
- local chunk = package.loaders [i ](modname )
89
- if type (chunk ) == " function" then
90
- local info = debug.getinfo (chunk , " S" )
91
- return chunk , (info .what ~= " C" and info .source :sub (2 ))
110
+ --- @param modname string
111
+ function M .find (modname )
112
+ if M .idx () then
113
+ -- find the module and its modpath
114
+ for i = M .loader_idx + 1 , # package.loaders do
115
+ --- @diagnostic disable-next-line : no-unknown
116
+ local chunk = package.loaders [i ](modname )
117
+ if type (chunk ) == " function" then
118
+ local info = debug.getinfo (chunk , " S" )
119
+ return chunk , (info .what ~= " C" and info .source :sub (2 ))
120
+ end
92
121
end
93
122
end
94
123
end
95
124
96
- function M .setup ()
125
+ --- @param opts ? LazyConfig
126
+ function M .setup (opts )
127
+ -- no fancy deep extend here. just set the options
128
+ if opts and opts .performance and opts .performance .cache then
129
+ for k , v in pairs (opts .performance .cache ) do
130
+ M .config [k ] = v
131
+ end
132
+ end
133
+
97
134
M .load_cache ()
98
135
table.insert (package.loaders , M .loader_idx , M .loader )
99
136
100
- vim .api .nvim_create_autocmd (" VimEnter" , {
101
- once = true ,
102
- callback = function ()
103
- -- startup done, so stop caching
104
- M .enabled = false
105
- end ,
106
- })
137
+ if M .config .strategy == " VimEnter" then
138
+ vim .api .nvim_create_autocmd (" VimEnter" , {
139
+ once = true ,
140
+ callback = function ()
141
+ -- use schedule so all other VimEnter handlers will have run
142
+ vim .schedule (function ()
143
+ -- startup done, so stop caching
144
+ M .disable ()
145
+ end )
146
+ end ,
147
+ })
148
+ end
149
+ return M
107
150
end
108
151
109
152
--- @return CacheHash ?
@@ -118,7 +161,7 @@ function M.eq(h1, h2)
118
161
end
119
162
120
163
function M .save_cache ()
121
- local f = assert (uv .fs_open (cache_path , " w" , 438 ))
164
+ local f = assert (uv .fs_open (M . config . path , " w" , 438 ))
122
165
for modname , entry in pairs (M .cache ) do
123
166
if entry .used > os.time () - M .ttl then
124
167
entry .modname = modname
142
185
143
186
function M .load_cache ()
144
187
M .cache = {}
145
- local f = uv .fs_open (cache_path , " r" , 438 )
188
+ local f = uv .fs_open (M . config . path , " r" , 438 )
146
189
if f then
147
190
cache_hash = uv .fs_fstat (f ) --[[ @as CacheHash]]
148
191
local data = uv .fs_read (f , cache_hash .size , 0 ) --[[ @as string]]
@@ -172,7 +215,7 @@ function M.autosave()
172
215
vim .api .nvim_create_autocmd (" VimLeavePre" , {
173
216
callback = function ()
174
217
if M .dirty then
175
- local hash = M .hash (cache_path )
218
+ local hash = M .hash (M . config . path )
176
219
-- abort when the file was changed in the meantime
177
220
if hash == nil or M .eq (cache_hash , hash ) then
178
221
M .save_cache ()
0 commit comments