Skip to content

Commit

Permalink
feat: add update to README
Browse files Browse the repository at this point in the history
feat: add uv as an installer for python

UV is a very fast installer for python packages
that can be 10-100x faster to resolve packages. This adds
an option for Mason to use it instead of pip to resolve
python packages that are installed via Mason.

More info about the replacement: https://github.com/astral-sh/uv
I have no relationship with uv, it is just very fast and
it would be nice to have updates for packages like sqlfluff take
a lot less time than they currently do to resolve during updates.
  • Loading branch information
KingMichaelPark committed Feb 28, 2024
1 parent 3b5068f commit 6843a02
Show file tree
Hide file tree
Showing 7 changed files with 94 additions and 24 deletions.
4 changes: 4 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -252,6 +252,10 @@ local DEFAULT_SETTINGS = {
-- Whether to upgrade pip to the latest version in the virtual environment before installing packages.
upgrade_pip = false,

---@since 1.8.0
-- Whether to use uv to install packages instead of pip
use_uv = false,

---@since 1.0.0
-- These args will be added to `pip install` calls. Note that setting extra args might impact intended behavior
-- and is not recommended.
Expand Down
4 changes: 4 additions & 0 deletions doc/mason.txt
Original file line number Diff line number Diff line change
Expand Up @@ -314,6 +314,10 @@ Example:
-- Whether to upgrade pip to the latest version in the virtual environment before installing packages.
upgrade_pip = false,

---@since 1.8.0
-- Whether to use uv to install packages instead of pip
use_uv = false,

---@since 1.0.0
-- These args will be added to `pip install` calls. Note that setting extra args might impact intended behavior
-- and is not recommended.
Expand Down
57 changes: 44 additions & 13 deletions lua/mason-core/installer/managers/pypi.lua
Original file line number Diff line number Diff line change
Expand Up @@ -7,23 +7,36 @@ local log = require "mason-core.log"
local path = require "mason-core.path"
local platform = require "mason-core.platform"
local semver = require "mason-core.semver"
local settings = require "mason.settings"
local spawn = require "mason-core.spawn"

local M = {}

local use_uv = settings.current.pip.use_uv
local VENV_DIR = "venv"
if use_uv then
VENV_DIR = ".venv"
end

local is_executable = _.compose(_.equals(1), vim.fn.executable)

---@async
---@param candidates string[]
local function resolve_python3(candidates)
a.scheduler()
if use_uv then
candidates = { "uv" }
end
local available_candidates = _.filter(is_executable, candidates)
for __, candidate in ipairs(available_candidates) do
---@type string
local version_output = spawn[candidate]({ "--version" }):map(_.prop "stdout"):get_or_else ""
local ok, version = pcall(semver.new, version_output:match "Python (3%.%d+.%d+)")
local ok, version
if use_uv then
ok, version = pcall(semver.new, version_output:match "uv (%d+.%d+.%d+)")
else
ok, version = pcall(semver.new, version_output:match "Python (3%.%d+.%d+)")
end
if ok then
return { executable = candidate, version = version }
end
Expand Down Expand Up @@ -70,9 +83,14 @@ local function create_venv()
)
return Result.failure "Failed to find python3 installation."
end
log.fmt_debug("Found python3 installation version=%s, executable=%s", target.version, target.executable)
ctx.stdio_sink.stdout "Creating virtual environment…\n"
return ctx.spawn[target.executable] { "-m", "venv", VENV_DIR }
if use_uv then
log.fmt_debug("Found uv installation version=%s, executable=%s", target.version, target.executable)
return ctx.spawn[target.executable] { "venv", VENV_DIR }
else
log.fmt_debug("Found python3 installation version=%s, executable=%s", target.version, target.executable)
return ctx.spawn[target.executable] { "-m", "venv", VENV_DIR }
end
end

---@param ctx InstallContext
Expand Down Expand Up @@ -107,15 +125,28 @@ end
---@param pkgs string[]
---@param extra_args? string[]
local function pip_install(pkgs, extra_args)
return venv_python {
"-m",
"pip",
"--disable-pip-version-check",
"install",
"-U",
extra_args or vim.NIL,
pkgs,
}
if use_uv then
local ctx = installer.context()
local task = ctx.spawn["uv"] {
"pip",
"install",
"-U",
extra_args or vim.NIL,
pkgs,
}
-- vim.api.nvim_set_current_dir(curdir)
return task
else
return venv_python {
"-m",
"pip",
"--disable-pip-version-check",
"install",
"-U",
extra_args or vim.NIL,
pkgs,
}
end
end

