Skip to content

Commit a2fdf36

Browse files
committed
feat(profile): added accurate startuptime to ui/stats/docs
1 parent d1739cb commit a2fdf36

File tree

6 files changed

+103
-11
lines changed

6 files changed

+103
-11
lines changed

README.md

+17-3
Original file line numberDiff line numberDiff line change
@@ -500,13 +500,25 @@ $ nvim --headless "+Lazy! sync" +qa
500500
- **plugins**: a list of plugin names to run the operation on
501501
- **concurrency**: limit the `number` of concurrently running tasks
502502

503-
If you want to display the number of plugins on your dashboard, you can use
504-
this simple API:
503+
Stats API (`require("lazy").stats()`):
504+
505+
<!-- stats:start -->
505506

506507
```lua
507-
local plugins = require("lazy").stats().count
508+
{
509+
-- startuptime in milliseconds till UIEnter
510+
startuptime = 0,
511+
-- when true, startuptime is the accurate cputime for the Neovim process. (Linux & Macos)
512+
-- this is more accurate than `nvim --startuptime`, and as such will be slightly higher
513+
-- when false, startuptime is calculated based on a delta with a timestamp when lazy started.
514+
startuptime_cputime = false,
515+
count = 0, -- total number of plugins
516+
loaded = 0, -- number of loaded plugins
517+
}
508518
```
509519

520+
<!-- stats:end -->
521+
510522
**lazy.nvim** provides a statusline component that you can use to show the number of pending updates.
511523
Make sure to enable `config.checker.enabled = true` to make this work.
512524

@@ -543,6 +555,8 @@ The following user events will be triggered:
543555
- **LazyLog**: after running log
544556
- **LazyReload**: triggered by change detection after reloading plugin specs
545557
- **VeryLazy**: triggered after `LazyDone` and processing `VimEnter` auto commands
558+
- **LazyVimStarted**: triggered after `UIEnter` when `require("lazy").stats().startuptime` has been calculated.
559+
Useful to update the startuptime on your dashboard.
546560

547561
## 🔒 Lockfile `lazy-lock.json`
548562

lua/lazy/core/config.lua

