From 064d19809ac7c20eaedac537520f6df36c9ebbf6 Mon Sep 17 00:00:00 2001 From: idlsoft Date: Fri, 12 Apr 2024 21:47:23 -0400 Subject: [PATCH] Support uv --override option (#668) --- CHANGELOG.md | 2 ++ rye/src/cli/add.rs | 24 ++++++++++++++++++++++-- rye/src/cli/test.rs | 4 ++-- rye/src/lock.rs | 43 ++++++++++++++++++++++++++++++++++++------- rye/src/pyproject.rs | 11 ++++++++++- rye/src/uv.rs | 4 ++++ 6 files changed, 76 insertions(+), 12 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 0b604114bf..4a0a58030a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -7,6 +7,8 @@ that were not yet released. _Unreleased_ +- Support dependency overrides via `tool.rye.override-dependencies` when using uv. #668 + ## 0.33.0 diff --git a/rye/src/cli/add.rs b/rye/src/cli/add.rs index 13a0f281e0..022616a3cb 100644 --- a/rye/src/cli/add.rs +++ b/rye/src/cli/add.rs @@ -201,11 +201,29 @@ pub struct Args { #[arg(long)] dev: bool, /// Add this as an excluded dependency that will not be installed even if it's a sub dependency. - #[arg(long, conflicts_with = "dev", conflicts_with = "optional")] + #[arg( + long, + conflicts_with = "dev", + conflicts_with = "excluded", + conflicts_with = "override" + )] excluded: bool, /// Add this to an optional dependency group. - #[arg(long, conflicts_with = "dev", conflicts_with = "excluded")] + #[arg( + long, + conflicts_with = "dev", + conflicts_with = "excluded", + conflicts_with = "override" + )] optional: Option, + /// Add this as an override dependency. + #[arg( + long, + conflicts_with = "dev", + conflicts_with = "optional", + conflicts_with = "excluded" + )] + r#override: bool, /// Include pre-releases when finding a package version. #[arg(long)] pre: bool, @@ -240,6 +258,8 @@ pub fn execute(cmd: Args) -> Result<(), Error> { DependencyKind::Excluded } else if let Some(ref section) = cmd.optional { DependencyKind::Optional(section.into()) + } else if cmd.r#override { + DependencyKind::Override } else { DependencyKind::Normal }; diff --git a/rye/src/cli/test.rs b/rye/src/cli/test.rs index ff81f820a0..9d44580303 100644 --- a/rye/src/cli/test.rs +++ b/rye/src/cli/test.rs @@ -137,8 +137,8 @@ pub fn execute(cmd: Args) -> Result<(), Error> { fn has_pytest_dependency(projects: &[PyProject]) -> Result { for project in projects { for dep in project - .iter_dependencies(DependencyKind::Dev) - .chain(project.iter_dependencies(DependencyKind::Normal)) + .iter_dependencies(&DependencyKind::Dev) + .chain(project.iter_dependencies(&DependencyKind::Normal)) { if let Ok(req) = dep.expand(|name| std::env::var(name).ok()) { if normalize_package_name(&req.name) == "pytest" { diff --git a/rye/src/lock.rs b/rye/src/lock.rs index bb30e03f04..b336caba4b 100644 --- a/rye/src/lock.rs +++ b/rye/src/lock.rs @@ -178,12 +178,15 @@ pub fn update_workspace_lockfile( req_file.flush()?; - let exclusions = find_exclusions(&projects)?; + let exclusions = find_requirements(&projects, &DependencyKind::Excluded)?; + let overrides = find_requirements(&projects, &DependencyKind::Override)?; + let overrides_file = maybe_write_requirements_to_temp(&overrides)?; generate_lockfile( output, py_ver, &workspace.path(), req_file.path(), + overrides_file.as_ref().map(|v| v.path()), lockfile, sources, &lock_options, @@ -194,6 +197,21 @@ pub fn update_workspace_lockfile( Ok(()) } +fn maybe_write_requirements_to_temp( + requirements: &HashSet, +) -> Result, Error> { + if requirements.is_empty() { + Ok(None) + } else { + let mut nt_file = NamedTempFile::new()?; + for dep in requirements { + writeln!(&nt_file, "{}", dep)?; + } + nt_file.flush()?; + Ok(Some(nt_file)) + } +} + /// Tries to restore the lock options from the given lockfile. fn restore_lock_options<'o>( lockfile: &Path, @@ -263,10 +281,13 @@ fn collect_workspace_features( Some(features_by_project) } -fn find_exclusions(projects: &[PyProject]) -> Result, Error> { +fn find_requirements( + projects: &[PyProject], + kind: &DependencyKind, +) -> Result, Error> { let mut rv = HashSet::new(); for project in projects { - for dep in project.iter_dependencies(DependencyKind::Excluded) { + for dep in project.iter_dependencies(kind) { rv.insert(dep.expand(|name: &str| { if name == "PROJECT_ROOT" { Some(project.workspace_path().to_string_lossy().to_string()) @@ -285,7 +306,7 @@ fn dump_dependencies( out: &mut fs::File, dep_kind: DependencyKind, ) -> Result<(), Error> { - for dep in pyproject.iter_dependencies(dep_kind) { + for dep in pyproject.iter_dependencies(&dep_kind) { if let Ok(expanded_dep) = dep.expand(|_| { // we actually do not care what it expands to much, for as long // as the end result parses @@ -334,23 +355,26 @@ pub fn update_single_project_lockfile( )?; } - for dep in pyproject.iter_dependencies(DependencyKind::Normal) { + for dep in pyproject.iter_dependencies(&DependencyKind::Normal) { writeln!(req_file, "{}", dep)?; } if lock_mode == LockMode::Dev { - for dep in pyproject.iter_dependencies(DependencyKind::Dev) { + for dep in pyproject.iter_dependencies(&DependencyKind::Dev) { writeln!(req_file, "{}", dep)?; } } req_file.flush()?; - let exclusions = find_exclusions(std::slice::from_ref(pyproject))?; + let exclusions = find_requirements(std::slice::from_ref(pyproject), &DependencyKind::Excluded)?; + let overrides = find_requirements(std::slice::from_ref(pyproject), &DependencyKind::Override)?; + let overrides_file = maybe_write_requirements_to_temp(&overrides)?; generate_lockfile( output, py_ver, &pyproject.workspace_path(), req_file.path(), + overrides_file.as_ref().map(|v| v.path()), lockfile, sources, &lock_options, @@ -367,6 +391,7 @@ fn generate_lockfile( py_ver: &PythonVersion, workspace_path: &Path, requirements_file_in: &Path, + overrides_file_in: Option<&Path>, lockfile: &Path, sources: &ExpandedSources, lock_options: &LockOptions, @@ -405,12 +430,16 @@ fn generate_lockfile( .lockfile( py_ver, requirements_file_in, + overrides_file_in, &requirements_file, lock_options.pre, env::var("__RYE_UV_EXCLUDE_NEWER").ok(), upgrade, )?; } else { + if overrides_file_in.is_some() { + bail!("dependency overrides are only supported by uv"); + } let mut cmd = Command::new(get_pip_compile(py_ver, output)?); // legacy pip tools requires some extra parameters if get_pip_tools_version(py_ver) == PipToolsVersion::Legacy { diff --git a/rye/src/pyproject.rs b/rye/src/pyproject.rs index bf32b073f4..15ff89ca15 100644 --- a/rye/src/pyproject.rs +++ b/rye/src/pyproject.rs @@ -55,6 +55,7 @@ pub enum DependencyKind<'a> { Normal, Dev, Excluded, + Override, Optional(Cow<'a, str>), } @@ -64,6 +65,7 @@ impl<'a> fmt::Display for DependencyKind<'a> { DependencyKind::Normal => f.write_str("regular"), DependencyKind::Dev => f.write_str("dev"), DependencyKind::Excluded => f.write_str("excluded"), + DependencyKind::Override => f.write_str("overrides"), DependencyKind::Optional(ref sect) => write!(f, "optional ({})", sect), } } @@ -903,6 +905,7 @@ impl PyProject { DependencyKind::Normal => &mut self.doc["project"]["dependencies"], DependencyKind::Dev => &mut self.doc["tool"]["rye"]["dev-dependencies"], DependencyKind::Excluded => &mut self.doc["tool"]["rye"]["excluded-dependencies"], + DependencyKind::Override => &mut self.doc["tool"]["rye"]["override-dependencies"], DependencyKind::Optional(ref section) => { // add this as a proper non-inline table if it's missing let table = &mut self.doc["project"]["optional-dependencies"]; @@ -934,6 +937,7 @@ impl PyProject { DependencyKind::Normal => &mut self.doc["project"]["dependencies"], DependencyKind::Dev => &mut self.doc["tool"]["rye"]["dev-dependencies"], DependencyKind::Excluded => &mut self.doc["tool"]["rye"]["excluded-dependencies"], + DependencyKind::Override => &mut self.doc["tool"]["rye"]["override-dependencies"], DependencyKind::Optional(ref section) => { &mut self.doc["project"]["optional-dependencies"][section as &str] } @@ -953,7 +957,7 @@ impl PyProject { /// Iterates over all dependencies. pub fn iter_dependencies( &self, - kind: DependencyKind, + kind: &DependencyKind, ) -> impl Iterator + '_ { let sec = match kind { DependencyKind::Normal => self.doc.get("project").and_then(|x| x.get("dependencies")), @@ -967,6 +971,11 @@ impl PyProject { .get("tool") .and_then(|x| x.get("rye")) .and_then(|x| x.get("excluded-dependencies")), + DependencyKind::Override => self + .doc + .get("tool") + .and_then(|x| x.get("rye")) + .and_then(|x| x.get("override-dependencies")), DependencyKind::Optional(ref section) => self .doc .get("project") diff --git a/rye/src/uv.rs b/rye/src/uv.rs index 76bda81e83..9207ea49a7 100644 --- a/rye/src/uv.rs +++ b/rye/src/uv.rs @@ -304,10 +304,12 @@ impl Uv { Ok(UvWithVenv::new(self.clone(), venv_dir, version)) } + #[allow(clippy::too_many_arguments)] pub fn lockfile( &self, py_version: &PythonVersion, source: &Path, + overrides: Option<&Path>, target: &Path, allow_prerelease: bool, exclude_newer: Option, @@ -334,6 +336,8 @@ impl Uv { cmd.arg(source); + overrides.map(|ref value| cmd.arg("--override").arg(value)); + let status = cmd.status().with_context(|| { format!( "Unable to run uv pip compile and generate {}",