diff options
| author | William Boman <william@redwill.se> | 2025-02-22 01:26:53 +0100 |
|---|---|---|
| committer | GitHub <noreply@github.com> | 2025-02-22 01:26:53 +0100 |
| commit | c8fa5dfaa3ecd104aa1a8c5f77de9c6287adbde3 (patch) | |
| tree | e9482aecc0677975b000f739047f5d0dd9031016 /lua/mason-core/spawn.lua | |
| parent | style: fix stylua and selene errors (diff) | |
| download | mason-c8fa5dfaa3ecd104aa1a8c5f77de9c6287adbde3.tar mason-c8fa5dfaa3ecd104aa1a8c5f77de9c6287adbde3.tar.gz mason-c8fa5dfaa3ecd104aa1a8c5f77de9c6287adbde3.tar.bz2 mason-c8fa5dfaa3ecd104aa1a8c5f77de9c6287adbde3.tar.lz mason-c8fa5dfaa3ecd104aa1a8c5f77de9c6287adbde3.tar.xz mason-c8fa5dfaa3ecd104aa1a8c5f77de9c6287adbde3.tar.zst mason-c8fa5dfaa3ecd104aa1a8c5f77de9c6287adbde3.zip | |
fix(spawn): expand executable paths on Windows before passing to uv_spawn (#1885)
This fixes issues on Windows where uv_spawn fails to locate certain
types of executables in PATH.
Diffstat (limited to 'lua/mason-core/spawn.lua')
| -rw-r--r-- | lua/mason-core/spawn.lua | 37 |
1 files changed, 13 insertions, 24 deletions
diff --git a/lua/mason-core/spawn.lua b/lua/mason-core/spawn.lua index 26434bd9..0a1bb828 100644 --- a/lua/mason-core/spawn.lua +++ b/lua/mason-core/spawn.lua @@ -10,15 +10,6 @@ local is_not_nil = _.complement(_.equals(vim.NIL)) ---@alias JobSpawn table<string, async fun(opts: SpawnArgs): Result> ---@type JobSpawn local spawn = { - _aliases = { - npm = platform.is.win and "npm.cmd" or "npm", - gem = platform.is.win and "gem.cmd" or "gem", - composer = platform.is.win and "composer.bat" or "composer", - gradlew = platform.is.win and "gradlew.bat" or "gradlew", - -- for hererocks installations - luarocks = (platform.is.win and vim.fn.executable "luarocks.bat" == 1) and "luarocks.bat" or "luarocks", - rebar3 = platform.is.win and "rebar3.cmd" or "rebar3", - }, _flatten_cmd_args = _.compose(_.filter(is_not_nil), _.flatten), } @@ -35,10 +26,7 @@ local function Failure(err, cmd) })) end -local is_executable = _.memoize(function(cmd) - a.scheduler() - return vim.fn.executable(cmd) == 1 -end, _.identity) +local has_path = _.any(_.starts_with "PATH=") ---@class SpawnArgs ---@field with_paths string[]? Paths to add to the PATH environment variable. @@ -47,11 +35,10 @@ end, _.identity) ---@field stdio_sink StdioSink? If provided, will be used to write to stdout and stderr. ---@field cwd string? ---@field on_spawn (fun(handle: luv_handle, stdio: luv_pipe[], pid: integer))? Will be called when the process successfully spawns. ----@field check_executable boolean? Whether to check if the provided command is executable (defaults to true). setmetatable(spawn, { - ---@param normalized_cmd string - __index = function(self, normalized_cmd) + ---@param canonical_cmd string + __index = function(self, canonical_cmd) ---@param args SpawnArgs return function(args) local cmd_args = self._flatten_cmd_args(args) @@ -74,13 +61,15 @@ setmetatable(spawn, { spawn_args.stdio_sink = process.BufferedSink:new() end - local cmd = self._aliases[normalized_cmd] or normalized_cmd + local cmd = canonical_cmd - if (env and env.PATH) == nil and args.check_executable ~= false and not is_executable(cmd) then - log.fmt_debug("%s is not executable", cmd) - return Failure({ - stderr = ("%s is not executable"):format(cmd), - }, cmd) + -- Find the executable path via vim.fn.exepath on Windows because libuv fails to resolve certain executables + -- in PATH. + if platform.is.win and (spawn_args.env and has_path(spawn_args.env)) == nil then + local expanded_cmd = vim.fn.exepath(canonical_cmd) + if expanded_cmd ~= "" then + cmd = expanded_cmd + end end local _, exit_code, signal = a.wait(function(resolve) @@ -108,12 +97,12 @@ setmetatable(spawn, { signal = signal, stdout = table.concat(sink.buffers.stdout, "") or nil, stderr = table.concat(sink.buffers.stderr, "") or nil, - }, cmd) + }, canonical_cmd) else return Failure({ exit_code = exit_code, signal = signal, - }, cmd) + }, canonical_cmd) end end end |