+6
Original file line numberDiff line numberDiff line change
@@ -200,6 +200,12 @@ function M.setup(spec, opts)
200200
if M.headless then
201201
require("lazy.view.commands").setup()
202202
else
203+
vim.api.nvim_create_autocmd("UIEnter", {
204+
callback = function()
205+
require("lazy.stats").on_ui_enter()
206+
end,
207+
})
208+
203209
vim.api.nvim_create_autocmd("User", {
204210
pattern = "VeryLazy",
205211
once = true,

lua/lazy/docs.lua

+1
Original file line numberDiff line numberDiff line change
@@ -126,6 +126,7 @@ function M.update()
126126
config = config:gsub("%s*debug = false.\n", "\n")
127127
M.save({
128128
bootstrap = M.extract("lua/lazy/init.lua", "function M%.bootstrap%(%)\n(.-)\nend"),
129+
stats = M.extract("lua/lazy/stats.lua", "\nM%._stats = ({.-\n})"),
129130
config = config,
130131
spec = Util.read_file("lua/lazy/example.lua"),
131132
commands = M.commands(),

lua/lazy/init.lua

+3-8
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,11 @@
11
---@type LazyCommands
22
local M = {}
3+
M._start = 0
34

45
---@param spec LazySpec Should be a module name to load, or a plugin spec
56
---@param opts? LazyConfig
67
function M.setup(spec, opts)
8+
M._start = M._start == 0 and vim.loop.hrtime() or M._start
79
if vim.g.lazy_did_setup then
810
return vim.notify(
911
"Re-sourcing your config is not supported with lazy.nvim",
@@ -62,14 +64,7 @@ function M.setup(spec, opts)
6264
end
6365

6466
function M.stats()
65-
local ret = { count = 0, loaded = 0 }
66-
for _, plugin in pairs(require("lazy.core.config").plugins) do
67-
ret.count = ret.count + 1
68-
if plugin._.loaded then
69-
ret.loaded = ret.loaded + 1
70-
end
71-
end
72-
return ret
67+
return require("lazy.stats").stats()
7368
end
7469

7570
function M.bootstrap()

lua/lazy/stats.lua

+56
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
local M = {}
2+
3+
---@class LazyStats
4+
M._stats = {
5+
-- startuptime in milliseconds till UIEnter
6+
startuptime = 0,
7+
-- when true, startuptime is the accurate cputime for the Neovim process. (Linux & Macos)
8+
-- this is more accurate than `nvim --startuptime`, and as such will be slightly higher
9+
-- when false, startuptime is calculated based on a delta with a timestamp when lazy started.
10+
startuptime_cputime = false,
11+
count = 0, -- total number of plugins
12+
loaded = 0, -- number of loaded plugins
13+
}
14+
15+
function M.on_ui_enter()
16+
if not M.C then
17+
pcall(function() end)
18+
end
19+
20+
local ok = pcall(function()
21+
local ffi = require("ffi")
22+
ffi.cdef([[
23+
typedef long time_t;
24+
typedef int clockid_t;
25+
26+
typedef struct timespec {
27+
time_t tv_sec; /* seconds */
28+
long tv_nsec; /* nanoseconds */
29+
} nanotime;
30+
int clock_gettime(clockid_t clk_id, struct timespec *tp);
31+
]])
32+
local pnano = assert(ffi.new("nanotime[?]", 1))
33+
local CLOCK_PROCESS_CPUTIME_ID = jit.os == "OSX" and 12 or 2
34+
ffi.C.clock_gettime(CLOCK_PROCESS_CPUTIME_ID, pnano)
35+
M._stats.startuptime = tonumber(pnano[0].tv_sec) / 1e6 + tonumber(pnano[0].tv_nsec) / 1e6
36+
M._stats.startuptime_cputime = true
37+
end)
38+
if not ok then
39+
M._stats.startuptime = (vim.loop.hrtime() - require("lazy")._start) / 1e6
40+
end
41+
vim.cmd([[do User LazyVimStarted]])
42+
end
43+
44+
function M.stats()
45+
M._stats.count = 0
46+
M._stats.loaded = 0
47+
for _, plugin in pairs(require("lazy.core.config").plugins) do
48+
M._stats.count = M._stats.count + 1
49+
if plugin._.loaded then
50+
M._stats.loaded = M._stats.loaded + 1
51+
end
52+
end
53+
return M._stats
54+
end
55+
56+
return M

lua/lazy/view/render.lua

+20
Original file line numberDiff line numberDiff line change
@@ -491,6 +491,24 @@ function M:details(plugin)
491491
end
492492

493493
function M:profile()
494+
local stats = require("lazy.stats").stats()
495+
local ms = (math.floor(stats.startuptime * 100 + 0.5) / 100)
496+
self:append("Startuptime: ", "LazyH2"):append(ms .. "ms", "Number"):nl():nl()
497+
if stats.startuptime_cputime then
498+
self:append("Based on the actual CPU time of the Neovim process till "):append("UIEnter", "LazySpecial")
499+
self:append("."):nl()
500+
self:append("This is more accurate than ")
501+
self:append("`nvim --startuptime`", "@text.literal.markdown_inline")
502+
self:append(".")
503+
else
504+
self:append("An accurate startuptime based on the actual CPU time of the Neovim process is not available."):nl()
505+
self
506+
:append("Startuptime is instead based on a delta with a timestamp when lazy started till ")
507+
:append("UIEnter", "LazySpecial")
508+
self:append(".")
509+
end
510+
self:nl():nl()
511+
494512
self:append("Profile", "LazyH2"):nl():nl()
495513
self
496514
:append("You can press ")
@@ -505,6 +523,8 @@ function M:profile()
505523

506524
self:nl()
507525

526+
self:nl():nl()
527+
508528
---@param a LazyProfile
509529
---@param b LazyProfile
510530
local function sort(a, b)

0 commit comments

Comments
 (0)