Skip to content

Commit 7933ae1

Browse files
committed
feat: error handler for loading modules, config and init, with custom error formatting
1 parent bad1b1f commit 7933ae1

File tree

2 files changed

+56
-7
lines changed

2 files changed

+56
-7
lines changed

lua/lazy/core/loader.lua

+2-2
Original file line numberDiff line numberDiff line change
@@ -154,7 +154,7 @@ function M.init_plugins()
154154
end
155155
if plugin.init then
156156
Util.track(plugin.name)
157-
plugin.init()
157+
Util.try(plugin.init, "Failed to run `init` for **" .. plugin.name .. "**")
158158
Util.track()
159159
end
160160
if plugin.opt == false then
@@ -235,7 +235,7 @@ function M.load(plugins, reason, opts)
235235
end
236236

237237
if plugin.config then
238-
plugin.config()
238+
Util.try(plugin.config, "Failed to run `config` for " .. plugin.name)
239239
end
240240

241241
plugin.loaded.time = Util.track().time

lua/lazy/core/util.lua

+54-5
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,46 @@ function M.track(name, time)
2727
end
2828
end
2929

30+
function M.try(fn, msg)
31+
-- error handler
32+
local error_handler = function(err)
33+
local Config = require("lazy.core.config")
34+
local trace = {}
35+
local level = 1
36+
while true do
37+
local info = debug.getinfo(level, "Sln")
38+
if not info then
39+
break
40+
end
41+
if info.what == "Lua" and not info.source:find("lazy.nvim") then
42+
local source = info.source:sub(2)
43+
if source:find(Config.options.package_path, 1, true) == 1 then
44+
source = source:sub(#Config.options.package_path + 1):gsub("^/opt/", ""):gsub("^/start/", "")
45+
end
46+
source = vim.fn.fnamemodify(source, ":p:~:.")
47+
local line = " - " .. source .. ":" .. info.currentline
48+
if info.name then
49+
line = line .. " _in_ **" .. info.name .. "**"
50+
end
51+
table.insert(trace, line)
52+
end
53+
level = level + 1
54+
end
55+
vim.schedule(function()
56+
msg = msg .. "\n\n" .. err
57+
if #trace > 0 then
58+
msg = msg .. "\n\n# stacktrace:\n" .. table.concat(trace, "\n")
59+
end
60+
M.error(msg)
61+
end)
62+
return err
63+
end
64+
65+
---@type boolean, any
66+
local ok, result = xpcall(fn, error_handler)
67+
return ok and result or nil
68+
end
69+
3070
-- Fast implementation to check if a table is a list
3171
---@param t table
3272
function M.is_list(t)
@@ -120,16 +160,25 @@ function M.lsmod(root, fn)
120160
end)
121161
end
122162

123-
function M.error(msg)
124-
vim.notify(msg, vim.log.levels.ERROR, {
163+
function M.notify(msg, level)
164+
vim.notify(msg, level, {
165+
on_open = function(win)
166+
vim.wo[win].conceallevel = 3
167+
vim.wo[win].concealcursor = ""
168+
vim.wo[win].spell = false
169+
local buf = vim.api.nvim_win_get_buf(win)
170+
vim.bo[buf].filetype = "markdown"
171+
end,
125172
title = "lazy.nvim",
126173
})
127174
end
128175

176+
function M.error(msg)
177+
M.notify(msg, vim.log.levels.ERROR)
178+
end
179+
129180
function M.info(msg)
130-
vim.notify(msg, vim.log.levels.INFO, {
131-
title = "lazy.nvim",
132-
})
181+
M.notify(msg, vim.log.levels.INFO)
133182
end
134183

135184
return M

0 commit comments

Comments
 (0)