Skip to content

Commit

Permalink
fix(sync): prune rocks sequentially to prevent partial uninstalls
Browse files Browse the repository at this point in the history
  • Loading branch information
mrcjkb committed Dec 10, 2023
1 parent 923c1c2 commit adc9c54
Show file tree
Hide file tree
Showing 4 changed files with 590 additions and 54 deletions.
2 changes: 1 addition & 1 deletion lua/rocks/constants.lua
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,7 @@ constants.DEFAULT_CONFIG = string.format(
# List of Neovim plugins to install alongside their versions.
# If the plugin name contains a dot then you must add quotes to the key name!
[plugins]
"rocks.nvim" = "%s-1" # rocks.nvim can also manage itself :D
"rocks.nvim" = "%s" # rocks.nvim can also manage itself :D
]],
constants.ROCKS_VERSION
)
Expand Down
133 changes: 81 additions & 52 deletions lua/rocks/operations.lua
Original file line number Diff line number Diff line change
Expand Up @@ -30,10 +30,11 @@ local operations = {}
---@field wait fun() Wait in an async context. Does not block in a sync context
---@field wait_sync fun() Wait in a sync context

---@alias rock_table { [string]: Rock[] | string }
---@alias rock_config_table { [string]: Rock|string }
---@alias rock_table { [string]: Rock }

