55 lines
1.5 KiB
Lua

-- Copyright (c) 2020-2021 shadmansaleh
-- MIT license, see LICENSE for more details.
--- wrapper arround job api
--- creates a job handler when called
local Job = setmetatable({
--- start the job
start = function(self)
self.job_id = vim.fn.jobstart(self.args.cmd, self.args)
return self.job_id > 0
end,
--- stop the job. also imidiately disables io from the job.
stop = function(self)
if self.killed then
return
end
if self.job_id and self.job_id > 0 then
vim.fn.jobstop(self.job_id)
end
self.job_id = 0
self.killed = true
end,
-- Wraps callbacks so they are only called when job is alive
-- This avoids race conditions
wrap_cb_alive = function(self, name)
local original_cb = self.args[name]
if original_cb then
self.args[name] = function(...)
if not self.killed then
return original_cb(...)
end
end
end
end,
}, {
---create new job handler
---@param self table base job table
---@param args table same args as jobstart except cmd is also passed in part of it
---@return table new job handler
__call = function(self, args)
args = vim.deepcopy(args or {})
if type(args.cmd) == 'string' then
args.cmd = vim.split(args.cmd, ' ')
end
self.__index = self
local job = setmetatable({ args = args }, self)
job:wrap_cb_alive('on_stdout')
job:wrap_cb_alive('on_stderr')
job:wrap_cb_alive('on_stdin')
return job
end,
})
return Job