---@async
Expand All @@ -129,7 +160,7 @@ function M.init(opts)
ctx:promote_cwd()
try(create_venv())

if opts.upgrade_pip then
if opts.upgrade_pip and not use_uv then
ctx.stdio_sink.stdout "Upgrading pip inside the virtual environment…\n"
try(pip_install({ "pip" }, opts.install_extra_args))
end
Expand Down
3 changes: 3 additions & 0 deletions lua/mason-core/installer/registry/providers/pypi.lua
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ function M.parse(source, purl)
pip = {
upgrade = settings.current.pip.upgrade_pip,
extra_args = settings.current.pip.install_args,
use_uv = settings.current.pip.use_uv,
},
}

Expand All @@ -44,11 +45,13 @@ function M.install(ctx, source)
try(pypi.init {
upgrade_pip = source.pip.upgrade,
install_extra_args = source.pip.extra_args,
use_uv = source.pip.use_uv,
})
try(pypi.install(source.package, source.version, {
extra = source.extra,
extra_packages = source.extra_packages,
install_extra_args = source.pip.extra_args,
use_uv = source.pip.use_uv,
}))
end)
end
Expand Down
45 changes: 34 additions & 11 deletions lua/mason/providers/client/pypi.lua
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,11 @@ local _ = require "mason-core.functional"
local a = require "mason-core.async"
local fs = require "mason-core.fs"
local platform = require "mason-core.platform"
local settings = require "mason.settings"
local spawn = require "mason-core.spawn"

local use_uv = settings.current.pip.use_uv

---@param args SpawnArgs
local function python(args)
a.scheduler()
Expand All @@ -15,21 +18,41 @@ local function python(args)
return spawn[py_exec](args)
end

---@param args SpawnArgs
local function uv(args)
a.scheduler()
-- run in tmpdir in case pip inadvertently produces some output
args.cwd = vim.fn.tempname()
fs.async.mkdir(args.cwd)
return spawn["uv"](args)
end

---@async
---@param pkg string
local function get_all_versions(pkg)
-- https://stackoverflow.com/a/26664162
return python({
"-m",
"pip",
"install",
"--disable-pip-version-check",
"--use-deprecated=legacy-resolver", -- for pip >= 20.3
("%s=="):format(pkg), -- invalid version specifier to trigger the wanted error message
})
:recover(_.prop "stderr")
:map(_.compose(_.split ", ", _.head, _.match "%(from versions: (.+)%)"))
:map(_.reverse)
if not use_uv then
return python({
"-m",
"pip",
"install",
"--disable-pip-version-check",
"--use-deprecated=legacy-resolver", -- for pip >= 20.3
("%s=="):format(pkg), -- invalid version specifier to trigger the wanted error message
})
:recover(_.prop "stderr")
:map(_.compose(_.split ", ", _.head, _.match "%(from versions: (.+)%)"))
:map(_.reverse)
else
return uv({
"pip",
"install",
("%s=="):format(pkg), -- invalid version specifier to trigger the wanted error message
})
:recover(_.prop "stderr")
:map(_.compose(_.split ", ", _.head, _.match "%(from versions: (.+)%)"))
:map(_.reverse)
end
end

---@param pkg string
Expand Down
4 changes: 4 additions & 0 deletions lua/mason/settings.lua
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,10 @@ local DEFAULT_SETTINGS = {
-- Whether to upgrade pip to the latest version in the virtual environment before installing packages.
upgrade_pip = false,

---@since 1.8.0
-- Whether to use uv to install packages instead of pip
use_uv = false,

---@since 1.0.0
-- These args will be added to `pip install` calls. Note that setting extra args might impact intended behavior
-- and is not recommended.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ describe("pypi provider :: parsing", function()
pip = {
upgrade = true,
extra_args = { "--proxy", "http://localghost" },
use_uv = false,
},
},
pypi.parse({ extra_packages = { "extra" } }, purl())
Expand Down

0 comments on commit 6843a02

Please sign in to comment.