aboutsummaryrefslogtreecommitdiffstats
path: root/lua/mason-core
diff options
context:
space:
mode:
authorWilliam Boman <william@redwill.se>2023-04-02 00:15:48 +0200
committerGitHub <noreply@github.com>2023-04-02 00:15:48 +0200
commit2e35781f5ee020fac6182b4494fdf2f12f3a9bbf (patch)
tree007649bea851c2b65d04fc685fb7a7ff4b16ff30 /lua/mason-core
parentfix(ui): remove new lines from error (#1167) (diff)
downloadmason-2e35781f5ee020fac6182b4494fdf2f12f3a9bbf.tar
mason-2e35781f5ee020fac6182b4494fdf2f12f3a9bbf.tar.gz
mason-2e35781f5ee020fac6182b4494fdf2f12f3a9bbf.tar.bz2
mason-2e35781f5ee020fac6182b4494fdf2f12f3a9bbf.tar.lz
mason-2e35781f5ee020fac6182b4494fdf2f12f3a9bbf.tar.xz
mason-2e35781f5ee020fac6182b4494fdf2f12f3a9bbf.tar.zst
mason-2e35781f5ee020fac6182b4494fdf2f12f3a9bbf.zip
fix(registry): use oneshot channel for updating registry (#1168)
Diffstat (limited to 'lua/mason-core')
-rw-r--r--lua/mason-core/async/control.lua52
-rw-r--r--lua/mason-core/async/init.lua40
2 files changed, 45 insertions, 47 deletions
diff --git a/lua/mason-core/async/control.lua b/lua/mason-core/async/control.lua
index 3252c070..df2627f6 100644
--- a/lua/mason-core/async/control.lua
+++ b/lua/mason-core/async/control.lua
@@ -5,28 +5,21 @@ local Condvar = {}
Condvar.__index = Condvar
function Condvar.new()
- return setmetatable({ handles = {}, queue = {}, is_notifying = false }, Condvar)
+ return setmetatable({ handles = {} }, Condvar)
end
---@async
function Condvar:wait()
a.wait(function(resolve)
- if self.is_notifying then
- self.queue[resolve] = true
- else
- self.handles[resolve] = true
- end
+ self.handles[#self.handles + 1] = resolve
end)
end
function Condvar:notify_all()
- self.is_notifying = true
- for handle in pairs(self.handles) do
- handle()
+ for _, handle in ipairs(self.handles) do
+ pcall(handle)
end
- self.handles = self.queue
- self.queue = {}
- self.is_notifying = false
+ self.handles = {}
end
local Permit = {}
@@ -69,7 +62,42 @@ function Semaphore:acquire()
return Permit.new(self)
end
+---@class OneShotChannel
+---@field has_sent boolean
+---@field value any
+---@field condvar Condvar
+local OneShotChannel = {}
+OneShotChannel.__index = OneShotChannel
+
+function OneShotChannel.new()
+ return setmetatable({
+ has_sent = false,
+ value = nil,
+ condvar = Condvar.new(),
+ }, OneShotChannel)
+end
+
+function OneShotChannel:is_closed()
+ return self.has_sent
+end
+
+function OneShotChannel:send(...)
+ assert(not self.has_sent, "Oneshot channel can only send once.")
+ self.has_sent = true
+ self.value = { ... }
+ self.condvar:notify_all()
+ self.condvar = nil
+end
+
+function OneShotChannel:receive()
+ if not self.has_sent then
+ self.condvar:wait()
+ end
+ return unpack(self.value)
+end
+
return {
Condvar = Condvar,
Semaphore = Semaphore,
+ OneShotChannel = OneShotChannel,
}
diff --git a/lua/mason-core/async/init.lua b/lua/mason-core/async/init.lua
index a6c7c8ef..df7c996c 100644
--- a/lua/mason-core/async/init.lua
+++ b/lua/mason-core/async/init.lua
@@ -153,41 +153,11 @@ exports.scheduler = function()
await(vim.schedule)
end
----Creates a oneshot channel that can only send once.
-local function oneshot_channel()
- local has_sent = false
- local sent_value
- local saved_callback
-
- return {
- is_closed = function()
- return has_sent
- end,
- send = function(...)
- assert(not has_sent, "Oneshot channel can only send once.")
- has_sent = true
- sent_value = { ... }
- if saved_callback then
- saved_callback(unpack(sent_value))
- end
- end,
- receive = function()
- return await(function(resolve)
- if has_sent then
- resolve(unpack(sent_value))
- else
- saved_callback = resolve
- end
- end)
- end,
- }
-end
-
---@async
---@param suspend_fns async fun()[]
---@param mode '"first"' | '"all"'
local function wait(suspend_fns, mode)
- local channel = oneshot_channel()
+ local channel = require("mason-core.async.control").OneShotChannel.new()
if #suspend_fns == 0 then
return
end
@@ -208,9 +178,9 @@ local function wait(suspend_fns, mode)
thread_cancellations[i] = exports.run(suspend_fn, function(success, result)
completed = completed + 1
if not success then
- if not channel.is_closed() then
+ if not channel:is_closed() then
cancel()
- channel.send(false, result)
+ channel:send(false, result)
results = nil
thread_cancellations = {}
end
@@ -218,7 +188,7 @@ local function wait(suspend_fns, mode)
results[i] = result
if mode == "first" or completed >= count then
cancel()
- channel.send(true, mode == "first" and { result } or results)
+ channel:send(true, mode == "first" and { result } or results)
results = nil
thread_cancellations = {}
end
@@ -227,7 +197,7 @@ local function wait(suspend_fns, mode)
end
end
- local ok, results = channel.receive()
+ local ok, results = channel:receive()
if not ok then
error(results, 2)
end