Skip to main content

uv_test/
lib.rs

1// The `unreachable_pub` is to silence false positives in RustRover.
2#![allow(dead_code, unreachable_pub)]
3
4use std::borrow::BorrowMut;
5use std::ffi::OsString;
6use std::io::Write as _;
7use std::iter::Iterator;
8use std::path::{Path, PathBuf};
9use std::process::{Command, ExitStatus, Output, Stdio};
10use std::str::FromStr;
11use std::{env, io};
12use uv_preview::Preview;
13use uv_python::downloads::ManagedPythonDownloadList;
14
15use assert_cmd::assert::{Assert, OutputAssertExt};
16use assert_fs::assert::PathAssert;
17use assert_fs::fixture::{
18    ChildPath, FileWriteStr, PathChild, PathCopy, PathCreateDir, SymlinkToFile,
19};
20use base64::{Engine, prelude::BASE64_STANDARD as base64};
21use futures::StreamExt;
22use indoc::{formatdoc, indoc};
23use itertools::Itertools;
24use predicates::prelude::predicate;
25use regex::Regex;
26use tokio::io::AsyncWriteExt;
27
28use uv_cache::{Cache, CacheBucket};
29use uv_fs::Simplified;
30use uv_python::managed::ManagedPythonInstallations;
31use uv_python::{
32    EnvironmentPreference, PythonInstallation, PythonPreference, PythonRequest, PythonVersion,
33};
34use uv_static::EnvVars;
35
36// Exclude any packages uploaded after this date.
37static EXCLUDE_NEWER: &str = "2024-03-25T00:00:00Z";
38
39pub const PACKSE_VERSION: &str = "0.3.59";
40pub const DEFAULT_PYTHON_VERSION: &str = "3.12";
41
42// The expected latest patch version for each Python minor version.
43pub const LATEST_PYTHON_3_15: &str = "3.15.0a7";
44pub const LATEST_PYTHON_3_14: &str = "3.14.3";
45pub const LATEST_PYTHON_3_13: &str = "3.13.12";
46pub const LATEST_PYTHON_3_12: &str = "3.12.13";
47pub const LATEST_PYTHON_3_11: &str = "3.11.15";
48pub const LATEST_PYTHON_3_10: &str = "3.10.20";
49
50/// Using a find links url allows using `--index-url` instead of `--extra-index-url` in tests
51/// to prevent dependency confusion attacks against our test suite.
52pub fn build_vendor_links_url() -> String {
53    env::var(EnvVars::UV_TEST_PACKSE_INDEX)
54        .map(|url| format!("{}/vendor/", url.trim_end_matches('/')))
55        .ok()
56        .unwrap_or(format!(
57            "https://astral-sh.github.io/packse/{PACKSE_VERSION}/vendor/"
58        ))
59}
60
61pub fn packse_index_url() -> String {
62    env::var(EnvVars::UV_TEST_PACKSE_INDEX)
63        .map(|url| format!("{}/simple-html/", url.trim_end_matches('/')))
64        .ok()
65        .unwrap_or(format!(
66            "https://astral-sh.github.io/packse/{PACKSE_VERSION}/simple-html/"
67        ))
68}
69
70/// Create a new [`TestContext`] with the given Python version.
71///
72/// Creates a virtual environment for the test.
73///
74/// This macro captures the uv binary path at compile time using `env!("CARGO_BIN_EXE_uv")`,
75/// which is only available in the test crate.
76#[macro_export]
77macro_rules! test_context {
78    ($python_version:expr) => {
79        $crate::TestContext::new_with_bin(
80            $python_version,
81            std::path::PathBuf::from(env!("CARGO_BIN_EXE_uv")),
82        )
83    };
84}
85
86/// Create a new [`TestContext`] with zero or more Python versions.
87///
88/// Unlike [`test_context!`], this does not create a virtual environment.
89///
90/// This macro captures the uv binary path at compile time using `env!("CARGO_BIN_EXE_uv")`,
91/// which is only available in the test crate.
92#[macro_export]
93macro_rules! test_context_with_versions {
94    ($python_versions:expr) => {
95        $crate::TestContext::new_with_versions_and_bin(
96            $python_versions,
97            std::path::PathBuf::from(env!("CARGO_BIN_EXE_uv")),
98        )
99    };
100}
101
102/// Return the path to the uv binary.
103///
104/// This macro captures the uv binary path at compile time using `env!("CARGO_BIN_EXE_uv")`,
105/// which is only available in the test crate.
106#[macro_export]
107macro_rules! get_bin {
108    () => {
109        std::path::PathBuf::from(env!("CARGO_BIN_EXE_uv"))
110    };
111}
112
113#[doc(hidden)] // Macro and test context only, don't use directly.
114pub const INSTA_FILTERS: &[(&str, &str)] = &[
115    (r"--cache-dir [^\s]+", "--cache-dir [CACHE_DIR]"),
116    // Operation times
117    (r"(\s|\()(\d+m )?(\d+\.)?\d+(ms|s)", "$1[TIME]"),
118    // File sizes
119    (r"(\s|\()(\d+\.)?\d+([KM]i)?B", "$1[SIZE]"),
120    // Timestamps
121    (r"tv_sec: \d+", "tv_sec: [TIME]"),
122    (r"tv_nsec: \d+", "tv_nsec: [TIME]"),
123    // Rewrite Windows output to Unix output
124    (r"\\([\w\d]|\.)", "/$1"),
125    (r"uv\.exe", "uv"),
126    // uv version display
127    (
128        r"uv(-.*)? \d+\.\d+\.\d+(-(alpha|beta|rc)\.\d+)?(\+\d+)?( \([^)]*\))?",
129        r"uv [VERSION] ([COMMIT] DATE)",
130    ),
131    // Trim end-of-line whitespaces, to allow removing them on save.
132    (r"([^\s])[ \t]+(\r?\n)", "$1$2"),
133    // Filter SSL certificate loading debug messages (environment-dependent)
134    (r"DEBUG Loaded \d+ certificate\(s\) from [^\n]+\n", ""),
135];
136
137/// Create a context for tests which simplifies shared behavior across tests.
138///
139/// * Set the current directory to a temporary directory (`temp_dir`).
140/// * Set the cache dir to a different temporary directory (`cache_dir`).
141/// * Set a cutoff for versions used in the resolution so the snapshots don't change after a new release.
142/// * Set the venv to a fresh `.venv` in `temp_dir`
143pub struct TestContext {
144    pub root: ChildPath,
145    pub temp_dir: ChildPath,
146    pub cache_dir: ChildPath,
147    pub python_dir: ChildPath,
148    pub home_dir: ChildPath,
149    pub user_config_dir: ChildPath,
150    pub bin_dir: ChildPath,
151    pub venv: ChildPath,
152    pub workspace_root: PathBuf,
153
154    /// The Python version used for the virtual environment, if any.
155    pub python_version: Option<PythonVersion>,
156
157    /// All the Python versions available during this test context.
158    pub python_versions: Vec<(PythonVersion, PathBuf)>,
159
160    /// Path to the uv binary.
161    uv_bin: PathBuf,
162
163    /// Standard filters for this test context.
164    filters: Vec<(String, String)>,
165
166    /// Extra environment variables to apply to all commands.
167    extra_env: Vec<(OsString, OsString)>,
168
169    #[allow(dead_code)]
170    _root: tempfile::TempDir,
171
172    /// Extra temporary directories whose lifetimes are tied to this context (e.g., directories
173    /// on alternate filesystems created by [`TestContext::with_cache_on_cow_fs`]).
174    #[allow(dead_code)]
175    _extra_tempdirs: Vec<tempfile::TempDir>,
176}
177
178impl TestContext {
179    /// Create a new test context with a virtual environment and explicit uv binary path.
180    ///
181    /// This is called by the `test_context!` macro.
182    pub fn new_with_bin(python_version: &str, uv_bin: PathBuf) -> Self {
183        let new = Self::new_with_versions_and_bin(&[python_version], uv_bin);
184        new.create_venv();
185        new
186    }
187
188    /// Set the "exclude newer" timestamp for all commands in this context.
189    #[must_use]
190    pub fn with_exclude_newer(mut self, exclude_newer: &str) -> Self {
191        self.extra_env
192            .push((EnvVars::UV_EXCLUDE_NEWER.into(), exclude_newer.into()));
193        self
194    }
195
196    /// Set the "http timeout" for all commands in this context.
197    #[must_use]
198    pub fn with_http_timeout(mut self, http_timeout: &str) -> Self {
199        self.extra_env
200            .push((EnvVars::UV_HTTP_TIMEOUT.into(), http_timeout.into()));
201        self
202    }
203
204    /// Set the "concurrent installs" for all commands in this context.
205    #[must_use]
206    pub fn with_concurrent_installs(mut self, concurrent_installs: &str) -> Self {
207        self.extra_env.push((
208            EnvVars::UV_CONCURRENT_INSTALLS.into(),
209            concurrent_installs.into(),
210        ));
211        self
212    }
213
214    /// Add extra standard filtering for messages like "Resolved 10 packages" which
215    /// can differ between platforms.
216    ///
217    /// In some cases, these counts are helpful for the snapshot and should not be filtered.
218    #[must_use]
219    pub fn with_filtered_counts(mut self) -> Self {
220        for verb in &[
221            "Resolved",
222            "Prepared",
223            "Installed",
224            "Uninstalled",
225            "Checked",
226        ] {
227            self.filters.push((
228                format!("{verb} \\d+ packages?"),
229                format!("{verb} [N] packages"),
230            ));
231        }
232        self.filters.push((
233            "Removed \\d+ files?".to_string(),
234            "Removed [N] files".to_string(),
235        ));
236        self
237    }
238
239    /// Add extra filtering for cache size output
240    #[must_use]
241    pub fn with_filtered_cache_size(mut self) -> Self {
242        // Filter raw byte counts (numbers on their own line)
243        self.filters
244            .push((r"(?m)^\d+\n".to_string(), "[SIZE]\n".to_string()));
245        // Filter human-readable sizes (e.g., "384.2 KiB")
246        self.filters.push((
247            r"(?m)^\d+(\.\d+)? [KMGT]i?B\n".to_string(),
248            "[SIZE]\n".to_string(),
249        ));
250        self
251    }
252
253    /// Add extra standard filtering for Windows-compatible missing file errors.
254    #[must_use]
255    pub fn with_filtered_missing_file_error(mut self) -> Self {
256        // The exact message string depends on the system language, so we remove it.
257        // We want to only remove the phrase after `Caused by:`
258        self.filters.push((
259            r"[^:\n]* \(os error 2\)".to_string(),
260            " [OS ERROR 2]".to_string(),
261        ));
262        // Replace the Windows "The system cannot find the path specified. (os error 3)"
263        // with the Unix "No such file or directory (os error 2)"
264        // and mask the language-dependent message.
265        self.filters.push((
266            r"[^:\n]* \(os error 3\)".to_string(),
267            " [OS ERROR 2]".to_string(),
268        ));
269        self
270    }
271
272    /// Add extra standard filtering for executable suffixes on the current platform e.g.
273    /// drops `.exe` on Windows.
274    #[must_use]
275    pub fn with_filtered_exe_suffix(mut self) -> Self {
276        self.filters
277            .push((regex::escape(env::consts::EXE_SUFFIX), String::new()));
278        self
279    }
280
281    /// Add extra standard filtering for Python interpreter sources
282    #[must_use]
283    pub fn with_filtered_python_sources(mut self) -> Self {
284        self.filters.push((
285            "virtual environments, managed installations, or search path".to_string(),
286            "[PYTHON SOURCES]".to_string(),
287        ));
288        self.filters.push((
289            "virtual environments, managed installations, search path, or registry".to_string(),
290            "[PYTHON SOURCES]".to_string(),
291        ));
292        self.filters.push((
293            "virtual environments, search path, or registry".to_string(),
294            "[PYTHON SOURCES]".to_string(),
295        ));
296        self.filters.push((
297            "virtual environments, registry, or search path".to_string(),
298            "[PYTHON SOURCES]".to_string(),
299        ));
300        self.filters.push((
301            "virtual environments or search path".to_string(),
302            "[PYTHON SOURCES]".to_string(),
303        ));
304        self.filters.push((
305            "managed installations or search path".to_string(),
306            "[PYTHON SOURCES]".to_string(),
307        ));
308        self.filters.push((
309            "managed installations, search path, or registry".to_string(),
310            "[PYTHON SOURCES]".to_string(),
311        ));
312        self.filters.push((
313            "search path or registry".to_string(),
314            "[PYTHON SOURCES]".to_string(),
315        ));
316        self.filters.push((
317            "registry or search path".to_string(),
318            "[PYTHON SOURCES]".to_string(),
319        ));
320        self.filters
321            .push(("search path".to_string(), "[PYTHON SOURCES]".to_string()));
322        self
323    }
324
325    /// Add extra standard filtering for Python executable names, e.g., stripping version number
326    /// and `.exe` suffixes.
327    #[must_use]
328    pub fn with_filtered_python_names(mut self) -> Self {
329        for name in ["python", "pypy"] {
330            // Note we strip version numbers from the executable names because, e.g., on Windows
331            // `python.exe` is the equivalent to a Unix `python3.12`.`
332            let suffix = if cfg!(windows) {
333                // On Windows, we'll require a `.exe` suffix for disambiguation
334                // We'll also strip version numbers if present, which is not common for `python.exe`
335                // but can occur for, e.g., `pypy3.12.exe`
336                let exe_suffix = regex::escape(env::consts::EXE_SUFFIX);
337                format!(r"(\d\.\d+|\d)?{exe_suffix}")
338            } else {
339                // On Unix, we'll strip version numbers
340                if name == "python" {
341                    // We can't require them in this case since `/python` is common
342                    r"(\d\.\d+|\d)?(t|d|td)?".to_string()
343                } else {
344                    // However, for other names we'll require them to avoid over-matching
345                    r"(\d\.\d+|\d)(t|d|td)?".to_string()
346                }
347            };
348
349            self.filters.push((
350                // We use a leading path separator to help disambiguate cases where the name is not
351                // used in a path.
352                format!(r"[\\/]{name}{suffix}"),
353                format!("/[{}]", name.to_uppercase()),
354            ));
355        }
356
357        self
358    }
359
360    /// Add extra standard filtering for venv executable directories on the current platform e.g.
361    /// `Scripts` on Windows and `bin` on Unix.
362    #[must_use]
363    pub fn with_filtered_virtualenv_bin(mut self) -> Self {
364        self.filters.push((
365            format!(
366                r"[\\/]{}[\\/]",
367                venv_bin_path(PathBuf::new()).to_string_lossy()
368            ),
369            "/[BIN]/".to_string(),
370        ));
371        self.filters.push((
372            format!(r"[\\/]{}", venv_bin_path(PathBuf::new()).to_string_lossy()),
373            "/[BIN]".to_string(),
374        ));
375        self
376    }
377
378    /// Add extra standard filtering for Python installation `bin/` directories, which are not
379    /// present on Windows but are on Unix. See [`TestContext::with_filtered_virtualenv_bin`] for
380    /// the virtual environment equivalent.
381    #[must_use]
382    pub fn with_filtered_python_install_bin(mut self) -> Self {
383        // We don't want to eagerly match paths that aren't actually Python executables, so we
384        // do our best to detect that case
385        let suffix = if cfg!(windows) {
386            let exe_suffix = regex::escape(env::consts::EXE_SUFFIX);
387            // On Windows, we usually don't have a version attached but we might, e.g., for pypy3.12
388            format!(r"(\d\.\d+|\d)?{exe_suffix}")
389        } else {
390            // On Unix, we'll require a version to be attached to avoid over-matching
391            r"\d\.\d+|\d".to_string()
392        };
393
394        if cfg!(unix) {
395            self.filters.push((
396                format!(r"[\\/]bin/python({suffix})"),
397                "/[INSTALL-BIN]/python$1".to_string(),
398            ));
399            self.filters.push((
400                format!(r"[\\/]bin/pypy({suffix})"),
401                "/[INSTALL-BIN]/pypy$1".to_string(),
402            ));
403        } else {
404            self.filters.push((
405                format!(r"[\\/]python({suffix})"),
406                "/[INSTALL-BIN]/python$1".to_string(),
407            ));
408            self.filters.push((
409                format!(r"[\\/]pypy({suffix})"),
410                "/[INSTALL-BIN]/pypy$1".to_string(),
411            ));
412        }
413        self
414    }
415
416    /// Filtering for various keys in a `pyvenv.cfg` file that will vary
417    /// depending on the specific machine used:
418    /// - `home = foo/bar/baz/python3.X.X/bin`
419    /// - `uv = X.Y.Z`
420    /// - `extends-environment = <path/to/parent/venv>`
421    #[must_use]
422    pub fn with_pyvenv_cfg_filters(mut self) -> Self {
423        let added_filters = [
424            (r"home = .+".to_string(), "home = [PYTHON_HOME]".to_string()),
425            (
426                r"uv = \d+\.\d+\.\d+(-(alpha|beta|rc)\.\d+)?(\+\d+)?".to_string(),
427                "uv = [UV_VERSION]".to_string(),
428            ),
429            (
430                r"extends-environment = .+".to_string(),
431                "extends-environment = [PARENT_VENV]".to_string(),
432            ),
433        ];
434        for filter in added_filters {
435            self.filters.insert(0, filter);
436        }
437        self
438    }
439
440    /// Add extra filtering for ` -> <PATH>` symlink display for Python versions in the test
441    /// context, e.g., for use in `uv python list`.
442    #[must_use]
443    pub fn with_filtered_python_symlinks(mut self) -> Self {
444        for (version, executable) in &self.python_versions {
445            if fs_err::symlink_metadata(executable).unwrap().is_symlink() {
446                self.filters.extend(
447                    Self::path_patterns(executable.read_link().unwrap())
448                        .into_iter()
449                        .map(|pattern| (format! {" -> {pattern}"}, String::new())),
450                );
451            }
452            // Drop links that are byproducts of the test context too
453            self.filters.push((
454                regex::escape(&format!(" -> [PYTHON-{version}]")),
455                String::new(),
456            ));
457        }
458        self
459    }
460
461    /// Add extra standard filtering for a given path.
462    #[must_use]
463    pub fn with_filtered_path(mut self, path: &Path, name: &str) -> Self {
464        // Note this is sloppy, ideally we wouldn't push to the front of the `Vec` but we need
465        // this to come in front of other filters or we can transform the path (e.g., with `[TMP]`)
466        // before we reach this filter.
467        for pattern in Self::path_patterns(path)
468            .into_iter()
469            .map(|pattern| (pattern, format!("[{name}]/")))
470        {
471            self.filters.insert(0, pattern);
472        }
473        self
474    }
475
476    /// Adds a filter that specifically ignores the link mode warning.
477    ///
478    /// This occurs in some cases and can be used on an ad hoc basis to squash
479    /// the warning in the snapshots. This is useful because the warning does
480    /// not consistently appear. It is dependent on the environment. (For
481    /// example, sometimes it's dependent on whether `/tmp` and `~/.local` live
482    /// on the same file system.)
483    #[inline]
484    #[must_use]
485    pub fn with_filtered_link_mode_warning(mut self) -> Self {
486        let pattern = "warning: Failed to hardlink files; .*\n.*\n.*\n";
487        self.filters.push((pattern.to_string(), String::new()));
488        self
489    }
490
491    /// Adds a filter for platform-specific errors when a file is not executable.
492    #[inline]
493    #[must_use]
494    pub fn with_filtered_not_executable(mut self) -> Self {
495        let pattern = if cfg!(unix) {
496            r"Permission denied \(os error 13\)"
497        } else {
498            r"\%1 is not a valid Win32 application. \(os error 193\)"
499        };
500        self.filters
501            .push((pattern.to_string(), "[PERMISSION DENIED]".to_string()));
502        self
503    }
504
505    /// Adds a filter that ignores platform information in a Python installation key.
506    #[must_use]
507    pub fn with_filtered_python_keys(mut self) -> Self {
508        // Filter platform keys
509        let platform_re = r"(?x)
510  (                         # We capture the group before the platform
511    (?:cpython|pypy|graalpy)# Python implementation
512    -
513    \d+\.\d+                # Major and minor version
514    (?:                     # The patch version is handled separately
515      \.
516      (?:
517        \[X\]               # A previously filtered patch version [X]
518        |                   # OR
519        \[LATEST\]          # A previously filtered latest patch version [LATEST]
520        |                   # OR
521        \d+                 # An actual patch version
522      )
523    )?                      # (we allow the patch version to be missing entirely, e.g., in a request)
524    (?:(?:a|b|rc)[0-9]+)?   # Pre-release version component, e.g., `a6` or `rc2`
525    (?:[td])?               # A short variant, such as `t` (for freethreaded) or `d` (for debug)
526    (?:(\+[a-z]+)+)?        # A long variant, such as `+freethreaded` or `+freethreaded+debug`
527  )
528  -
529  [a-z0-9]+                 # Operating system (e.g., 'macos')
530  -
531  [a-z0-9_]+                # Architecture (e.g., 'aarch64')
532  -
533  [a-z]+                    # Libc (e.g., 'none')
534";
535        self.filters
536            .push((platform_re.to_string(), "$1-[PLATFORM]".to_string()));
537        self
538    }
539
540    /// Adds a filter that replaces the latest Python patch versions with `[LATEST]` placeholder.
541    #[must_use]
542    pub fn with_filtered_latest_python_versions(mut self) -> Self {
543        // Filter the latest patch versions with [LATEST] placeholder
544        // The order matters - we want to match the full version first
545        for (minor, patch) in [
546            ("3.15", LATEST_PYTHON_3_15.strip_prefix("3.15.").unwrap()),
547            ("3.14", LATEST_PYTHON_3_14.strip_prefix("3.14.").unwrap()),
548            ("3.13", LATEST_PYTHON_3_13.strip_prefix("3.13.").unwrap()),
549            ("3.12", LATEST_PYTHON_3_12.strip_prefix("3.12.").unwrap()),
550            ("3.11", LATEST_PYTHON_3_11.strip_prefix("3.11.").unwrap()),
551            ("3.10", LATEST_PYTHON_3_10.strip_prefix("3.10.").unwrap()),
552        ] {
553            // Match the full version in various contexts (cpython-X.Y.Z, Python X.Y.Z, etc.)
554            let pattern = format!(r"(\b){minor}\.{patch}(\b)");
555            let replacement = format!("${{1}}{minor}.[LATEST]${{2}}");
556            self.filters.push((pattern, replacement));
557        }
558        self
559    }
560
561    /// Add a filter that ignores temporary directory in path.
562    #[must_use]
563    pub fn with_filtered_windows_temp_dir(mut self) -> Self {
564        let pattern = regex::escape(
565            &self
566                .temp_dir
567                .simplified_display()
568                .to_string()
569                .replace('/', "\\"),
570        );
571        self.filters.push((pattern, "[TEMP_DIR]".to_string()));
572        self
573    }
574
575    /// Add a filter for (bytecode) compilation file counts
576    #[must_use]
577    pub fn with_filtered_compiled_file_count(mut self) -> Self {
578        self.filters.push((
579            r"compiled \d+ files".to_string(),
580            "compiled [COUNT] files".to_string(),
581        ));
582        self
583    }
584
585    /// Add a (not context aware) filter for the current uv version `v<major>.<minor>.<patch>`
586    #[must_use]
587    pub fn with_filtered_current_version(mut self) -> Self {
588        self.filters.push((
589            regex::escape(&format!("v{}", env!("CARGO_PKG_VERSION"))),
590            "v[CURRENT_VERSION]".to_string(),
591        ));
592        self
593    }
594
595    /// Adds filters for non-deterministic `CycloneDX` data
596    #[must_use]
597    pub fn with_cyclonedx_filters(mut self) -> Self {
598        self.filters.push((
599            r"urn:uuid:[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}".to_string(),
600            "[SERIAL_NUMBER]".to_string(),
601        ));
602        self.filters.push((
603            r#""timestamp": "[0-9]{4}-[0-9]{2}-[0-9]{2}T[0-9]{2}:[0-9]{2}:[0-9]{2}\.[0-9]+Z""#
604                .to_string(),
605            r#""timestamp": "[TIMESTAMP]""#.to_string(),
606        ));
607        self.filters.push((
608            r#""name": "uv",\s*"version": "\d+\.\d+\.\d+(-(alpha|beta|rc)\.\d+)?(\+\d+)?""#
609                .to_string(),
610            r#""name": "uv",
611        "version": "[VERSION]""#
612                .to_string(),
613        ));
614        self
615    }
616
617    /// Add a filter that collapses duplicate whitespace.
618    #[must_use]
619    pub fn with_collapsed_whitespace(mut self) -> Self {
620        self.filters.push((r"[ \t]+".to_string(), " ".to_string()));
621        self
622    }
623
624    /// Use a shared global cache for Python downloads.
625    #[must_use]
626    pub fn with_python_download_cache(mut self) -> Self {
627        self.extra_env.push((
628            EnvVars::UV_PYTHON_CACHE_DIR.into(),
629            // Respect `UV_PYTHON_CACHE_DIR` if set, or use the default cache directory
630            env::var_os(EnvVars::UV_PYTHON_CACHE_DIR).unwrap_or_else(|| {
631                uv_cache::Cache::from_settings(false, None)
632                    .unwrap()
633                    .bucket(CacheBucket::Python)
634                    .into()
635            }),
636        ));
637        self
638    }
639
640    #[must_use]
641    pub fn with_empty_python_install_mirror(mut self) -> Self {
642        self.extra_env.push((
643            EnvVars::UV_PYTHON_INSTALL_MIRROR.into(),
644            String::new().into(),
645        ));
646        self
647    }
648
649    /// Add extra directories and configuration for managed Python installations.
650    #[must_use]
651    pub fn with_managed_python_dirs(mut self) -> Self {
652        let managed = self.temp_dir.join("managed");
653
654        self.extra_env.push((
655            EnvVars::UV_PYTHON_BIN_DIR.into(),
656            self.bin_dir.as_os_str().to_owned(),
657        ));
658        self.extra_env
659            .push((EnvVars::UV_PYTHON_INSTALL_DIR.into(), managed.into()));
660        self.extra_env
661            .push((EnvVars::UV_PYTHON_DOWNLOADS.into(), "automatic".into()));
662
663        self
664    }
665
666    #[must_use]
667    pub fn with_versions_as_managed(mut self, versions: &[&str]) -> Self {
668        self.extra_env.push((
669            EnvVars::UV_INTERNAL__TEST_PYTHON_MANAGED.into(),
670            versions.iter().join(" ").into(),
671        ));
672
673        self
674    }
675
676    /// Add a custom filter to the `TestContext`.
677    #[must_use]
678    pub fn with_filter(mut self, filter: (impl Into<String>, impl Into<String>)) -> Self {
679        self.filters.push((filter.0.into(), filter.1.into()));
680        self
681    }
682
683    // Unsets the git credential helper using temp home gitconfig
684    #[must_use]
685    pub fn with_unset_git_credential_helper(self) -> Self {
686        let git_config = self.home_dir.child(".gitconfig");
687        git_config
688            .write_str(indoc! {r"
689                [credential]
690                    helper =
691            "})
692            .expect("Failed to unset git credential helper");
693
694        self
695    }
696
697    /// Clear filters on `TestContext`.
698    #[must_use]
699    pub fn clear_filters(mut self) -> Self {
700        self.filters.clear();
701        self
702    }
703
704    /// Use a cache directory on the filesystem specified by
705    /// [`EnvVars::UV_INTERNAL__TEST_COW_FS`].
706    ///
707    /// Returns `Ok(None)` if the environment variable is not set.
708    pub fn with_cache_on_cow_fs(self) -> anyhow::Result<Option<Self>> {
709        let Some(dir) = env::var(EnvVars::UV_INTERNAL__TEST_COW_FS).ok() else {
710            return Ok(None);
711        };
712        self.with_cache_on_fs(&dir, "COW_FS").map(Some)
713    }
714
715    /// Use a cache directory on the filesystem specified by
716    /// [`EnvVars::UV_INTERNAL__TEST_ALT_FS`].
717    ///
718    /// Returns `Ok(None)` if the environment variable is not set.
719    pub fn with_cache_on_alt_fs(self) -> anyhow::Result<Option<Self>> {
720        let Some(dir) = env::var(EnvVars::UV_INTERNAL__TEST_ALT_FS).ok() else {
721            return Ok(None);
722        };
723        self.with_cache_on_fs(&dir, "ALT_FS").map(Some)
724    }
725
726    /// Use a cache directory on the filesystem specified by
727    /// [`EnvVars::UV_INTERNAL__TEST_LOWLINKS_FS`].
728    ///
729    /// Returns `Ok(None)` if the environment variable is not set.
730    pub fn with_cache_on_lowlinks_fs(self) -> anyhow::Result<Option<Self>> {
731        let Some(dir) = env::var(EnvVars::UV_INTERNAL__TEST_LOWLINKS_FS).ok() else {
732            return Ok(None);
733        };
734        self.with_cache_on_fs(&dir, "LOWLINKS_FS").map(Some)
735    }
736
737    /// Use a cache directory on the filesystem specified by
738    /// [`EnvVars::UV_INTERNAL__TEST_NOCOW_FS`].
739    ///
740    /// Returns `Ok(None)` if the environment variable is not set.
741    pub fn with_cache_on_nocow_fs(self) -> anyhow::Result<Option<Self>> {
742        let Some(dir) = env::var(EnvVars::UV_INTERNAL__TEST_NOCOW_FS).ok() else {
743            return Ok(None);
744        };
745        self.with_cache_on_fs(&dir, "NOCOW_FS").map(Some)
746    }
747
748    /// Use a working directory on the filesystem specified by
749    /// [`EnvVars::UV_INTERNAL__TEST_COW_FS`].
750    ///
751    /// Returns `Ok(None)` if the environment variable is not set.
752    ///
753    /// Note a virtual environment is not created automatically.
754    pub fn with_working_dir_on_cow_fs(self) -> anyhow::Result<Option<Self>> {
755        let Some(dir) = env::var(EnvVars::UV_INTERNAL__TEST_COW_FS).ok() else {
756            return Ok(None);
757        };
758        self.with_working_dir_on_fs(&dir, "COW_FS").map(Some)
759    }
760
761    /// Use a working directory on the filesystem specified by
762    /// [`EnvVars::UV_INTERNAL__TEST_ALT_FS`].
763    ///
764    /// Returns `Ok(None)` if the environment variable is not set.
765    ///
766    /// Note a virtual environment is not created automatically.
767    pub fn with_working_dir_on_alt_fs(self) -> anyhow::Result<Option<Self>> {
768        let Some(dir) = env::var(EnvVars::UV_INTERNAL__TEST_ALT_FS).ok() else {
769            return Ok(None);
770        };
771        self.with_working_dir_on_fs(&dir, "ALT_FS").map(Some)
772    }
773
774    /// Use a working directory on the filesystem specified by
775    /// [`EnvVars::UV_INTERNAL__TEST_NOCOW_FS`].
776    ///
777    /// Returns `Ok(None)` if the environment variable is not set.
778    ///
779    /// Note a virtual environment is not created automatically.
780    pub fn with_working_dir_on_nocow_fs(self) -> anyhow::Result<Option<Self>> {
781        let Some(dir) = env::var(EnvVars::UV_INTERNAL__TEST_NOCOW_FS).ok() else {
782            return Ok(None);
783        };
784        self.with_working_dir_on_fs(&dir, "NOCOW_FS").map(Some)
785    }
786
787    fn with_cache_on_fs(mut self, dir: &str, name: &str) -> anyhow::Result<Self> {
788        fs_err::create_dir_all(dir)?;
789        let tmp = tempfile::TempDir::new_in(dir)?;
790        self.cache_dir = ChildPath::new(tmp.path()).child("cache");
791        fs_err::create_dir_all(&self.cache_dir)?;
792        let replacement = format!("[{name}]/[CACHE_DIR]/");
793        self.filters.extend(
794            Self::path_patterns(&self.cache_dir)
795                .into_iter()
796                .map(|pattern| (pattern, replacement.clone())),
797        );
798        self._extra_tempdirs.push(tmp);
799        Ok(self)
800    }
801
802    fn with_working_dir_on_fs(mut self, dir: &str, name: &str) -> anyhow::Result<Self> {
803        fs_err::create_dir_all(dir)?;
804        let tmp = tempfile::TempDir::new_in(dir)?;
805        self.temp_dir = ChildPath::new(tmp.path()).child("temp");
806        fs_err::create_dir_all(&self.temp_dir)?;
807        // Place the venv inside temp_dir (matching the default TestContext layout)
808        // so that `context.venv()` creates it at the same path that `VIRTUAL_ENV` points to.
809        let canonical_temp_dir = self.temp_dir.canonicalize()?;
810        self.venv = ChildPath::new(canonical_temp_dir.join(".venv"));
811        let temp_replacement = format!("[{name}]/[TEMP_DIR]/");
812        self.filters.extend(
813            Self::path_patterns(&self.temp_dir)
814                .into_iter()
815                .map(|pattern| (pattern, temp_replacement.clone())),
816        );
817        let venv_replacement = format!("[{name}]/[VENV]/");
818        self.filters.extend(
819            Self::path_patterns(&self.venv)
820                .into_iter()
821                .map(|pattern| (pattern, venv_replacement.clone())),
822        );
823        self._extra_tempdirs.push(tmp);
824        Ok(self)
825    }
826
827    /// Default to the canonicalized path to the temp directory. We need to do this because on
828    /// macOS (and Windows on GitHub Actions) the standard temp dir is a symlink. (On macOS, the
829    /// temporary directory is, like `/var/...`, which resolves to `/private/var/...`.)
830    ///
831    /// It turns out that, at least on macOS, if we pass a symlink as `current_dir`, it gets
832    /// _immediately_ resolved (such that if you call `current_dir` in the running `Command`, it
833    /// returns resolved symlink). This breaks some snapshot tests, since we _don't_ want to
834    /// resolve symlinks for user-provided paths.
835    pub fn test_bucket_dir() -> PathBuf {
836        std::env::temp_dir()
837            .simple_canonicalize()
838            .expect("failed to canonicalize temp dir")
839            .join("uv")
840            .join("tests")
841    }
842
843    /// Create a new test context with multiple Python versions and explicit uv binary path.
844    ///
845    /// Does not create a virtual environment by default, but the first Python version
846    /// can be used to create a virtual environment with [`TestContext::create_venv`].
847    ///
848    /// This is called by the `test_context_with_versions!` macro.
849    pub fn new_with_versions_and_bin(python_versions: &[&str], uv_bin: PathBuf) -> Self {
850        let bucket = Self::test_bucket_dir();
851        fs_err::create_dir_all(&bucket).expect("Failed to create test bucket");
852
853        let root = tempfile::TempDir::new_in(bucket).expect("Failed to create test root directory");
854
855        // Create a `.git` directory to isolate tests that search for git boundaries from the state
856        // of the file system
857        fs_err::create_dir_all(root.path().join(".git"))
858            .expect("Failed to create `.git` placeholder in test root directory");
859
860        let temp_dir = ChildPath::new(root.path()).child("temp");
861        fs_err::create_dir_all(&temp_dir).expect("Failed to create test working directory");
862
863        let cache_dir = ChildPath::new(root.path()).child("cache");
864        fs_err::create_dir_all(&cache_dir).expect("Failed to create test cache directory");
865
866        let python_dir = ChildPath::new(root.path()).child("python");
867        fs_err::create_dir_all(&python_dir).expect("Failed to create test Python directory");
868
869        let bin_dir = ChildPath::new(root.path()).child("bin");
870        fs_err::create_dir_all(&bin_dir).expect("Failed to create test bin directory");
871
872        // When the `git` feature is disabled, enforce that the test suite does not use `git`
873        if cfg!(not(feature = "git")) {
874            Self::disallow_git_cli(&bin_dir).expect("Failed to setup disallowed `git` command");
875        }
876
877        let home_dir = ChildPath::new(root.path()).child("home");
878        fs_err::create_dir_all(&home_dir).expect("Failed to create test home directory");
879
880        let user_config_dir = if cfg!(windows) {
881            ChildPath::new(home_dir.path())
882        } else {
883            ChildPath::new(home_dir.path()).child(".config")
884        };
885
886        // Canonicalize the temp dir for consistent snapshot behavior
887        let canonical_temp_dir = temp_dir.canonicalize().unwrap();
888        let venv = ChildPath::new(canonical_temp_dir.join(".venv"));
889
890        let python_version = python_versions
891            .first()
892            .map(|version| PythonVersion::from_str(version).unwrap());
893
894        let site_packages = python_version
895            .as_ref()
896            .map(|version| site_packages_path(&venv, &format!("python{version}")));
897
898        // The workspace root directory is not available without walking up the tree
899        // https://github.com/rust-lang/cargo/issues/3946
900        let workspace_root = Path::new(&env::var(EnvVars::CARGO_MANIFEST_DIR).unwrap())
901            .parent()
902            .expect("CARGO_MANIFEST_DIR should be nested in workspace")
903            .parent()
904            .expect("CARGO_MANIFEST_DIR should be doubly nested in workspace")
905            .to_path_buf();
906
907        let download_list = ManagedPythonDownloadList::new_only_embedded().unwrap();
908
909        let python_versions: Vec<_> = python_versions
910            .iter()
911            .map(|version| PythonVersion::from_str(version).unwrap())
912            .zip(
913                python_installations_for_versions(&temp_dir, python_versions, &download_list)
914                    .expect("Failed to find test Python versions"),
915            )
916            .collect();
917
918        // Construct directories for each Python executable on Unix where the executable names
919        // need to be normalized
920        if cfg!(unix) {
921            for (version, executable) in &python_versions {
922                let parent = python_dir.child(version.to_string());
923                parent.create_dir_all().unwrap();
924                parent.child("python3").symlink_to_file(executable).unwrap();
925            }
926        }
927
928        let mut filters = Vec::new();
929
930        filters.extend(
931            Self::path_patterns(&uv_bin)
932                .into_iter()
933                .map(|pattern| (pattern, "[UV]".to_string())),
934        );
935
936        // Exclude `link-mode` on Windows since we set it in the remote test suite
937        if cfg!(windows) {
938            filters.push((" --link-mode <LINK_MODE>".to_string(), String::new()));
939            filters.push((r#"link-mode = "copy"\n"#.to_string(), String::new()));
940            // Unix uses "exit status", Windows uses "exit code"
941            filters.push((r"exit code: ".to_string(), "exit status: ".to_string()));
942        }
943
944        for (version, executable) in &python_versions {
945            // Add filtering for the interpreter path
946            filters.extend(
947                Self::path_patterns(executable)
948                    .into_iter()
949                    .map(|pattern| (pattern, format!("[PYTHON-{version}]"))),
950            );
951
952            // And for the symlink we created in the test the Python path
953            filters.extend(
954                Self::path_patterns(python_dir.join(version.to_string()))
955                    .into_iter()
956                    .map(|pattern| {
957                        (
958                            format!("{pattern}[a-zA-Z0-9]*"),
959                            format!("[PYTHON-{version}]"),
960                        )
961                    }),
962            );
963
964            // Add Python patch version filtering unless explicitly requested to ensure
965            // snapshots are patch version agnostic when it is not a part of the test.
966            if version.patch().is_none() {
967                filters.push((
968                    format!(r"({})\.\d+", regex::escape(version.to_string().as_str())),
969                    "$1.[X]".to_string(),
970                ));
971            }
972        }
973
974        filters.extend(
975            Self::path_patterns(&bin_dir)
976                .into_iter()
977                .map(|pattern| (pattern, "[BIN]/".to_string())),
978        );
979        filters.extend(
980            Self::path_patterns(&cache_dir)
981                .into_iter()
982                .map(|pattern| (pattern, "[CACHE_DIR]/".to_string())),
983        );
984        if let Some(ref site_packages) = site_packages {
985            filters.extend(
986                Self::path_patterns(site_packages)
987                    .into_iter()
988                    .map(|pattern| (pattern, "[SITE_PACKAGES]/".to_string())),
989            );
990        }
991        filters.extend(
992            Self::path_patterns(&venv)
993                .into_iter()
994                .map(|pattern| (pattern, "[VENV]/".to_string())),
995        );
996
997        // Account for [`Simplified::user_display`] which is relative to the command working directory
998        if let Some(site_packages) = site_packages {
999            filters.push((
1000                Self::path_pattern(
1001                    site_packages
1002                        .strip_prefix(&canonical_temp_dir)
1003                        .expect("The test site-packages directory is always in the tempdir"),
1004                ),
1005                "[SITE_PACKAGES]/".to_string(),
1006            ));
1007        }
1008
1009        // Filter Python library path differences between Windows and Unix
1010        filters.push((
1011            r"[\\/]lib[\\/]python\d+\.\d+[\\/]".to_string(),
1012            "/[PYTHON-LIB]/".to_string(),
1013        ));
1014        filters.push((r"[\\/]Lib[\\/]".to_string(), "/[PYTHON-LIB]/".to_string()));
1015
1016        filters.extend(
1017            Self::path_patterns(&temp_dir)
1018                .into_iter()
1019                .map(|pattern| (pattern, "[TEMP_DIR]/".to_string())),
1020        );
1021        filters.extend(
1022            Self::path_patterns(&python_dir)
1023                .into_iter()
1024                .map(|pattern| (pattern, "[PYTHON_DIR]/".to_string())),
1025        );
1026        let mut uv_user_config_dir = PathBuf::from(user_config_dir.path());
1027        uv_user_config_dir.push("uv");
1028        filters.extend(
1029            Self::path_patterns(&uv_user_config_dir)
1030                .into_iter()
1031                .map(|pattern| (pattern, "[UV_USER_CONFIG_DIR]/".to_string())),
1032        );
1033        filters.extend(
1034            Self::path_patterns(&user_config_dir)
1035                .into_iter()
1036                .map(|pattern| (pattern, "[USER_CONFIG_DIR]/".to_string())),
1037        );
1038        filters.extend(
1039            Self::path_patterns(&home_dir)
1040                .into_iter()
1041                .map(|pattern| (pattern, "[HOME]/".to_string())),
1042        );
1043        filters.extend(
1044            Self::path_patterns(&workspace_root)
1045                .into_iter()
1046                .map(|pattern| (pattern, "[WORKSPACE]/".to_string())),
1047        );
1048
1049        // Make virtual environment activation cross-platform and shell-agnostic
1050        filters.push((
1051            r"Activate with: (.*)\\Scripts\\activate".to_string(),
1052            "Activate with: source $1/[BIN]/activate".to_string(),
1053        ));
1054        filters.push((
1055            r"Activate with: Scripts\\activate".to_string(),
1056            "Activate with: source [BIN]/activate".to_string(),
1057        ));
1058        filters.push((
1059            r"Activate with: source (.*/|)bin/activate(?:\.\w+)?".to_string(),
1060            "Activate with: source $1[BIN]/activate".to_string(),
1061        ));
1062
1063        // Filter non-deterministic temporary directory names
1064        // Note we apply this _after_ all the full paths to avoid breaking their matching
1065        filters.push((r"(\\|\/)\.tmp.*(\\|\/)".to_string(), "/[TMP]/".to_string()));
1066
1067        // Account for platform prefix differences `file://` (Unix) vs `file:///` (Windows)
1068        filters.push((r"file:///".to_string(), "file://".to_string()));
1069
1070        // Destroy any remaining UNC prefixes (Windows only)
1071        filters.push((r"\\\\\?\\".to_string(), String::new()));
1072
1073        // Remove the version from the packse url in lockfile snapshots. This avoids having a huge
1074        // diff any time we upgrade packse
1075        filters.push((
1076            format!("https://astral-sh.github.io/packse/{PACKSE_VERSION}"),
1077            "https://astral-sh.github.io/packse/PACKSE_VERSION".to_string(),
1078        ));
1079        // Developer convenience
1080        if let Ok(packse_test_index) = env::var(EnvVars::UV_TEST_PACKSE_INDEX) {
1081            filters.push((
1082                packse_test_index.trim_end_matches('/').to_string(),
1083                "https://astral-sh.github.io/packse/PACKSE_VERSION".to_string(),
1084            ));
1085        }
1086        // For wiremock tests
1087        filters.push((r"127\.0\.0\.1:\d*".to_string(), "[LOCALHOST]".to_string()));
1088        // Avoid breaking the tests when bumping the uv version
1089        filters.push((
1090            format!(
1091                r#"requires = \["uv_build>={},<[0-9.]+"\]"#,
1092                uv_version::version()
1093            ),
1094            r#"requires = ["uv_build>=[CURRENT_VERSION],<[NEXT_BREAKING]"]"#.to_string(),
1095        ));
1096        // Filter script environment hashes
1097        filters.push((
1098            r"environments-v(\d+)[\\/](\w+)-[a-z0-9]+".to_string(),
1099            "environments-v$1/$2-[HASH]".to_string(),
1100        ));
1101        // Filter archive hashes
1102        filters.push((
1103            r"archive-v(\d+)[\\/][A-Za-z0-9\-\_]+".to_string(),
1104            "archive-v$1/[HASH]".to_string(),
1105        ));
1106
1107        Self {
1108            root: ChildPath::new(root.path()),
1109            temp_dir,
1110            cache_dir,
1111            python_dir,
1112            home_dir,
1113            user_config_dir,
1114            bin_dir,
1115            venv,
1116            workspace_root,
1117            python_version,
1118            python_versions,
1119            uv_bin,
1120            filters,
1121            extra_env: vec![],
1122            _root: root,
1123            _extra_tempdirs: vec![],
1124        }
1125    }
1126
1127    /// Create a uv command for testing.
1128    pub fn command(&self) -> Command {
1129        let mut command = self.new_command();
1130        self.add_shared_options(&mut command, true);
1131        command
1132    }
1133
1134    pub fn disallow_git_cli(bin_dir: &Path) -> std::io::Result<()> {
1135        let contents = r"#!/bin/sh
1136    echo 'error: `git` operations are not allowed — are you missing a cfg for the `git` feature?' >&2
1137    exit 127";
1138        let git = bin_dir.join(format!("git{}", env::consts::EXE_SUFFIX));
1139        fs_err::write(&git, contents)?;
1140
1141        #[cfg(unix)]
1142        {
1143            use std::os::unix::fs::PermissionsExt;
1144            let mut perms = fs_err::metadata(&git)?.permissions();
1145            perms.set_mode(0o755);
1146            fs_err::set_permissions(&git, perms)?;
1147        }
1148
1149        Ok(())
1150    }
1151
1152    /// Setup Git LFS Filters
1153    ///
1154    /// You can find the default filters in <https://github.com/git-lfs/git-lfs/blob/v3.7.1/lfs/attribute.go#L66-L71>
1155    /// We set required to true to get a full stacktrace when these commands fail.
1156    #[must_use]
1157    pub fn with_git_lfs_config(mut self) -> Self {
1158        let git_lfs_config = self.root.child(".gitconfig");
1159        git_lfs_config
1160            .write_str(indoc! {r#"
1161                [filter "lfs"]
1162                    clean = git-lfs clean -- %f
1163                    smudge = git-lfs smudge -- %f
1164                    process = git-lfs filter-process
1165                    required = true
1166            "#})
1167            .expect("Failed to setup `git-lfs` filters");
1168
1169        // Its possible your system config can cause conflicts with the Git LFS tests.
1170        // In such cases, add self.extra_env.push(("GIT_CONFIG_NOSYSTEM".into(), "1".into()));
1171        self.extra_env.push((
1172            EnvVars::GIT_CONFIG_GLOBAL.into(),
1173            git_lfs_config.as_os_str().into(),
1174        ));
1175        self
1176    }
1177
1178    /// Shared behaviour for almost all test commands.
1179    ///
1180    /// * Use a temporary cache directory
1181    /// * Use a temporary virtual environment with the Python version of [`Self`]
1182    /// * Don't wrap text output based on the terminal we're in, the test output doesn't get printed
1183    ///   but snapshotted to a string.
1184    /// * Use a fake `HOME` to avoid accidentally changing the developer's machine.
1185    /// * Hide other Pythons with `UV_PYTHON_INSTALL_DIR` and installed interpreters with
1186    ///   `UV_TEST_PYTHON_PATH` and an active venv (if applicable) by removing `VIRTUAL_ENV`.
1187    /// * Increase the stack size to avoid stack overflows on windows due to large async functions.
1188    pub fn add_shared_options(&self, command: &mut Command, activate_venv: bool) {
1189        self.add_shared_args(command);
1190        self.add_shared_env(command, activate_venv);
1191    }
1192
1193    /// Only the arguments of [`TestContext::add_shared_options`].
1194    pub fn add_shared_args(&self, command: &mut Command) {
1195        command.arg("--cache-dir").arg(self.cache_dir.path());
1196    }
1197
1198    /// Only the environment variables of [`TestContext::add_shared_options`].
1199    pub fn add_shared_env(&self, command: &mut Command, activate_venv: bool) {
1200        // Push the test context bin to the front of the PATH
1201        let path = env::join_paths(std::iter::once(self.bin_dir.to_path_buf()).chain(
1202            env::split_paths(&env::var(EnvVars::PATH).unwrap_or_default()),
1203        ))
1204        .unwrap();
1205
1206        // Ensure the tests aren't sensitive to the running user's shell without forcing
1207        // `bash` on Windows
1208        if cfg!(not(windows)) {
1209            command.env(EnvVars::SHELL, "bash");
1210        }
1211
1212        command
1213            // When running the tests in a venv, ignore that venv, otherwise we'll capture warnings.
1214            .env_remove(EnvVars::VIRTUAL_ENV)
1215            // Disable wrapping of uv output for readability / determinism in snapshots.
1216            .env(EnvVars::UV_NO_WRAP, "1")
1217            // While we disable wrapping in uv above, invoked tools may still wrap their output so
1218            // we set a fixed `COLUMNS` value for isolation from terminal width.
1219            .env(EnvVars::COLUMNS, "100")
1220            .env(EnvVars::PATH, path)
1221            .env(EnvVars::HOME, self.home_dir.as_os_str())
1222            .env(EnvVars::APPDATA, self.home_dir.as_os_str())
1223            .env(EnvVars::USERPROFILE, self.home_dir.as_os_str())
1224            .env(
1225                EnvVars::XDG_CONFIG_DIRS,
1226                self.home_dir.join("config").as_os_str(),
1227            )
1228            .env(
1229                EnvVars::XDG_DATA_HOME,
1230                self.home_dir.join("data").as_os_str(),
1231            )
1232            .env(EnvVars::UV_PYTHON_INSTALL_DIR, "")
1233            // Installations are not allowed by default; see `Self::with_managed_python_dirs`
1234            .env(EnvVars::UV_PYTHON_DOWNLOADS, "never")
1235            .env(EnvVars::UV_TEST_PYTHON_PATH, self.python_path())
1236            // Lock to a point in time view of the world
1237            .env(EnvVars::UV_EXCLUDE_NEWER, EXCLUDE_NEWER)
1238            .env(EnvVars::UV_TEST_CURRENT_TIMESTAMP, EXCLUDE_NEWER)
1239            // When installations are allowed, we don't want to write to global state, like the
1240            // Windows registry
1241            .env(EnvVars::UV_PYTHON_INSTALL_REGISTRY, "0")
1242            // Since downloads, fetches and builds run in parallel, their message output order is
1243            // non-deterministic, so can't capture them in test output.
1244            .env(EnvVars::UV_TEST_NO_CLI_PROGRESS, "1")
1245            // I believe the intent of all tests is that they are run outside the
1246            // context of an existing git repository. And when they aren't, state
1247            // from the parent git repository can bleed into the behavior of `uv
1248            // init` in a way that makes it difficult to test consistently. By
1249            // setting GIT_CEILING_DIRECTORIES, we specifically prevent git from
1250            // climbing up past the root of our test directory to look for any
1251            // other git repos.
1252            //
1253            // If one wants to write a test specifically targeting uv within a
1254            // pre-existing git repository, then the test should make the parent
1255            // git repo explicitly. The GIT_CEILING_DIRECTORIES here shouldn't
1256            // impact it, since it only prevents git from discovering repositories
1257            // at or above the root.
1258            .env(EnvVars::GIT_CEILING_DIRECTORIES, self.root.path())
1259            .current_dir(self.temp_dir.path());
1260
1261        for (key, value) in &self.extra_env {
1262            command.env(key, value);
1263        }
1264
1265        if activate_venv {
1266            command.env(EnvVars::VIRTUAL_ENV, self.venv.as_os_str());
1267        }
1268
1269        if cfg!(unix) {
1270            // Avoid locale issues in tests
1271            command.env(EnvVars::LC_ALL, "C");
1272        }
1273    }
1274
1275    /// Create a `pip compile` command for testing.
1276    pub fn pip_compile(&self) -> Command {
1277        let mut command = self.new_command();
1278        command.arg("pip").arg("compile");
1279        self.add_shared_options(&mut command, true);
1280        command
1281    }
1282
1283    /// Create a `pip compile` command for testing.
1284    pub fn pip_sync(&self) -> Command {
1285        let mut command = self.new_command();
1286        command.arg("pip").arg("sync");
1287        self.add_shared_options(&mut command, true);
1288        command
1289    }
1290
1291    pub fn pip_show(&self) -> Command {
1292        let mut command = self.new_command();
1293        command.arg("pip").arg("show");
1294        self.add_shared_options(&mut command, true);
1295        command
1296    }
1297
1298    /// Create a `pip freeze` command with options shared across scenarios.
1299    pub fn pip_freeze(&self) -> Command {
1300        let mut command = self.new_command();
1301        command.arg("pip").arg("freeze");
1302        self.add_shared_options(&mut command, true);
1303        command
1304    }
1305
1306    /// Create a `pip check` command with options shared across scenarios.
1307    pub fn pip_check(&self) -> Command {
1308        let mut command = self.new_command();
1309        command.arg("pip").arg("check");
1310        self.add_shared_options(&mut command, true);
1311        command
1312    }
1313
1314    pub fn pip_list(&self) -> Command {
1315        let mut command = self.new_command();
1316        command.arg("pip").arg("list");
1317        self.add_shared_options(&mut command, true);
1318        command
1319    }
1320
1321    /// Create a `uv venv` command
1322    pub fn venv(&self) -> Command {
1323        let mut command = self.new_command();
1324        command.arg("venv");
1325        self.add_shared_options(&mut command, false);
1326        command
1327    }
1328
1329    /// Create a `pip install` command with options shared across scenarios.
1330    pub fn pip_install(&self) -> Command {
1331        let mut command = self.new_command();
1332        command.arg("pip").arg("install");
1333        self.add_shared_options(&mut command, true);
1334        command
1335    }
1336
1337    /// Create a `pip uninstall` command with options shared across scenarios.
1338    pub fn pip_uninstall(&self) -> Command {
1339        let mut command = self.new_command();
1340        command.arg("pip").arg("uninstall");
1341        self.add_shared_options(&mut command, true);
1342        command
1343    }
1344
1345    /// Create a `pip tree` command for testing.
1346    pub fn pip_tree(&self) -> Command {
1347        let mut command = self.new_command();
1348        command.arg("pip").arg("tree");
1349        self.add_shared_options(&mut command, true);
1350        command
1351    }
1352
1353    /// Create a `pip debug` command for testing.
1354    pub fn pip_debug(&self) -> Command {
1355        let mut command = self.new_command();
1356        command.arg("pip").arg("debug");
1357        self.add_shared_options(&mut command, true);
1358        command
1359    }
1360
1361    /// Create a `uv help` command with options shared across scenarios.
1362    pub fn help(&self) -> Command {
1363        let mut command = self.new_command();
1364        command.arg("help");
1365        self.add_shared_env(&mut command, false);
1366        command
1367    }
1368
1369    /// Create a `uv init` command with options shared across scenarios and
1370    /// isolated from any git repository that may exist in a parent directory.
1371    pub fn init(&self) -> Command {
1372        let mut command = self.new_command();
1373        command.arg("init");
1374        self.add_shared_options(&mut command, false);
1375        command
1376    }
1377
1378    /// Create a `uv sync` command with options shared across scenarios.
1379    pub fn sync(&self) -> Command {
1380        let mut command = self.new_command();
1381        command.arg("sync");
1382        self.add_shared_options(&mut command, false);
1383        command
1384    }
1385
1386    /// Create a `uv lock` command with options shared across scenarios.
1387    pub fn lock(&self) -> Command {
1388        let mut command = self.new_command();
1389        command.arg("lock");
1390        self.add_shared_options(&mut command, false);
1391        command
1392    }
1393
1394    /// Create a `uv audit` command with options shared across scenarios.
1395    pub fn audit(&self) -> Command {
1396        let mut command = self.new_command();
1397        command.arg("audit");
1398        self.add_shared_options(&mut command, false);
1399        command
1400    }
1401
1402    /// Create a `uv workspace metadata` command with options shared across scenarios.
1403    pub fn workspace_metadata(&self) -> Command {
1404        let mut command = self.new_command();
1405        command.arg("workspace").arg("metadata");
1406        self.add_shared_options(&mut command, false);
1407        command
1408    }
1409
1410    /// Create a `uv workspace dir` command with options shared across scenarios.
1411    pub fn workspace_dir(&self) -> Command {
1412        let mut command = self.new_command();
1413        command.arg("workspace").arg("dir");
1414        self.add_shared_options(&mut command, false);
1415        command
1416    }
1417
1418    /// Create a `uv workspace list` command with options shared across scenarios.
1419    pub fn workspace_list(&self) -> Command {
1420        let mut command = self.new_command();
1421        command.arg("workspace").arg("list");
1422        self.add_shared_options(&mut command, false);
1423        command
1424    }
1425
1426    /// Create a `uv export` command with options shared across scenarios.
1427    pub fn export(&self) -> Command {
1428        let mut command = self.new_command();
1429        command.arg("export");
1430        self.add_shared_options(&mut command, false);
1431        command
1432    }
1433
1434    /// Create a `uv format` command with options shared across scenarios.
1435    pub fn format(&self) -> Command {
1436        let mut command = self.new_command();
1437        command.arg("format");
1438        self.add_shared_options(&mut command, false);
1439        // Override to a more recent date for ruff version resolution
1440        command.env(EnvVars::UV_EXCLUDE_NEWER, "2026-02-15T00:00:00Z");
1441        command
1442    }
1443
1444    /// Create a `uv build` command with options shared across scenarios.
1445    pub fn build(&self) -> Command {
1446        let mut command = self.new_command();
1447        command.arg("build");
1448        self.add_shared_options(&mut command, false);
1449        command
1450    }
1451
1452    pub fn version(&self) -> Command {
1453        let mut command = self.new_command();
1454        command.arg("version");
1455        self.add_shared_options(&mut command, false);
1456        command
1457    }
1458
1459    pub fn self_version(&self) -> Command {
1460        let mut command = self.new_command();
1461        command.arg("self").arg("version");
1462        self.add_shared_options(&mut command, false);
1463        command
1464    }
1465
1466    pub fn self_update(&self) -> Command {
1467        let mut command = self.new_command();
1468        command.arg("self").arg("update");
1469        self.add_shared_options(&mut command, false);
1470        command
1471    }
1472
1473    /// Create a `uv publish` command with options shared across scenarios.
1474    pub fn publish(&self) -> Command {
1475        let mut command = self.new_command();
1476        command.arg("publish");
1477        self.add_shared_options(&mut command, false);
1478        command
1479    }
1480
1481    /// Create a `uv python find` command with options shared across scenarios.
1482    pub fn python_find(&self) -> Command {
1483        let mut command = self.new_command();
1484        command
1485            .arg("python")
1486            .arg("find")
1487            .env(EnvVars::UV_PREVIEW, "1")
1488            .env(EnvVars::UV_PYTHON_INSTALL_DIR, "");
1489        self.add_shared_options(&mut command, false);
1490        command
1491    }
1492
1493    /// Create a `uv python list` command with options shared across scenarios.
1494    pub fn python_list(&self) -> Command {
1495        let mut command = self.new_command();
1496        command
1497            .arg("python")
1498            .arg("list")
1499            .env(EnvVars::UV_PYTHON_INSTALL_DIR, "");
1500        self.add_shared_options(&mut command, false);
1501        command
1502    }
1503
1504    /// Create a `uv python install` command with options shared across scenarios.
1505    pub fn python_install(&self) -> Command {
1506        let mut command = self.new_command();
1507        command.arg("python").arg("install");
1508        self.add_shared_options(&mut command, true);
1509        command
1510    }
1511
1512    /// Create a `uv python uninstall` command with options shared across scenarios.
1513    pub fn python_uninstall(&self) -> Command {
1514        let mut command = self.new_command();
1515        command.arg("python").arg("uninstall");
1516        self.add_shared_options(&mut command, true);
1517        command
1518    }
1519
1520    /// Create a `uv python upgrade` command with options shared across scenarios.
1521    pub fn python_upgrade(&self) -> Command {
1522        let mut command = self.new_command();
1523        command.arg("python").arg("upgrade");
1524        self.add_shared_options(&mut command, true);
1525        command
1526    }
1527
1528    /// Create a `uv python pin` command with options shared across scenarios.
1529    pub fn python_pin(&self) -> Command {
1530        let mut command = self.new_command();
1531        command.arg("python").arg("pin");
1532        self.add_shared_options(&mut command, true);
1533        command
1534    }
1535
1536    /// Create a `uv python dir` command with options shared across scenarios.
1537    pub fn python_dir(&self) -> Command {
1538        let mut command = self.new_command();
1539        command.arg("python").arg("dir");
1540        self.add_shared_options(&mut command, true);
1541        command
1542    }
1543
1544    /// Create a `uv run` command with options shared across scenarios.
1545    pub fn run(&self) -> Command {
1546        let mut command = self.new_command();
1547        command.arg("run").env(EnvVars::UV_SHOW_RESOLUTION, "1");
1548        self.add_shared_options(&mut command, true);
1549        command
1550    }
1551
1552    /// Create a `uv tool run` command with options shared across scenarios.
1553    pub fn tool_run(&self) -> Command {
1554        let mut command = self.new_command();
1555        command
1556            .arg("tool")
1557            .arg("run")
1558            .env(EnvVars::UV_SHOW_RESOLUTION, "1");
1559        self.add_shared_options(&mut command, false);
1560        command
1561    }
1562
1563    /// Create a `uv upgrade run` command with options shared across scenarios.
1564    pub fn tool_upgrade(&self) -> Command {
1565        let mut command = self.new_command();
1566        command.arg("tool").arg("upgrade");
1567        self.add_shared_options(&mut command, false);
1568        command
1569    }
1570
1571    /// Create a `uv tool install` command with options shared across scenarios.
1572    pub fn tool_install(&self) -> Command {
1573        let mut command = self.new_command();
1574        command.arg("tool").arg("install");
1575        self.add_shared_options(&mut command, false);
1576        command
1577    }
1578
1579    /// Create a `uv tool list` command with options shared across scenarios.
1580    pub fn tool_list(&self) -> Command {
1581        let mut command = self.new_command();
1582        command.arg("tool").arg("list");
1583        self.add_shared_options(&mut command, false);
1584        command
1585    }
1586
1587    /// Create a `uv tool dir` command with options shared across scenarios.
1588    pub fn tool_dir(&self) -> Command {
1589        let mut command = self.new_command();
1590        command.arg("tool").arg("dir");
1591        self.add_shared_options(&mut command, false);
1592        command
1593    }
1594
1595    /// Create a `uv tool uninstall` command with options shared across scenarios.
1596    pub fn tool_uninstall(&self) -> Command {
1597        let mut command = self.new_command();
1598        command.arg("tool").arg("uninstall");
1599        self.add_shared_options(&mut command, false);
1600        command
1601    }
1602
1603    /// Create a `uv add` command for the given requirements.
1604    pub fn add(&self) -> Command {
1605        let mut command = self.new_command();
1606        command.arg("add");
1607        self.add_shared_options(&mut command, false);
1608        command
1609    }
1610
1611    /// Create a `uv remove` command for the given requirements.
1612    pub fn remove(&self) -> Command {
1613        let mut command = self.new_command();
1614        command.arg("remove");
1615        self.add_shared_options(&mut command, false);
1616        command
1617    }
1618
1619    /// Create a `uv tree` command with options shared across scenarios.
1620    pub fn tree(&self) -> Command {
1621        let mut command = self.new_command();
1622        command.arg("tree");
1623        self.add_shared_options(&mut command, false);
1624        command
1625    }
1626
1627    /// Create a `uv cache clean` command.
1628    pub fn clean(&self) -> Command {
1629        let mut command = self.new_command();
1630        command.arg("cache").arg("clean");
1631        self.add_shared_options(&mut command, false);
1632        command
1633    }
1634
1635    /// Create a `uv cache prune` command.
1636    pub fn prune(&self) -> Command {
1637        let mut command = self.new_command();
1638        command.arg("cache").arg("prune");
1639        self.add_shared_options(&mut command, false);
1640        command
1641    }
1642
1643    /// Create a `uv cache size` command.
1644    pub fn cache_size(&self) -> Command {
1645        let mut command = self.new_command();
1646        command.arg("cache").arg("size");
1647        self.add_shared_options(&mut command, false);
1648        command
1649    }
1650
1651    /// Create a `uv build_backend` command.
1652    ///
1653    /// Note that this command is hidden and only invoking it through a build frontend is supported.
1654    pub fn build_backend(&self) -> Command {
1655        let mut command = self.new_command();
1656        command.arg("build-backend");
1657        self.add_shared_options(&mut command, false);
1658        command
1659    }
1660
1661    /// The path to the Python interpreter in the venv.
1662    ///
1663    /// Don't use this for `Command::new`, use `Self::python_command` instead.
1664    pub fn interpreter(&self) -> PathBuf {
1665        let venv = &self.venv;
1666        if cfg!(unix) {
1667            venv.join("bin").join("python")
1668        } else if cfg!(windows) {
1669            venv.join("Scripts").join("python.exe")
1670        } else {
1671            unimplemented!("Only Windows and Unix are supported")
1672        }
1673    }
1674
1675    pub fn python_command(&self) -> Command {
1676        let mut interpreter = self.interpreter();
1677
1678        // If there's not a virtual environment, use the first Python interpreter in the context
1679        if !interpreter.exists() {
1680            interpreter.clone_from(
1681                &self
1682                    .python_versions
1683                    .first()
1684                    .expect("At least one Python version is required")
1685                    .1,
1686            );
1687        }
1688
1689        let mut command = Self::new_command_with(&interpreter);
1690        command
1691            // Our tests change files in <1s, so we must disable CPython bytecode caching or we'll get stale files
1692            // https://github.com/python/cpython/issues/75953
1693            .arg("-B")
1694            // Python on windows
1695            .env(EnvVars::PYTHONUTF8, "1");
1696
1697        self.add_shared_env(&mut command, false);
1698
1699        command
1700    }
1701
1702    /// Create a `uv auth login` command.
1703    pub fn auth_login(&self) -> Command {
1704        let mut command = self.new_command();
1705        command.arg("auth").arg("login");
1706        self.add_shared_options(&mut command, false);
1707        command
1708    }
1709
1710    /// Create a `uv auth logout` command.
1711    pub fn auth_logout(&self) -> Command {
1712        let mut command = self.new_command();
1713        command.arg("auth").arg("logout");
1714        self.add_shared_options(&mut command, false);
1715        command
1716    }
1717
1718    /// Create a `uv auth helper --protocol bazel get` command.
1719    pub fn auth_helper(&self) -> Command {
1720        let mut command = self.new_command();
1721        command.arg("auth").arg("helper");
1722        self.add_shared_options(&mut command, false);
1723        command
1724    }
1725
1726    /// Create a `uv auth token` command.
1727    pub fn auth_token(&self) -> Command {
1728        let mut command = self.new_command();
1729        command.arg("auth").arg("token");
1730        self.add_shared_options(&mut command, false);
1731        command
1732    }
1733
1734    /// Set `HOME` to the real home directory.
1735    ///
1736    /// We need this for testing commands which use the macOS keychain.
1737    #[must_use]
1738    pub fn with_real_home(mut self) -> Self {
1739        if let Some(home) = env::var_os(EnvVars::HOME) {
1740            self.extra_env
1741                .push((EnvVars::HOME.to_string().into(), home));
1742        }
1743        // Use the test's isolated config directory to avoid reading user
1744        // configuration files (like `.python-version`) that could interfere with tests.
1745        self.extra_env.push((
1746            EnvVars::XDG_CONFIG_HOME.into(),
1747            self.user_config_dir.as_os_str().into(),
1748        ));
1749        self
1750    }
1751
1752    /// Run the given python code and check whether it succeeds.
1753    pub fn assert_command(&self, command: &str) -> Assert {
1754        self.python_command()
1755            .arg("-c")
1756            .arg(command)
1757            .current_dir(&self.temp_dir)
1758            .assert()
1759    }
1760
1761    /// Run the given python file and check whether it succeeds.
1762    pub fn assert_file(&self, file: impl AsRef<Path>) -> Assert {
1763        self.python_command()
1764            .arg(file.as_ref())
1765            .current_dir(&self.temp_dir)
1766            .assert()
1767    }
1768
1769    /// Assert a package is installed with the given version.
1770    pub fn assert_installed(&self, package: &'static str, version: &'static str) {
1771        self.assert_command(
1772            format!("import {package} as package; print(package.__version__, end='')").as_str(),
1773        )
1774        .success()
1775        .stdout(version);
1776    }
1777
1778    /// Assert a package is not installed.
1779    pub fn assert_not_installed(&self, package: &'static str) {
1780        self.assert_command(format!("import {package}").as_str())
1781            .failure();
1782    }
1783
1784    /// Generate various escaped regex patterns for the given path.
1785    pub fn path_patterns(path: impl AsRef<Path>) -> Vec<String> {
1786        let mut patterns = Vec::new();
1787
1788        // We can only canonicalize paths that exist already
1789        if path.as_ref().exists() {
1790            patterns.push(Self::path_pattern(
1791                path.as_ref()
1792                    .canonicalize()
1793                    .expect("Failed to create canonical path"),
1794            ));
1795        }
1796
1797        // Include a non-canonicalized version
1798        patterns.push(Self::path_pattern(path));
1799
1800        patterns
1801    }
1802
1803    /// Generate an escaped regex pattern for the given path.
1804    fn path_pattern(path: impl AsRef<Path>) -> String {
1805        format!(
1806            // Trim the trailing separator for cross-platform directories filters
1807            r"{}\\?/?",
1808            regex::escape(&path.as_ref().simplified_display().to_string())
1809                // Make separators platform agnostic because on Windows we will display
1810                // paths with Unix-style separators sometimes
1811                .replace(r"\\", r"(\\|\/)")
1812        )
1813    }
1814
1815    pub fn python_path(&self) -> OsString {
1816        if cfg!(unix) {
1817            // On Unix, we needed to normalize the Python executable names to `python3` for the tests
1818            env::join_paths(
1819                self.python_versions
1820                    .iter()
1821                    .map(|(version, _)| self.python_dir.join(version.to_string())),
1822            )
1823            .unwrap()
1824        } else {
1825            // On Windows, just join the parent directories of the executables
1826            env::join_paths(
1827                self.python_versions
1828                    .iter()
1829                    .map(|(_, executable)| executable.parent().unwrap().to_path_buf()),
1830            )
1831            .unwrap()
1832        }
1833    }
1834
1835    /// Standard snapshot filters _plus_ those for this test context.
1836    pub fn filters(&self) -> Vec<(&str, &str)> {
1837        // Put test context snapshots before the default filters
1838        // This ensures we don't replace other patterns inside paths from the test context first
1839        self.filters
1840            .iter()
1841            .map(|(p, r)| (p.as_str(), r.as_str()))
1842            .chain(INSTA_FILTERS.iter().copied())
1843            .collect()
1844    }
1845
1846    /// Only the filters added to this test context.
1847    pub fn filters_without_standard_filters(&self) -> Vec<(&str, &str)> {
1848        self.filters
1849            .iter()
1850            .map(|(p, r)| (p.as_str(), r.as_str()))
1851            .collect()
1852    }
1853
1854    /// For when we add pypy to the test suite.
1855    #[allow(clippy::unused_self)]
1856    pub fn python_kind(&self) -> &'static str {
1857        "python"
1858    }
1859
1860    /// Returns the site-packages folder inside the venv.
1861    pub fn site_packages(&self) -> PathBuf {
1862        site_packages_path(
1863            &self.venv,
1864            &format!(
1865                "{}{}",
1866                self.python_kind(),
1867                self.python_version.as_ref().expect(
1868                    "A Python version must be provided to retrieve the test site packages path"
1869                )
1870            ),
1871        )
1872    }
1873
1874    /// Reset the virtual environment in the test context.
1875    pub fn reset_venv(&self) {
1876        self.create_venv();
1877    }
1878
1879    /// Create a new virtual environment named `.venv` in the test context.
1880    fn create_venv(&self) {
1881        let executable = get_python(
1882            self.python_version
1883                .as_ref()
1884                .expect("A Python version must be provided to create a test virtual environment"),
1885        );
1886        create_venv_from_executable(&self.venv, &self.cache_dir, &executable, &self.uv_bin);
1887    }
1888
1889    /// Copies the files from the ecosystem project given into this text
1890    /// context.
1891    ///
1892    /// This will almost always write at least a `pyproject.toml` into this
1893    /// test context.
1894    ///
1895    /// The given name should correspond to the name of a sub-directory (not a
1896    /// path to it) in the `test/ecosystem` directory.
1897    ///
1898    /// This panics (fails the current test) for any failure.
1899    pub fn copy_ecosystem_project(&self, name: &str) {
1900        let project_dir = PathBuf::from(format!("../../test/ecosystem/{name}"));
1901        self.temp_dir.copy_from(project_dir, &["*"]).unwrap();
1902        // If there is a (gitignore) lockfile, remove it.
1903        if let Err(err) = fs_err::remove_file(self.temp_dir.join("uv.lock")) {
1904            assert_eq!(
1905                err.kind(),
1906                io::ErrorKind::NotFound,
1907                "Failed to remove uv.lock: {err}"
1908            );
1909        }
1910    }
1911
1912    /// Creates a way to compare the changes made to a lock file.
1913    ///
1914    /// This routine starts by copying (not moves) the generated lock file to
1915    /// memory. It then calls the given closure with this test context to get a
1916    /// `Command` and runs the command. The diff between the old lock file and
1917    /// the new one is then returned.
1918    ///
1919    /// This assumes that a lock has already been performed.
1920    pub fn diff_lock(&self, change: impl Fn(&Self) -> Command) -> String {
1921        static TRIM_TRAILING_WHITESPACE: std::sync::LazyLock<Regex> =
1922            std::sync::LazyLock::new(|| Regex::new(r"(?m)^\s+$").unwrap());
1923
1924        let lock_path = ChildPath::new(self.temp_dir.join("uv.lock"));
1925        let old_lock = fs_err::read_to_string(&lock_path).unwrap();
1926        let (snapshot, _, status) = run_and_format_with_status(
1927            change(self),
1928            self.filters(),
1929            "diff_lock",
1930            Some(WindowsFilters::Platform),
1931            None,
1932        );
1933        assert!(status.success(), "{snapshot}");
1934        let new_lock = fs_err::read_to_string(&lock_path).unwrap();
1935        diff_snapshot(&old_lock, &new_lock)
1936    }
1937
1938    /// Read a file in the temporary directory
1939    pub fn read(&self, file: impl AsRef<Path>) -> String {
1940        fs_err::read_to_string(self.temp_dir.join(&file))
1941            .unwrap_or_else(|_| panic!("Missing file: `{}`", file.user_display()))
1942    }
1943
1944    /// Creates a new `Command` that is intended to be suitable for use in
1945    /// all tests.
1946    fn new_command(&self) -> Command {
1947        Self::new_command_with(&self.uv_bin)
1948    }
1949
1950    /// Creates a new `Command` that is intended to be suitable for use in
1951    /// all tests, but with the given binary.
1952    ///
1953    /// Clears environment variables defined in [`EnvVars`] to avoid reading
1954    /// test host settings.
1955    fn new_command_with(bin: &Path) -> Command {
1956        let mut command = Command::new(bin);
1957
1958        let passthrough = [
1959            // For linux distributions
1960            EnvVars::PATH,
1961            // For debugging tests.
1962            EnvVars::RUST_LOG,
1963            EnvVars::RUST_BACKTRACE,
1964            // Windows System configuration.
1965            EnvVars::SYSTEMDRIVE,
1966            // Work around small default stack sizes and large futures in debug builds.
1967            EnvVars::RUST_MIN_STACK,
1968            EnvVars::UV_STACK_SIZE,
1969            // Allow running tests with custom network settings.
1970            EnvVars::ALL_PROXY,
1971            EnvVars::HTTPS_PROXY,
1972            EnvVars::HTTP_PROXY,
1973            EnvVars::NO_PROXY,
1974            EnvVars::SSL_CERT_DIR,
1975            EnvVars::SSL_CERT_FILE,
1976            EnvVars::UV_NATIVE_TLS,
1977            EnvVars::UV_SYSTEM_CERTS,
1978        ];
1979
1980        for env_var in EnvVars::all_names()
1981            .iter()
1982            .filter(|name| !passthrough.contains(name))
1983        {
1984            command.env_remove(env_var);
1985        }
1986
1987        command
1988    }
1989}
1990
1991/// Creates a "unified" diff between the two line-oriented strings suitable
1992/// for snapshotting.
1993pub fn diff_snapshot(old: &str, new: &str) -> String {
1994    static TRIM_TRAILING_WHITESPACE: std::sync::LazyLock<Regex> =
1995        std::sync::LazyLock::new(|| Regex::new(r"(?m)^\s+$").unwrap());
1996
1997    let diff = similar::TextDiff::from_lines(old, new);
1998    let unified = diff
1999        .unified_diff()
2000        .context_radius(10)
2001        .header("old", "new")
2002        .to_string();
2003    // Not totally clear why, but some lines end up containing only
2004    // whitespace in the diff, even though they don't appear in the
2005    // original data. So just strip them here.
2006    TRIM_TRAILING_WHITESPACE
2007        .replace_all(&unified, "")
2008        .into_owned()
2009}
2010
2011pub fn site_packages_path(venv: &Path, python: &str) -> PathBuf {
2012    if cfg!(unix) {
2013        venv.join("lib").join(python).join("site-packages")
2014    } else if cfg!(windows) {
2015        venv.join("Lib").join("site-packages")
2016    } else {
2017        unimplemented!("Only Windows and Unix are supported")
2018    }
2019}
2020
2021pub fn venv_bin_path(venv: impl AsRef<Path>) -> PathBuf {
2022    if cfg!(unix) {
2023        venv.as_ref().join("bin")
2024    } else if cfg!(windows) {
2025        venv.as_ref().join("Scripts")
2026    } else {
2027        unimplemented!("Only Windows and Unix are supported")
2028    }
2029}
2030
2031/// Get the path to the python interpreter for a specific python version.
2032pub fn get_python(version: &PythonVersion) -> PathBuf {
2033    ManagedPythonInstallations::from_settings(None)
2034        .map(|installed_pythons| {
2035            installed_pythons
2036                .find_version(version)
2037                .expect("Tests are run on a supported platform")
2038                .next()
2039                .as_ref()
2040                .map(|python| python.executable(false))
2041        })
2042        // We'll search for the request Python on the PATH if not found in the python versions
2043        // We hack this into a `PathBuf` to satisfy the compiler but it's just a string
2044        .unwrap_or_default()
2045        .unwrap_or(PathBuf::from(version.to_string()))
2046}
2047
2048/// Create a virtual environment at the given path.
2049pub fn create_venv_from_executable<P: AsRef<Path>>(
2050    path: P,
2051    cache_dir: &ChildPath,
2052    python: &Path,
2053    uv_bin: &Path,
2054) {
2055    TestContext::new_command_with(uv_bin)
2056        .arg("venv")
2057        .arg(path.as_ref().as_os_str())
2058        .arg("--clear")
2059        .arg("--cache-dir")
2060        .arg(cache_dir.path())
2061        .arg("--python")
2062        .arg(python)
2063        .current_dir(path.as_ref().parent().unwrap())
2064        .assert()
2065        .success();
2066    ChildPath::new(path.as_ref()).assert(predicate::path::is_dir());
2067}
2068
2069/// Create a `PATH` with the requested Python versions available in order.
2070///
2071/// Generally this should be used with `UV_TEST_PYTHON_PATH`.
2072pub fn python_path_with_versions(
2073    temp_dir: &ChildPath,
2074    python_versions: &[&str],
2075) -> anyhow::Result<OsString> {
2076    let download_list = ManagedPythonDownloadList::new_only_embedded().unwrap();
2077    Ok(env::join_paths(
2078        python_installations_for_versions(temp_dir, python_versions, &download_list)?
2079            .into_iter()
2080            .map(|path| path.parent().unwrap().to_path_buf()),
2081    )?)
2082}
2083
2084/// Returns a list of Python executables for the given versions.
2085///
2086/// Generally this should be used with `UV_TEST_PYTHON_PATH`.
2087pub fn python_installations_for_versions(
2088    temp_dir: &ChildPath,
2089    python_versions: &[&str],
2090    download_list: &ManagedPythonDownloadList,
2091) -> anyhow::Result<Vec<PathBuf>> {
2092    let cache = Cache::from_path(temp_dir.child("cache").to_path_buf())
2093        .init_no_wait()?
2094        .expect("No cache contention when setting up Python in tests");
2095    let selected_pythons = python_versions
2096        .iter()
2097        .map(|python_version| {
2098            if let Ok(python) = PythonInstallation::find(
2099                &PythonRequest::parse(python_version),
2100                EnvironmentPreference::OnlySystem,
2101                PythonPreference::Managed,
2102                download_list,
2103                &cache,
2104                Preview::default(),
2105            ) {
2106                python.into_interpreter().sys_executable().to_owned()
2107            } else {
2108                panic!("Could not find Python {python_version} for test\nTry `cargo run python install` first, or refer to CONTRIBUTING.md");
2109            }
2110        })
2111        .collect::<Vec<_>>();
2112
2113    assert!(
2114        python_versions.is_empty() || !selected_pythons.is_empty(),
2115        "Failed to fulfill requested test Python versions: {selected_pythons:?}"
2116    );
2117
2118    Ok(selected_pythons)
2119}
2120
2121#[derive(Debug, Copy, Clone)]
2122pub enum WindowsFilters {
2123    Platform,
2124    Universal,
2125}
2126
2127/// Helper method to apply filters to a string. Useful when `!uv_snapshot` cannot be used.
2128pub fn apply_filters<T: AsRef<str>>(mut snapshot: String, filters: impl AsRef<[(T, T)]>) -> String {
2129    for (matcher, replacement) in filters.as_ref() {
2130        // TODO(konstin): Cache regex compilation
2131        let re = Regex::new(matcher.as_ref()).expect("Do you need to regex::escape your filter?");
2132        if re.is_match(&snapshot) {
2133            snapshot = re.replace_all(&snapshot, replacement.as_ref()).to_string();
2134        }
2135    }
2136    snapshot
2137}
2138
2139/// Execute the command and format its output status, stdout and stderr into a snapshot string.
2140///
2141/// This function is derived from `insta_cmd`s `spawn_with_info`.
2142pub fn run_and_format<T: AsRef<str>>(
2143    command: impl BorrowMut<Command>,
2144    filters: impl AsRef<[(T, T)]>,
2145    function_name: &str,
2146    windows_filters: Option<WindowsFilters>,
2147    input: Option<&str>,
2148) -> (String, Output) {
2149    let (snapshot, output, _) =
2150        run_and_format_with_status(command, filters, function_name, windows_filters, input);
2151    (snapshot, output)
2152}
2153
2154/// Execute the command and format its output status, stdout and stderr into a snapshot string.
2155///
2156/// This function is derived from `insta_cmd`s `spawn_with_info`.
2157#[expect(clippy::print_stderr)]
2158pub fn run_and_format_with_status<T: AsRef<str>>(
2159    mut command: impl BorrowMut<Command>,
2160    filters: impl AsRef<[(T, T)]>,
2161    function_name: &str,
2162    windows_filters: Option<WindowsFilters>,
2163    input: Option<&str>,
2164) -> (String, Output, ExitStatus) {
2165    let program = command
2166        .borrow_mut()
2167        .get_program()
2168        .to_string_lossy()
2169        .to_string();
2170
2171    // Support profiling test run commands with traces.
2172    if let Ok(root) = env::var(EnvVars::TRACING_DURATIONS_TEST_ROOT) {
2173        // We only want to fail if the variable is set at runtime.
2174        #[allow(clippy::assertions_on_constants)]
2175        {
2176            assert!(
2177                cfg!(feature = "tracing-durations-export"),
2178                "You need to enable the tracing-durations-export feature to use `TRACING_DURATIONS_TEST_ROOT`"
2179            );
2180        }
2181        command.borrow_mut().env(
2182            EnvVars::TRACING_DURATIONS_FILE,
2183            Path::new(&root).join(function_name).with_extension("jsonl"),
2184        );
2185    }
2186
2187    let output = if let Some(input) = input {
2188        let mut child = command
2189            .borrow_mut()
2190            .stdin(Stdio::piped())
2191            .stdout(Stdio::piped())
2192            .stderr(Stdio::piped())
2193            .spawn()
2194            .unwrap_or_else(|err| panic!("Failed to spawn {program}: {err}"));
2195        child
2196            .stdin
2197            .as_mut()
2198            .expect("Failed to open stdin")
2199            .write_all(input.as_bytes())
2200            .expect("Failed to write to stdin");
2201
2202        child
2203            .wait_with_output()
2204            .unwrap_or_else(|err| panic!("Failed to read output from {program}: {err}"))
2205    } else {
2206        command
2207            .borrow_mut()
2208            .output()
2209            .unwrap_or_else(|err| panic!("Failed to spawn {program}: {err}"))
2210    };
2211
2212    eprintln!("\n━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ Unfiltered output ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━");
2213    eprintln!(
2214        "----- stdout -----\n{}\n----- stderr -----\n{}",
2215        String::from_utf8_lossy(&output.stdout),
2216        String::from_utf8_lossy(&output.stderr),
2217    );
2218    eprintln!("────────────────────────────────────────────────────────────────────────────────\n");
2219
2220    let mut snapshot = apply_filters(
2221        format!(
2222            "success: {:?}\nexit_code: {}\n----- stdout -----\n{}\n----- stderr -----\n{}",
2223            output.status.success(),
2224            output.status.code().unwrap_or(!0),
2225            String::from_utf8_lossy(&output.stdout),
2226            String::from_utf8_lossy(&output.stderr),
2227        ),
2228        filters,
2229    );
2230
2231    // This is a heuristic filter meant to try and make *most* of our tests
2232    // pass whether it's on Windows or Unix. In particular, there are some very
2233    // common Windows-only dependencies that, when removed from a resolution,
2234    // cause the set of dependencies to be the same across platforms.
2235    if cfg!(windows) {
2236        if let Some(windows_filters) = windows_filters {
2237            // The optional leading +/-/~ is for install logs, the optional next line is for lockfiles
2238            let windows_only_deps = [
2239                (r"( ?[-+~] ?)?colorama==\d+(\.\d+)+( [\\]\n\s+--hash=.*)?\n(\s+# via .*\n)?"),
2240                (r"( ?[-+~] ?)?colorama==\d+(\.\d+)+(\s+[-+~]?\s+# via .*)?\n"),
2241                (r"( ?[-+~] ?)?tzdata==\d+(\.\d+)+( [\\]\n\s+--hash=.*)?\n(\s+# via .*\n)?"),
2242                (r"( ?[-+~] ?)?tzdata==\d+(\.\d+)+(\s+[-+~]?\s+# via .*)?\n"),
2243            ];
2244            let mut removed_packages = 0;
2245            for windows_only_dep in windows_only_deps {
2246                // TODO(konstin): Cache regex compilation
2247                let re = Regex::new(windows_only_dep).unwrap();
2248                if re.is_match(&snapshot) {
2249                    snapshot = re.replace(&snapshot, "").to_string();
2250                    removed_packages += 1;
2251                }
2252            }
2253            if removed_packages > 0 {
2254                for i in 1..20 {
2255                    for verb in match windows_filters {
2256                        WindowsFilters::Platform => [
2257                            "Resolved",
2258                            "Prepared",
2259                            "Installed",
2260                            "Checked",
2261                            "Uninstalled",
2262                        ]
2263                        .iter(),
2264                        WindowsFilters::Universal => {
2265                            ["Prepared", "Installed", "Checked", "Uninstalled"].iter()
2266                        }
2267                    } {
2268                        snapshot = snapshot.replace(
2269                            &format!("{verb} {} packages", i + removed_packages),
2270                            &format!("{verb} {} package{}", i, if i > 1 { "s" } else { "" }),
2271                        );
2272                    }
2273                }
2274            }
2275        }
2276    }
2277
2278    let status = output.status;
2279    (snapshot, output, status)
2280}
2281
2282/// Recursively copy a directory and its contents, skipping gitignored files.
2283pub fn copy_dir_ignore(src: impl AsRef<Path>, dst: impl AsRef<Path>) -> anyhow::Result<()> {
2284    for entry in ignore::Walk::new(&src) {
2285        let entry = entry?;
2286        let relative = entry.path().strip_prefix(&src)?;
2287        let ty = entry.file_type().unwrap();
2288        if ty.is_dir() {
2289            fs_err::create_dir(dst.as_ref().join(relative))?;
2290        } else {
2291            fs_err::copy(entry.path(), dst.as_ref().join(relative))?;
2292        }
2293    }
2294    Ok(())
2295}
2296
2297/// Create a stub package `name` in `dir` with the given `pyproject.toml` body.
2298pub fn make_project(dir: &Path, name: &str, body: &str) -> anyhow::Result<()> {
2299    let pyproject_toml = formatdoc! {r#"
2300        [project]
2301        name = "{name}"
2302        version = "0.1.0"
2303        requires-python = ">=3.11,<3.13"
2304        {body}
2305
2306        [build-system]
2307        requires = ["uv_build>=0.9.0,<10000"]
2308        build-backend = "uv_build"
2309        "#
2310    };
2311    fs_err::create_dir_all(dir)?;
2312    fs_err::write(dir.join("pyproject.toml"), pyproject_toml)?;
2313    fs_err::create_dir_all(dir.join("src").join(name))?;
2314    fs_err::write(dir.join("src").join(name).join("__init__.py"), "")?;
2315    Ok(())
2316}
2317
2318// This is a fine-grained token that only has read-only access to the `uv-private-pypackage` repository
2319pub const READ_ONLY_GITHUB_TOKEN: &[&str] = &[
2320    "Z2l0aHViCg==",
2321    "cGF0Cg==",
2322    "MTFBQlVDUjZBMERMUTQ3aVphN3hPdV9qQmhTMkZUeHZ4ZE13OHczakxuZndsV2ZlZjc2cE53eHBWS2tiRUFwdnpmUk8zV0dDSUhicDFsT01aago=",
2323];
2324
2325// This is a fine-grained token that only has read-only access to the `uv-private-pypackage-2` repository
2326#[cfg(not(windows))]
2327pub const READ_ONLY_GITHUB_TOKEN_2: &[&str] = &[
2328    "Z2l0aHViCg==",
2329    "cGF0Cg==",
2330    "MTFBQlVDUjZBMDJTOFYwMTM4YmQ0bV9uTXpueWhxZDBrcllROTQ5SERTeTI0dENKZ2lmdzIybDFSR2s1SE04QW8xTUVYQ1I0Q1YxYUdPRGpvZQo=",
2331];
2332
2333pub const READ_ONLY_GITHUB_SSH_DEPLOY_KEY: &str = "LS0tLS1CRUdJTiBPUEVOU1NIIFBSSVZBVEUgS0VZLS0tLS0KYjNCbGJuTnphQzFyWlhrdGRqRUFBQUFBQkc1dmJtVUFBQUFFYm05dVpRQUFBQUFBQUFBQkFBQUFNd0FBQUF0emMyZ3RaVwpReU5UVXhPUUFBQUNBeTF1SnNZK1JXcWp1NkdIY3Z6a3AwS21yWDEwdmo3RUZqTkpNTkRqSGZPZ0FBQUpqWUpwVnAyQ2FWCmFRQUFBQXR6YzJndFpXUXlOVFV4T1FBQUFDQXkxdUpzWStSV3FqdTZHSGN2emtwMEttclgxMHZqN0VGak5KTU5EakhmT2cKQUFBRUMwbzBnd1BxbGl6TFBJOEFXWDVaS2dVZHJyQ2ptMDhIQm9FenB4VDg3MXBqTFc0bXhqNUZhcU83b1lkeS9PU25RcQphdGZYUytQc1FXTTBrdzBPTWQ4NkFBQUFFR3R2Ym5OMGFVQmhjM1J5WVd3dWMyZ0JBZ01FQlE9PQotLS0tLUVORCBPUEVOU1NIIFBSSVZBVEUgS0VZLS0tLS0K";
2334
2335/// Decode a split, base64 encoded authentication token.
2336/// We split and encode the token to bypass revoke by GitHub's secret scanning
2337pub fn decode_token(content: &[&str]) -> String {
2338    content
2339        .iter()
2340        .map(|part| base64.decode(part).unwrap())
2341        .map(|decoded| {
2342            std::str::from_utf8(decoded.as_slice())
2343                .unwrap()
2344                .trim_end()
2345                .to_string()
2346        })
2347        .join("_")
2348}
2349
2350/// Simulates `reqwest::blocking::get` but returns bytes directly, and disables
2351/// certificate verification, passing through the `BaseClient`
2352#[tokio::main(flavor = "current_thread")]
2353pub async fn download_to_disk(url: &str, path: &Path) {
2354    let trusted_hosts: Vec<_> = env::var(EnvVars::UV_INSECURE_HOST)
2355        .unwrap_or_default()
2356        .split(' ')
2357        .map(|h| uv_configuration::TrustedHost::from_str(h).unwrap())
2358        .collect();
2359
2360    let client = uv_client::BaseClientBuilder::default()
2361        .allow_insecure_host(trusted_hosts)
2362        .build();
2363    let url = url.parse().unwrap();
2364    let response = client
2365        .for_host(&url)
2366        .get(reqwest::Url::from(url))
2367        .send()
2368        .await
2369        .unwrap();
2370
2371    let mut file = fs_err::tokio::File::create(path).await.unwrap();
2372    let mut stream = response.bytes_stream();
2373    while let Some(chunk) = stream.next().await {
2374        file.write_all(&chunk.unwrap()).await.unwrap();
2375    }
2376    file.sync_all().await.unwrap();
2377}
2378
2379/// A guard that sets a directory to read-only and restores original permissions when dropped.
2380///
2381/// This is useful for tests that need to make a directory read-only and ensure
2382/// the permissions are restored even if the test panics.
2383#[cfg(unix)]
2384pub struct ReadOnlyDirectoryGuard {
2385    path: PathBuf,
2386    original_mode: u32,
2387}
2388
2389#[cfg(unix)]
2390impl ReadOnlyDirectoryGuard {
2391    /// Sets the directory to read-only (removes write permission) and returns a guard
2392    /// that will restore the original permissions when dropped.
2393    pub fn new(path: impl Into<PathBuf>) -> std::io::Result<Self> {
2394        use std::os::unix::fs::PermissionsExt;
2395        let path = path.into();
2396        let metadata = fs_err::metadata(&path)?;
2397        let original_mode = metadata.permissions().mode();
2398        // Remove write permissions (keep read and execute)
2399        let readonly_mode = original_mode & !0o222;
2400        fs_err::set_permissions(&path, std::fs::Permissions::from_mode(readonly_mode))?;
2401        Ok(Self {
2402            path,
2403            original_mode,
2404        })
2405    }
2406}
2407
2408#[cfg(unix)]
2409impl Drop for ReadOnlyDirectoryGuard {
2410    fn drop(&mut self) {
2411        use std::os::unix::fs::PermissionsExt;
2412        let _ = fs_err::set_permissions(
2413            &self.path,
2414            std::fs::Permissions::from_mode(self.original_mode),
2415        );
2416    }
2417}
2418
2419/// Utility macro to return the name of the current function.
2420///
2421/// https://stackoverflow.com/a/40234666/3549270
2422#[doc(hidden)]
2423#[macro_export]
2424macro_rules! function_name {
2425    () => {{
2426        fn f() {}
2427        fn type_name_of_val<T>(_: T) -> &'static str {
2428            std::any::type_name::<T>()
2429        }
2430        let mut name = type_name_of_val(f).strip_suffix("::f").unwrap_or("");
2431        while let Some(rest) = name.strip_suffix("::{{closure}}") {
2432            name = rest;
2433        }
2434        name
2435    }};
2436}
2437
2438/// Run [`assert_cmd_snapshot!`], with default filters or with custom filters.
2439///
2440/// By default, the filters will search for the generally windows-only deps colorama and tzdata,
2441/// filter them out and decrease the package counts by one for each match.
2442#[macro_export]
2443macro_rules! uv_snapshot {
2444    ($spawnable:expr, @$snapshot:literal) => {{
2445        uv_snapshot!($crate::INSTA_FILTERS.to_vec(), $spawnable, @$snapshot)
2446    }};
2447    ($filters:expr, $spawnable:expr, @$snapshot:literal) => {{
2448        // Take a reference for backwards compatibility with the vec-expecting insta filters.
2449        let (snapshot, output) = $crate::run_and_format($spawnable, &$filters, $crate::function_name!(), Some($crate::WindowsFilters::Platform), None);
2450        ::insta::assert_snapshot!(snapshot, @$snapshot);
2451        output
2452    }};
2453    ($filters:expr, $spawnable:expr, input=$input:expr, @$snapshot:literal) => {{
2454        // Take a reference for backwards compatibility with the vec-expecting insta filters.
2455        let (snapshot, output) = $crate::run_and_format($spawnable, &$filters, $crate::function_name!(), Some($crate::WindowsFilters::Platform), Some($input));
2456        ::insta::assert_snapshot!(snapshot, @$snapshot);
2457        output
2458    }};
2459    ($filters:expr, windows_filters=false, $spawnable:expr, @$snapshot:literal) => {{
2460        // Take a reference for backwards compatibility with the vec-expecting insta filters.
2461        let (snapshot, output) = $crate::run_and_format($spawnable, &$filters, $crate::function_name!(), None, None);
2462        ::insta::assert_snapshot!(snapshot, @$snapshot);
2463        output
2464    }};
2465    ($filters:expr, universal_windows_filters=true, $spawnable:expr, @$snapshot:literal) => {{
2466        // Take a reference for backwards compatibility with the vec-expecting insta filters.
2467        let (snapshot, output) = $crate::run_and_format($spawnable, &$filters, $crate::function_name!(), Some($crate::WindowsFilters::Universal), None);
2468        ::insta::assert_snapshot!(snapshot, @$snapshot);
2469        output
2470    }};
2471}