From db949fa7a71755ac3acbe1114aa2bcb7d608a526 Mon Sep 17 00:00:00 2001 From: Zanie Blue Date: Wed, 9 Oct 2024 20:01:44 -0500 Subject: [PATCH] Add `--show-urls` and `--only-downloads` to `uv python list` These are useful for creating a mirror of the Python downloads for a given uv version --- crates/uv-cli/src/lib.rs | 14 ++- crates/uv/src/commands/python/list.rs | 134 +++++++++++++++----------- crates/uv/src/lib.rs | 1 + crates/uv/src/settings.rs | 9 ++ 4 files changed, 103 insertions(+), 55 deletions(-) diff --git a/crates/uv-cli/src/lib.rs b/crates/uv-cli/src/lib.rs index 4f6bd465de2d..3e7ada75284e 100644 --- a/crates/uv-cli/src/lib.rs +++ b/crates/uv-cli/src/lib.rs @@ -3631,8 +3631,20 @@ pub struct PythonListArgs { /// Only show installed Python versions, exclude available downloads. /// /// By default, available downloads for the current platform are shown. - #[arg(long)] + #[arg(long, conflicts_with("only_downloads"))] pub only_installed: bool, + + /// Only show Python downloads, exclude installed distributions. + /// + /// By default, available downloads for the current platform are shown. + #[arg(long, conflicts_with("only_installed"))] + pub only_downloads: bool, + + /// Show the URLs of available Python downloads. + /// + /// By default, these display as ``. + #[arg(long)] + pub show_urls: bool, } #[derive(Args)] diff --git a/crates/uv/src/commands/python/list.rs b/crates/uv/src/commands/python/list.rs index b8f089cf3ef1..8b2c11d610b5 100644 --- a/crates/uv/src/commands/python/list.rs +++ b/crates/uv/src/commands/python/list.rs @@ -2,6 +2,7 @@ use std::collections::BTreeSet; use std::fmt::Write; use anyhow::Result; +use itertools::Either; use owo_colors::OwoColorize; use rustc_hash::FxHashSet; use uv_cache::Cache; @@ -29,6 +30,7 @@ pub(crate) async fn list( kinds: PythonListKinds, all_versions: bool, all_platforms: bool, + show_urls: bool, python_preference: PythonPreference, python_downloads: PythonDownloads, cache: &Cache, @@ -38,6 +40,11 @@ pub(crate) async fn list( if python_preference != PythonPreference::OnlySystem { let download_request = match kinds { PythonListKinds::Installed => None, + PythonListKinds::Downloads => Some(if all_platforms { + PythonDownloadRequest::default() + } else { + PythonDownloadRequest::from_env()? + }), PythonListKinds::Default => { if python_downloads.is_automatic() { Some(if all_platforms { @@ -61,48 +68,60 @@ pub(crate) async fn list( .flatten(); for download in downloads { - output.insert((download.key().clone(), Kind::Download, None)); + output.insert(( + download.key().clone(), + Kind::Download, + Either::Right(download.url()), + )); } }; - let installed = find_python_installations( - &PythonRequest::Any, - EnvironmentPreference::OnlySystem, - python_preference, - cache, - ) - // Raise discovery errors if critical - .filter(|result| { - result - .as_ref() - .err() - .map_or(true, DiscoveryError::is_critical) - }) - .collect::>, DiscoveryError>>()? - .into_iter() - // Drop any "missing" installations - .filter_map(Result::ok); - - for installation in installed { - let kind = if matches!(installation.source(), PythonSource::Managed) { - Kind::Managed - } else { - Kind::System + let installed = + match kinds { + PythonListKinds::Installed | PythonListKinds::Default => { + Some(find_python_installations( + &PythonRequest::Any, + EnvironmentPreference::OnlySystem, + python_preference, + cache, + ) + // Raise discovery errors if critical + .filter(|result| { + result + .as_ref() + .err() + .map_or(true, DiscoveryError::is_critical) + }) + .collect::>, DiscoveryError>>()? + .into_iter() + // Drop any "missing" installations + .filter_map(Result::ok)) + } + PythonListKinds::Downloads => None, }; - output.insert(( - installation.key(), - kind, - Some(installation.interpreter().sys_executable().to_path_buf()), - )); + + if let Some(installed) = installed { + for installation in installed { + let kind = if matches!(installation.source(), PythonSource::Managed) { + Kind::Managed + } else { + Kind::System + }; + output.insert(( + installation.key(), + kind, + Either::Left(installation.interpreter().sys_executable().to_path_buf()), + )); + } } let mut seen_minor = FxHashSet::default(); let mut seen_patch = FxHashSet::default(); let mut seen_paths = FxHashSet::default(); let mut include = Vec::new(); - for (key, kind, path) in output.iter().rev() { + for (key, kind, uri) in output.iter().rev() { // Do not show the same path more than once - if let Some(path) = path { + if let Either::Left(path) = uri { if !seen_paths.insert(path) { continue; } @@ -140,7 +159,7 @@ pub(crate) async fn list( } } } - include.push((key, path)); + include.push((key, uri)); } // Compute the width of the first column. @@ -148,30 +167,37 @@ pub(crate) async fn list( .iter() .fold(0usize, |acc, (key, _)| acc.max(key.to_string().len())); - for (key, path) in include { + for (key, uri) in include { let key = key.to_string(); - if let Some(path) = path { - let is_symlink = fs_err::symlink_metadata(path)?.is_symlink(); - if is_symlink { - writeln!( - printer.stdout(), - "{key:width$} {} -> {}", - path.user_display().cyan(), - path.read_link()?.user_display().cyan() - )?; - } else { - writeln!( - printer.stdout(), - "{key:width$} {}", - path.user_display().cyan() - )?; + match uri { + Either::Left(path) => { + let is_symlink = fs_err::symlink_metadata(path)?.is_symlink(); + if is_symlink { + writeln!( + printer.stdout(), + "{key:width$} {} -> {}", + path.user_display().cyan(), + path.read_link()?.user_display().cyan() + )?; + } else { + writeln!( + printer.stdout(), + "{key:width$} {}", + path.user_display().cyan() + )?; + } + } + Either::Right(url) => { + if show_urls { + writeln!(printer.stdout(), "{key:width$} {}", url.dimmed())?; + } else { + writeln!( + printer.stdout(), + "{key:width$} {}", + "".dimmed() + )?; + } } - } else { - writeln!( - printer.stdout(), - "{key:width$} {}", - "".dimmed() - )?; } } diff --git a/crates/uv/src/lib.rs b/crates/uv/src/lib.rs index 533b0f3c41c9..da32ec043f30 100644 --- a/crates/uv/src/lib.rs +++ b/crates/uv/src/lib.rs @@ -1002,6 +1002,7 @@ async fn run(cli: Cli) -> Result { args.kinds, args.all_versions, args.all_platforms, + args.show_urls, globals.python_preference, globals.python_downloads, &cache, diff --git a/crates/uv/src/settings.rs b/crates/uv/src/settings.rs index 22c35107c043..79191dc475ae 100644 --- a/crates/uv/src/settings.rs +++ b/crates/uv/src/settings.rs @@ -548,6 +548,9 @@ impl ToolDirSettings { pub(crate) enum PythonListKinds { #[default] Default, + /// Only list version downloads. + Downloads, + /// Only list installed versions. Installed, } @@ -558,6 +561,7 @@ pub(crate) struct PythonListSettings { pub(crate) kinds: PythonListKinds, pub(crate) all_platforms: bool, pub(crate) all_versions: bool, + pub(crate) show_urls: bool, } impl PythonListSettings { @@ -568,10 +572,14 @@ impl PythonListSettings { all_versions, all_platforms, only_installed, + only_downloads, + show_urls, } = args; let kinds = if only_installed { PythonListKinds::Installed + } else if only_downloads { + PythonListKinds::Downloads } else { PythonListKinds::default() }; @@ -580,6 +588,7 @@ impl PythonListSettings { kinds, all_platforms, all_versions, + show_urls, } } }