mirror of
https://github.com/astral-sh/uv.git
synced 2025-08-04 10:58:28 +00:00

## Summary Builds on https://github.com/astral-sh/uv/pull/11724. Closes https://github.com/astral-sh/uv/issues/13476.
383 lines
11 KiB
Rust
383 lines
11 KiB
Rust
use anyhow::{Result, anyhow};
|
|
use clap::{Args, ValueEnum};
|
|
|
|
use uv_warnings::warn_user;
|
|
|
|
pub trait CompatArgs {
|
|
fn validate(&self) -> Result<()>;
|
|
}
|
|
|
|
/// Arguments for `pip-compile` compatibility.
|
|
///
|
|
/// These represent a subset of the `pip-compile` interface that uv supports by default.
|
|
/// For example, users often pass `--allow-unsafe`, which is unnecessary with uv. But it's a
|
|
/// nice user experience to warn, rather than fail, when users pass `--allow-unsafe`.
|
|
#[derive(Args)]
|
|
#[allow(clippy::struct_excessive_bools)]
|
|
pub struct PipCompileCompatArgs {
|
|
#[clap(long, hide = true)]
|
|
allow_unsafe: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
no_allow_unsafe: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
reuse_hashes: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
no_reuse_hashes: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
resolver: Option<Resolver>,
|
|
|
|
#[clap(long, hide = true)]
|
|
max_rounds: Option<usize>,
|
|
|
|
#[clap(long, hide = true)]
|
|
cert: Option<String>,
|
|
|
|
#[clap(long, hide = true)]
|
|
client_cert: Option<String>,
|
|
|
|
#[clap(long, hide = true)]
|
|
emit_trusted_host: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
no_emit_trusted_host: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
config: Option<String>,
|
|
|
|
#[clap(long, hide = true)]
|
|
no_config: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
emit_options: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
no_emit_options: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
pip_args: Option<String>,
|
|
}
|
|
|
|
impl CompatArgs for PipCompileCompatArgs {
|
|
/// Validate the arguments passed for `pip-compile` compatibility.
|
|
///
|
|
/// This method will warn when an argument is passed that has no effect but matches uv's
|
|
/// behavior. If an argument is passed that does _not_ match uv's behavior (e.g.,
|
|
/// `--no-build-isolation`), this method will return an error.
|
|
fn validate(&self) -> Result<()> {
|
|
if self.allow_unsafe {
|
|
warn_user!(
|
|
"pip-compile's `--allow-unsafe` has no effect (uv can safely pin `pip` and other packages)"
|
|
);
|
|
}
|
|
|
|
if self.no_allow_unsafe {
|
|
warn_user!(
|
|
"pip-compile's `--no-allow-unsafe` has no effect (uv can safely pin `pip` and other packages)"
|
|
);
|
|
}
|
|
|
|
if self.reuse_hashes {
|
|
return Err(anyhow!(
|
|
"pip-compile's `--reuse-hashes` is unsupported (uv doesn't reuse hashes)"
|
|
));
|
|
}
|
|
|
|
if self.no_reuse_hashes {
|
|
warn_user!("pip-compile's `--no-reuse-hashes` has no effect (uv doesn't reuse hashes)");
|
|
}
|
|
|
|
if let Some(resolver) = self.resolver {
|
|
match resolver {
|
|
Resolver::Backtracking => {
|
|
warn_user!(
|
|
"pip-compile's `--resolver=backtracking` has no effect (uv always backtracks)"
|
|
);
|
|
}
|
|
Resolver::Legacy => {
|
|
return Err(anyhow!(
|
|
"pip-compile's `--resolver=legacy` is unsupported (uv always backtracks)"
|
|
));
|
|
}
|
|
}
|
|
}
|
|
|
|
if self.max_rounds.is_some() {
|
|
return Err(anyhow!(
|
|
"pip-compile's `--max-rounds` is unsupported (uv always resolves until convergence)"
|
|
));
|
|
}
|
|
|
|
if self.client_cert.is_some() {
|
|
return Err(anyhow!(
|
|
"pip-compile's `--client-cert` is unsupported (uv doesn't support dedicated client certificates)"
|
|
));
|
|
}
|
|
|
|
if self.emit_trusted_host {
|
|
return Err(anyhow!(
|
|
"pip-compile's `--emit-trusted-host` is unsupported"
|
|
));
|
|
}
|
|
|
|
if self.no_emit_trusted_host {
|
|
warn_user!(
|
|
"pip-compile's `--no-emit-trusted-host` has no effect (uv never emits trusted hosts)"
|
|
);
|
|
}
|
|
|
|
if self.config.is_some() {
|
|
return Err(anyhow!(
|
|
"pip-compile's `--config` is unsupported (uv does not use a configuration file)"
|
|
));
|
|
}
|
|
|
|
if self.emit_options {
|
|
return Err(anyhow!(
|
|
"pip-compile's `--emit-options` is unsupported (uv never emits options)"
|
|
));
|
|
}
|
|
|
|
if self.no_emit_options {
|
|
warn_user!("pip-compile's `--no-emit-options` has no effect (uv never emits options)");
|
|
}
|
|
|
|
if self.pip_args.is_some() {
|
|
return Err(anyhow!(
|
|
"pip-compile's `--pip-args` is unsupported (try passing arguments to uv directly)"
|
|
));
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
/// Arguments for `pip list` compatibility.
|
|
///
|
|
/// These represent a subset of the `pip list` interface that uv supports by default.
|
|
#[derive(Args)]
|
|
#[allow(clippy::struct_excessive_bools)]
|
|
pub struct PipListCompatArgs {
|
|
#[clap(long, hide = true)]
|
|
disable_pip_version_check: bool,
|
|
}
|
|
|
|
impl CompatArgs for PipListCompatArgs {
|
|
/// Validate the arguments passed for `pip list` compatibility.
|
|
///
|
|
/// This method will warn when an argument is passed that has no effect but matches uv's
|
|
/// behavior. If an argument is passed that does _not_ match uv's behavior (e.g.,
|
|
/// `--disable-pip-version-check`), this method will return an error.
|
|
fn validate(&self) -> Result<()> {
|
|
if self.disable_pip_version_check {
|
|
warn_user!("pip's `--disable-pip-version-check` has no effect");
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
/// Arguments for `pip-sync` compatibility.
|
|
///
|
|
/// These represent a subset of the `pip-sync` interface that uv supports by default.
|
|
#[derive(Args)]
|
|
#[allow(clippy::struct_excessive_bools)]
|
|
pub struct PipSyncCompatArgs {
|
|
#[clap(short, long, hide = true)]
|
|
ask: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
python_executable: Option<String>,
|
|
|
|
#[clap(long, hide = true)]
|
|
user: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
cert: Option<String>,
|
|
|
|
#[clap(long, hide = true)]
|
|
client_cert: Option<String>,
|
|
|
|
#[clap(long, hide = true)]
|
|
config: Option<String>,
|
|
|
|
#[clap(long, hide = true)]
|
|
no_config: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
pip_args: Option<String>,
|
|
}
|
|
|
|
impl CompatArgs for PipSyncCompatArgs {
|
|
/// Validate the arguments passed for `pip-sync` compatibility.
|
|
///
|
|
/// This method will warn when an argument is passed that has no effect but matches uv's
|
|
/// behavior. If an argument is passed that does _not_ match uv's behavior, this method will
|
|
/// return an error.
|
|
fn validate(&self) -> Result<()> {
|
|
if self.ask {
|
|
return Err(anyhow!(
|
|
"pip-sync's `--ask` is unsupported (uv never asks for confirmation)"
|
|
));
|
|
}
|
|
|
|
if self.python_executable.is_some() {
|
|
return Err(anyhow!(
|
|
"pip-sync's `--python-executable` is unsupported (to install into a separate Python environment, try setting `VIRTUAL_ENV` instead)"
|
|
));
|
|
}
|
|
|
|
if self.user {
|
|
return Err(anyhow!(
|
|
"pip-sync's `--user` is unsupported (use a virtual environment instead)"
|
|
));
|
|
}
|
|
|
|
if self.client_cert.is_some() {
|
|
return Err(anyhow!(
|
|
"pip-sync's `--client-cert` is unsupported (uv doesn't support dedicated client certificates)"
|
|
));
|
|
}
|
|
|
|
if self.config.is_some() {
|
|
return Err(anyhow!(
|
|
"pip-sync's `--config` is unsupported (uv does not use a configuration file)"
|
|
));
|
|
}
|
|
|
|
if self.pip_args.is_some() {
|
|
return Err(anyhow!(
|
|
"pip-sync's `--pip-args` is unsupported (try passing arguments to uv directly)"
|
|
));
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Copy, Clone, ValueEnum)]
|
|
enum Resolver {
|
|
Backtracking,
|
|
Legacy,
|
|
}
|
|
|
|
/// Arguments for `venv` compatibility.
|
|
///
|
|
/// These represent a subset of the `virtualenv` interface that uv supports by default.
|
|
#[derive(Args)]
|
|
#[allow(clippy::struct_excessive_bools)]
|
|
pub struct VenvCompatArgs {
|
|
#[clap(long, hide = true)]
|
|
clear: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
no_seed: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
no_pip: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
no_setuptools: bool,
|
|
|
|
#[clap(long, hide = true)]
|
|
no_wheel: bool,
|
|
}
|
|
|
|
impl CompatArgs for VenvCompatArgs {
|
|
/// Validate the arguments passed for `venv` compatibility.
|
|
///
|
|
/// This method will warn when an argument is passed that has no effect but matches uv's
|
|
/// behavior. If an argument is passed that does _not_ match uv's behavior, this method will
|
|
/// return an error.
|
|
fn validate(&self) -> Result<()> {
|
|
if self.clear {
|
|
warn_user!(
|
|
"virtualenv's `--clear` has no effect (uv always clears the virtual environment)"
|
|
);
|
|
}
|
|
|
|
if self.no_seed {
|
|
warn_user!(
|
|
"virtualenv's `--no-seed` has no effect (uv omits seed packages by default)"
|
|
);
|
|
}
|
|
|
|
if self.no_pip {
|
|
warn_user!("virtualenv's `--no-pip` has no effect (uv omits `pip` by default)");
|
|
}
|
|
|
|
if self.no_setuptools {
|
|
warn_user!(
|
|
"virtualenv's `--no-setuptools` has no effect (uv omits `setuptools` by default)"
|
|
);
|
|
}
|
|
|
|
if self.no_wheel {
|
|
warn_user!("virtualenv's `--no-wheel` has no effect (uv omits `wheel` by default)");
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
/// Arguments for `pip install` compatibility.
|
|
///
|
|
/// These represent a subset of the `pip install` interface that uv supports by default.
|
|
#[derive(Args)]
|
|
#[allow(clippy::struct_excessive_bools)]
|
|
pub struct PipInstallCompatArgs {
|
|
#[clap(long, hide = true)]
|
|
disable_pip_version_check: bool,
|
|
|
|
#[clap(long, hide = false)]
|
|
user: bool,
|
|
}
|
|
|
|
impl CompatArgs for PipInstallCompatArgs {
|
|
/// Validate the arguments passed for `pip install` compatibility.
|
|
///
|
|
/// This method will warn when an argument is passed that has no effect but matches uv's
|
|
/// behavior. If an argument is passed that does _not_ match uv's behavior, this method will
|
|
/// return an error.
|
|
fn validate(&self) -> Result<()> {
|
|
if self.disable_pip_version_check {
|
|
warn_user!("pip's `--disable-pip-version-check` has no effect");
|
|
}
|
|
|
|
if self.user {
|
|
return Err(anyhow!(
|
|
"pip's `--user` is unsupported (use a virtual environment instead)"
|
|
));
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
/// Arguments for generic `pip` command compatibility.
|
|
///
|
|
/// These represent a subset of the `pip` interface that exists on all commands.
|
|
#[derive(Args)]
|
|
#[allow(clippy::struct_excessive_bools)]
|
|
pub struct PipGlobalCompatArgs {
|
|
#[clap(long, hide = true)]
|
|
disable_pip_version_check: bool,
|
|
}
|
|
|
|
impl CompatArgs for PipGlobalCompatArgs {
|
|
/// Validate the arguments passed for `pip` compatibility.
|
|
///
|
|
/// This method will warn when an argument is passed that has no effect but matches uv's
|
|
/// behavior. If an argument is passed that does _not_ match uv's behavior, this method will
|
|
/// return an error.
|
|
fn validate(&self) -> Result<()> {
|
|
if self.disable_pip_version_check {
|
|
warn_user!("pip's `--disable-pip-version-check` has no effect");
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|