---Decode the user rocks from rocks.toml, creating a default config file if it does not exist
---@return { rocks?: rock_table, plugins?: rock_table }
---@return { rocks?: rock_config_table, plugins?: rock_config_table }
local function parse_user_rocks()
local config_file = fs.read_or_create(config.config_path, constants.DEFAULT_CONFIG)
return require("toml_edit").parse(config_file)
Expand Down Expand Up @@ -79,7 +80,7 @@ operations.install = function(name, version, progress_handle)
name = name,
-- The `gsub` makes sure to escape all punctuation characters
-- so they do not get misinterpreted by the lua pattern engine.
version = sc.stdout:match(name:gsub("%p", "%%%1") .. "%s+(%S+)"),
version = sc.stdout:match(name:gsub("%p", "%%%1") .. "%s+([^-%s]+)"),
}
if progress_handle then
progress_handle:report({
Expand Down Expand Up @@ -192,6 +193,7 @@ operations.sync = function(user_rocks)
}
end
end
---@cast user_rocks rock_table

local installed_rocks = state.installed_rocks()

Expand All @@ -201,24 +203,27 @@ operations.sync = function(user_rocks)
---@diagnostic disable-next-line: invisible
local key_list = nio.fn.keys(vim.tbl_deep_extend("force", installed_rocks, user_rocks))

local actions = vim.empty_dict()
---@cast actions (fun():any)[]
local install_actions = vim.empty_dict()
---@cast install_actions (fun():any)[]

local ct = 1

local dependencies = vim.empty_dict()
---@cast dependencies {[string]: RockDependency}
local to_prune_keys = vim.empty_dict()
---@cast to_prune_keys string[]

local action_count

local to_remove_keys = vim.empty_dict()
---@cast to_remove_keys string[]
local function get_progress_percentage()
return math.floor(ct / action_count * 100)
end

for _, key in ipairs(key_list) do
if user_rocks[key] and not installed_rocks[key] then
nio.scheduler()
progress_handle:report({
message = ("Installing: %s"):format(key),
})
table.insert(actions, function()
table.insert(install_actions, function()
-- If the plugin version is a development release then we pass `dev` as the version to the install function
-- as it gets converted to the `--dev` flag on there, allowing luarocks to pull the `scm-1` rockspec manifest
local future
Expand All @@ -236,13 +241,13 @@ operations.sync = function(user_rocks)
-- TODO: Keep track of failures: #55
progress_handle:report({
message = ("Failed to install %s: %s"):format(key, vim.inspect(ret)),
percentage = math.floor(ct / #actions * 100),
percentage = get_progress_percentage(),
})
return
end
progress_handle:report({
message = ("Installed: %s"):format(key),
percentage = math.floor(ct / #actions * 100),
percentage = get_progress_percentage(),
})
return ret
end)
Expand All @@ -259,7 +264,7 @@ operations.sync = function(user_rocks)
message = is_downgrading and ("Downgrading: %s"):format(key) or ("Updating: %s"):format(key),
})

table.insert(actions, function()
table.insert(install_actions, function()
local future = operations.install(user_rocks[key].name, user_rocks[key].version)
local success, ret = pcall(future.wait)

Expand All @@ -268,23 +273,42 @@ operations.sync = function(user_rocks)
if not success then
has_errors = true
progress_handle:report({
message = ("Failed to downgrade %s: %s"):format(key, vim.inspect(ret)),
percentage = math.floor(ct / #actions * 100),
message = is_downgrading and ("Failed to downgrade %s: %s"):format(key, vim.inspect(ret))
or ("Failed to upgrade %s: %s"):format(key, vim.inspect(ret)),
percentage = get_progress_percentage(),
})
return
end
progress_handle:report({
message = is_downgrading and ("Downgraded: %s"):format(key) or ("Upgraded: %s"):format(key),
percentage = math.floor(ct / #actions * 100),
percentage = get_progress_percentage(),
})

return ret
end)
elseif not user_rocks[key] and installed_rocks[key] then
table.insert(to_remove_keys, key)
table.insert(to_prune_keys, key)
end
end

if installed_rocks[key] then
-- For now, we estimate assuming all values to prune
-- can be pruned
action_count = #install_actions + #to_prune_keys

-- Run install actions before removals, to make sure they don't conflict
-- TODO: Error handling
if not vim.tbl_isempty(install_actions) then
nio.gather(install_actions)
end

-- Determine dependencies of installed user rocks, so they can be excluded from rocks to prune
-- NOTE(mrcjkb): This has to be done after installation,
-- so that we don't prune dependencies of newly installed rocks.
installed_rocks = state.installed_rocks()
local dependencies = vim.empty_dict()
---@cast dependencies {[string]: RockDependency}
for _, key in ipairs(key_list) do
if user_rocks[key] and installed_rocks[key] then
-- NOTE(vhyrro): It is not possible to use the vim.tbl_extend or vim.tbl_deep_extend
-- functions here within the async context. It simply refuses to work.
for k, v in pairs(state.rock_dependencies(installed_rocks[key])) do
Expand All @@ -293,44 +317,50 @@ operations.sync = function(user_rocks)
end
end

for _, key in ipairs(to_remove_keys) do
local is_dependency = dependencies[key] ~= nil
if not is_dependency then
nio.scheduler()
progress_handle:report({
message = ("Removing: %s"):format(key),
})
---@type string[]
local rocks_to_prune = vim.iter(to_prune_keys)
:filter(function(key)
return dependencies[key] == nil
end)
:totable()

table.insert(actions, function()
local future = operations.remove(installed_rocks[key].name)
local success, ret = pcall(future.wait)
if vim.tbl_isempty(install_actions) and vim.tbl_isempty(rocks_to_prune) then
nio.scheduler()
progress_handle:report({ message = "Everything is in-sync!", percentage = 100 })
progress_handle:finish()
return
end

ct = ct + 1
nio.scheduler()
if not success then
has_errors = true
-- TODO: Keep track of failures: #55
progress_handle:report({
message = ("Failed to install %s: %s"):format(key, vim.inspect(ret)),
percentage = math.floor(ct / #actions * 100),
})
return
end
progress_handle:report({
message = ("Removed: %s"):format(key),
percentage = math.floor(ct / #actions * 100),
})
return ret
end)
action_count = #install_actions + #rocks_to_prune

---@diagnostic disable-next-line: invisible
local user_rock_names = nio.fn.keys(user_rocks)
-- Prune rocks sequentially, to prevent conflicts
for _, key in ipairs(rocks_to_prune) do
nio.scheduler()
progress_handle:report({
message = ("Removing: %s"):format(key),
})

local success = operations.remove_recursive(installed_rocks[key].name, user_rock_names)

ct = ct + 1
nio.scheduler()
if not success then
has_errors = true
-- TODO: Keep track of failures: #55
progress_handle:report({
message = ("Failed to prune %s"):format(key),
percentage = get_progress_percentage(),
})
else
progress_handle:report({
message = ("Removed: %s"):format(key),
percentage = get_progress_percentage(),
})
end
end

if not vim.tbl_isempty(actions) then
-- TODO: Error handling
nio.gather(actions)
else
progress_handle:report({ message = "Everything is in-sync!", percentage = 100 })
end
if has_errors then
progress_handle:report({
title = "Error",
Expand Down Expand Up @@ -392,7 +422,6 @@ operations.update = function()

if not vim.tbl_isempty(actions) then
nio.gather(actions)
fs.write_file(config.config_path, "w", tostring(user_rocks))
else
nio.scheduler()
progress_handle:report({ message = "Nothing to update!", percentage = 100 })
Expand Down
Loading

0 comments on commit adc9c54

Please sign in to comment.