lazy.nvim/lua/lazy/manage/task/init.lua

263 lines
6.4 KiB
Lua
Raw Normal View History

2024-06-26 14:11:31 +01:00
local Async = require("lazy.async")
local Config = require("lazy.core.config")
local Process = require("lazy.manage.process")
local Terminal = require("lazy.terminal")
local colors = Config.options.headless.colors
2022-11-28 10:04:32 +00:00
---@class LazyTaskDef
---@field skip? fun(plugin:LazyPlugin, opts?:TaskOptions):any?
2024-06-26 17:31:31 +01:00
---@field run async fun(task:LazyTask, opts:TaskOptions)
2022-11-28 10:04:32 +00:00
---@alias LazyTaskFn async fun(task:LazyTask, opts:TaskOptions)
2024-06-26 13:28:53 +01:00
2024-06-26 17:31:31 +01:00
---@class LazyMsg
---@field msg string
---@field level? number
2022-11-20 22:04:56 +00:00
---@class LazyTask
---@field plugin LazyPlugin
---@field name string
2024-06-26 17:31:31 +01:00
---@field private _log LazyMsg[]
---@field private _started? number
---@field private _ended? number
2022-11-28 10:04:32 +00:00
---@field private _opts TaskOptions
---@field private _running Async
2024-06-26 17:31:31 +01:00
---@field private _level number
2022-11-20 22:04:56 +00:00
local Task = {}
---@class TaskOptions: {[string]:any}
2022-11-28 06:35:58 +00:00
---@field on_done? fun(task:LazyTask)
2022-11-20 22:04:56 +00:00
---@param plugin LazyPlugin
---@param name string
2022-11-23 15:12:43 +00:00
---@param opts? TaskOptions
---@param task LazyTaskFn
function Task.new(plugin, name, task, opts)
local self = setmetatable({}, { __index = Task })
2022-11-28 10:04:32 +00:00
self._opts = opts or {}
2024-06-26 17:31:31 +01:00
self._log = {}
self:set_level()
2022-11-20 22:04:56 +00:00
self.plugin = plugin
self.name = name
---@param other LazyTask
plugin._.tasks = vim.tbl_filter(function(other)
return other.name ~= name or other:is_running()
2024-06-26 13:28:53 +01:00
end, plugin._.tasks or {})
table.insert(plugin._.tasks, self)
self:_start(task)
2022-11-20 22:04:56 +00:00
return self
end
2024-06-26 17:31:31 +01:00
---@param level? number
---@return LazyMsg[]
function Task:get_log(level)
level = level or vim.log.levels.DEBUG
return vim.tbl_filter(function(msg)
return msg.level >= level
end, self._log)
end
---@param level? number
function Task:output(level)
return table.concat(
---@param m LazyMsg
vim.tbl_map(function(m)
return m.msg
end, self:get_log(level)),
"\n"
)
end
function Task:status()
local ret = self._log[#self._log]
2024-06-26 21:44:57 +01:00
local msg = ret and vim.trim(ret.msg) or ""
return msg ~= "" and msg or nil
2024-06-26 17:31:31 +01:00
end
2022-11-28 10:04:32 +00:00
function Task:is_running()
2024-06-27 13:43:35 +01:00
return self._ended == nil
2022-11-20 22:04:56 +00:00
end
2024-06-26 17:31:31 +01:00
function Task:has_errors()
return self._level >= vim.log.levels.ERROR
end
function Task:has_warnings()
return self._level >= vim.log.levels.WARN
end
---@param level? number
function Task:set_level(level)
self._level = level or vim.log.levels.TRACE
end
---@private
---@param task LazyTaskFn
function Task:_start(task)
2024-06-27 13:43:35 +01:00
assert(not self._started, "task already started")
assert(not self._ended, "task already done")
2024-06-26 13:28:53 +01:00
if Config.headless() and Config.options.headless.task then
self:log("Running task " .. self.name, vim.log.levels.INFO)
end
self._started = vim.uv.hrtime()
---@async
self._running = Async.run(function()
task(self, self._opts)
end, {
on_done = function()
self:_done()
end,
on_error = function(err)
2024-06-26 17:31:31 +01:00
self:error(err)
end,
on_yield = function(res)
2024-06-26 17:31:31 +01:00
self:log(res)
end,
})
2022-11-20 22:04:56 +00:00
end
---@param msg string|string[]
2024-06-26 17:31:31 +01:00
---@param level? number
function Task:log(msg, level)
level = level or vim.log.levels.DEBUG
self._level = math.max(self._level or 0, level or 0)
msg = type(msg) == "table" and table.concat(msg, "\n") or msg
---@cast msg string
2024-06-26 17:31:31 +01:00
table.insert(self._log, { msg = msg, level = level })
vim.api.nvim_exec_autocmds("User", { pattern = "LazyRender", modeline = false })
if Config.headless() then
self:headless()
end
end
function Task:headless()
if not Config.options.headless.log then
return
end
local msg = self._log[#self._log]
if not msg or msg.level == vim.log.levels.TRACE then
return
end
local map = {
[vim.log.levels.ERROR] = Terminal.red,
[vim.log.levels.WARN] = Terminal.yellow,
[vim.log.levels.INFO] = Terminal.blue,
}
local color = Config.options.headless.colors and map[msg.level]
io.write(Terminal.prefix(color and color(msg.msg) or msg.msg, self:prefix()))
io.write("\n")
end
---@param msg string|string[]
2024-06-26 17:31:31 +01:00
function Task:error(msg)
self:log(msg, vim.log.levels.ERROR)
end
---@param msg string|string[]
2024-06-26 17:31:31 +01:00
function Task:warn(msg)
self:log(msg, vim.log.levels.WARN)
end
2024-06-26 13:28:53 +01:00
---@private
function Task:_done()
2024-06-27 13:43:35 +01:00
assert(self._started, "task not started")
assert(not self._ended, "task already done")
2024-06-26 14:11:31 +01:00
if self._running and self._running:running() then
return
2024-06-26 14:11:31 +01:00
end
if Config.headless() and Config.options.headless.task then
local ms = math.floor(self:time() + 0.5)
self:log("Finished task " .. self.name .. " in " .. ms .. "ms", vim.log.levels.INFO)
end
self._ended = vim.uv.hrtime()
2022-11-28 10:04:32 +00:00
if self._opts.on_done then
self._opts.on_done(self)
2022-11-23 15:12:43 +00:00
end
vim.api.nvim_exec_autocmds("User", { pattern = "LazyRender", modeline = false })
2022-11-28 10:04:32 +00:00
vim.api.nvim_exec_autocmds("User", {
pattern = "LazyPlugin" .. self.name:sub(1, 1):upper() .. self.name:sub(2),
2022-11-28 10:04:32 +00:00
data = { plugin = self.plugin.name },
})
end
function Task:time()
2024-06-27 13:43:35 +01:00
if not self._started then
return 0
end
2024-06-27 13:43:35 +01:00
if not self._ended then
return (vim.uv.hrtime() - self._started) / 1e6
end
return (self._ended - self._started) / 1e6
end
2024-06-26 13:28:53 +01:00
---@async
2022-11-20 22:04:56 +00:00
---@param cmd string
2022-11-28 10:04:32 +00:00
---@param opts? ProcessOpts
2022-11-20 22:04:56 +00:00
function Task:spawn(cmd, opts)
opts = opts or {}
local on_line = opts.on_line
local on_exit = opts.on_exit
local headless = Config.headless() and Config.options.headless.process
2022-11-20 22:04:56 +00:00
function opts.on_line(line)
if not headless then
return self:log(line, vim.log.levels.TRACE)
end
2022-11-20 22:04:56 +00:00
if on_line then
pcall(on_line, line)
end
end
self._running:suspend()
2024-06-26 13:28:53 +01:00
local running = true
local ret = { ok = true, output = "" }
2022-11-28 10:04:32 +00:00
---@param output string
2022-11-20 22:04:56 +00:00
function opts.on_exit(ok, output)
if not headless then
self:log(vim.trim(output), ok and vim.log.levels.DEBUG or vim.log.levels.ERROR)
end
ret = { ok = ok, output = output }
2024-06-26 13:28:53 +01:00
running = false
self._running:resume()
2024-06-26 13:28:53 +01:00
end
if headless then
opts.on_data = function(data)
-- prefix with plugin name
local prefix = self:prefix()
io.write(Terminal.prefix(data, prefix))
end
end
2024-06-26 13:28:53 +01:00
Process.spawn(cmd, opts)
coroutine.yield()
assert(not running, "process still running?")
if on_exit then
pcall(on_exit, ret.ok, ret.output)
end
coroutine.yield()
return ret.ok
2022-11-22 20:12:50 +00:00
end
2022-11-20 22:04:56 +00:00
function Task:prefix()
local plugin = "[" .. self.plugin.name .. "] "
local task = string.rep(" ", 20 - #(self.name .. self.plugin.name)) .. self.name
return colors and Terminal.magenta(plugin) .. Terminal.cyan(task) .. Terminal.bright_black(" | ")
or plugin .. " " .. task .. " | "
end
2022-11-28 10:04:32 +00:00
function Task:wait()
while self:is_running() do
vim.wait(10)
2022-11-20 22:04:56 +00:00
end
end
return Task