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 workspace metadata` command with options shared across scenarios.
1395    pub fn workspace_metadata(&self) -> Command {
1396        let mut command = self.new_command();
1397        command.arg("workspace").arg("metadata");
1398        self.add_shared_options(&mut command, false);
1399        command
1400    }
1401
1402    /// Create a `uv workspace dir` command with options shared across scenarios.
1403    pub fn workspace_dir(&self) -> Command {
1404        let mut command = self.new_command();
1405        command.arg("workspace").arg("dir");
1406        self.add_shared_options(&mut command, false);
1407        command
1408    }
1409
1410    /// Create a `uv workspace list` command with options shared across scenarios.
1411    pub fn workspace_list(&self) -> Command {
1412        let mut command = self.new_command();
1413        command.arg("workspace").arg("list");
1414        self.add_shared_options(&mut command, false);
1415        command
1416    }
1417
1418    /// Create a `uv export` command with options shared across scenarios.
1419    pub fn export(&self) -> Command {
1420        let mut command = self.new_command();
1421        command.arg("export");
1422        self.add_shared_options(&mut command, false);
1423        command
1424    }
1425
1426    /// Create a `uv format` command with options shared across scenarios.
1427    pub fn format(&self) -> Command {
1428        let mut command = self.new_command();
1429        command.arg("format");
1430        self.add_shared_options(&mut command, false);
1431        // Override to a more recent date for ruff version resolution
1432        command.env(EnvVars::UV_EXCLUDE_NEWER, "2026-02-15T00:00:00Z");
1433        command
1434    }
1435
1436    /// Create a `uv build` command with options shared across scenarios.
1437    pub fn build(&self) -> Command {
1438        let mut command = self.new_command();
1439        command.arg("build");
1440        self.add_shared_options(&mut command, false);
1441        command
1442    }
1443
1444    pub fn version(&self) -> Command {
1445        let mut command = self.new_command();
1446        command.arg("version");
1447        self.add_shared_options(&mut command, false);
1448        command
1449    }
1450
1451    pub fn self_version(&self) -> Command {
1452        let mut command = self.new_command();
1453        command.arg("self").arg("version");
1454        self.add_shared_options(&mut command, false);
1455        command
1456    }
1457
1458    pub fn self_update(&self) -> Command {
1459        let mut command = self.new_command();
1460        command.arg("self").arg("update");
1461        self.add_shared_options(&mut command, false);
1462        command
1463    }
1464
1465    /// Create a `uv publish` command with options shared across scenarios.
1466    pub fn publish(&self) -> Command {
1467        let mut command = self.new_command();
1468        command.arg("publish");
1469        self.add_shared_options(&mut command, false);
1470        command
1471    }
1472
1473    /// Create a `uv python find` command with options shared across scenarios.
1474    pub fn python_find(&self) -> Command {
1475        let mut command = self.new_command();
1476        command
1477            .arg("python")
1478            .arg("find")
1479            .env(EnvVars::UV_PREVIEW, "1")
1480            .env(EnvVars::UV_PYTHON_INSTALL_DIR, "");
1481        self.add_shared_options(&mut command, false);
1482        command
1483    }
1484
1485    /// Create a `uv python list` command with options shared across scenarios.
1486    pub fn python_list(&self) -> Command {
1487        let mut command = self.new_command();
1488        command
1489            .arg("python")
1490            .arg("list")
1491            .env(EnvVars::UV_PYTHON_INSTALL_DIR, "");
1492        self.add_shared_options(&mut command, false);
1493        command
1494    }
1495
1496    /// Create a `uv python install` command with options shared across scenarios.
1497    pub fn python_install(&self) -> Command {
1498        let mut command = self.new_command();
1499        command.arg("python").arg("install");
1500        self.add_shared_options(&mut command, true);
1501        command
1502    }
1503
1504    /// Create a `uv python uninstall` command with options shared across scenarios.
1505    pub fn python_uninstall(&self) -> Command {
1506        let mut command = self.new_command();
1507        command.arg("python").arg("uninstall");
1508        self.add_shared_options(&mut command, true);
1509        command
1510    }
1511
1512    /// Create a `uv python upgrade` command with options shared across scenarios.
1513    pub fn python_upgrade(&self) -> Command {
1514        let mut command = self.new_command();
1515        command.arg("python").arg("upgrade");
1516        self.add_shared_options(&mut command, true);
1517        command
1518    }
1519
1520    /// Create a `uv python pin` command with options shared across scenarios.
1521    pub fn python_pin(&self) -> Command {
1522        let mut command = self.new_command();
1523        command.arg("python").arg("pin");
1524        self.add_shared_options(&mut command, true);
1525        command
1526    }
1527
1528    /// Create a `uv python dir` command with options shared across scenarios.
1529    pub fn python_dir(&self) -> Command {
1530        let mut command = self.new_command();
1531        command.arg("python").arg("dir");
1532        self.add_shared_options(&mut command, true);
1533        command
1534    }
1535
1536    /// Create a `uv run` command with options shared across scenarios.
1537    pub fn run(&self) -> Command {
1538        let mut command = self.new_command();
1539        command.arg("run").env(EnvVars::UV_SHOW_RESOLUTION, "1");
1540        self.add_shared_options(&mut command, true);
1541        command
1542    }
1543
1544    /// Create a `uv tool run` command with options shared across scenarios.
1545    pub fn tool_run(&self) -> Command {
1546        let mut command = self.new_command();
1547        command
1548            .arg("tool")
1549            .arg("run")
1550            .env(EnvVars::UV_SHOW_RESOLUTION, "1");
1551        self.add_shared_options(&mut command, false);
1552        command
1553    }
1554
1555    /// Create a `uv upgrade run` command with options shared across scenarios.
1556    pub fn tool_upgrade(&self) -> Command {
1557        let mut command = self.new_command();
1558        command.arg("tool").arg("upgrade");
1559        self.add_shared_options(&mut command, false);
1560        command
1561    }
1562
1563    /// Create a `uv tool install` command with options shared across scenarios.
1564    pub fn tool_install(&self) -> Command {
1565        let mut command = self.new_command();
1566        command.arg("tool").arg("install");
1567        self.add_shared_options(&mut command, false);
1568        command
1569    }
1570
1571    /// Create a `uv tool list` command with options shared across scenarios.
1572    pub fn tool_list(&self) -> Command {
1573        let mut command = self.new_command();
1574        command.arg("tool").arg("list");
1575        self.add_shared_options(&mut command, false);
1576        command
1577    }
1578
1579    /// Create a `uv tool dir` command with options shared across scenarios.
1580    pub fn tool_dir(&self) -> Command {
1581        let mut command = self.new_command();
1582        command.arg("tool").arg("dir");
1583        self.add_shared_options(&mut command, false);
1584        command
1585    }
1586
1587    /// Create a `uv tool uninstall` command with options shared across scenarios.
1588    pub fn tool_uninstall(&self) -> Command {
1589        let mut command = self.new_command();
1590        command.arg("tool").arg("uninstall");
1591        self.add_shared_options(&mut command, false);
1592        command
1593    }
1594
1595    /// Create a `uv add` command for the given requirements.
1596    pub fn add(&self) -> Command {
1597        let mut command = self.new_command();
1598        command.arg("add");
1599        self.add_shared_options(&mut command, false);
1600        command
1601    }
1602
1603    /// Create a `uv remove` command for the given requirements.
1604    pub fn remove(&self) -> Command {
1605        let mut command = self.new_command();
1606        command.arg("remove");
1607        self.add_shared_options(&mut command, false);
1608        command
1609    }
1610
1611    /// Create a `uv tree` command with options shared across scenarios.
1612    pub fn tree(&self) -> Command {
1613        let mut command = self.new_command();
1614        command.arg("tree");
1615        self.add_shared_options(&mut command, false);
1616        command
1617    }
1618
1619    /// Create a `uv cache clean` command.
1620    pub fn clean(&self) -> Command {
1621        let mut command = self.new_command();
1622        command.arg("cache").arg("clean");
1623        self.add_shared_options(&mut command, false);
1624        command
1625    }
1626
1627    /// Create a `uv cache prune` command.
1628    pub fn prune(&self) -> Command {
1629        let mut command = self.new_command();
1630        command.arg("cache").arg("prune");
1631        self.add_shared_options(&mut command, false);
1632        command
1633    }
1634
1635    /// Create a `uv cache size` command.
1636    pub fn cache_size(&self) -> Command {
1637        let mut command = self.new_command();
1638        command.arg("cache").arg("size");
1639        self.add_shared_options(&mut command, false);
1640        command
1641    }
1642
1643    /// Create a `uv build_backend` command.
1644    ///
1645    /// Note that this command is hidden and only invoking it through a build frontend is supported.
1646    pub fn build_backend(&self) -> Command {
1647        let mut command = self.new_command();
1648        command.arg("build-backend");
1649        self.add_shared_options(&mut command, false);
1650        command
1651    }
1652
1653    /// The path to the Python interpreter in the venv.
1654    ///
1655    /// Don't use this for `Command::new`, use `Self::python_command` instead.
1656    pub fn interpreter(&self) -> PathBuf {
1657        let venv = &self.venv;
1658        if cfg!(unix) {
1659            venv.join("bin").join("python")
1660        } else if cfg!(windows) {
1661            venv.join("Scripts").join("python.exe")
1662        } else {
1663            unimplemented!("Only Windows and Unix are supported")
1664        }
1665    }
1666
1667    pub fn python_command(&self) -> Command {
1668        let mut interpreter = self.interpreter();
1669
1670        // If there's not a virtual environment, use the first Python interpreter in the context
1671        if !interpreter.exists() {
1672            interpreter.clone_from(
1673                &self
1674                    .python_versions
1675                    .first()
1676                    .expect("At least one Python version is required")
1677                    .1,
1678            );
1679        }
1680
1681        let mut command = Self::new_command_with(&interpreter);
1682        command
1683            // Our tests change files in <1s, so we must disable CPython bytecode caching or we'll get stale files
1684            // https://github.com/python/cpython/issues/75953
1685            .arg("-B")
1686            // Python on windows
1687            .env(EnvVars::PYTHONUTF8, "1");
1688
1689        self.add_shared_env(&mut command, false);
1690
1691        command
1692    }
1693
1694    /// Create a `uv auth login` command.
1695    pub fn auth_login(&self) -> Command {
1696        let mut command = self.new_command();
1697        command.arg("auth").arg("login");
1698        self.add_shared_options(&mut command, false);
1699        command
1700    }
1701
1702    /// Create a `uv auth logout` command.
1703    pub fn auth_logout(&self) -> Command {
1704        let mut command = self.new_command();
1705        command.arg("auth").arg("logout");
1706        self.add_shared_options(&mut command, false);
1707        command
1708    }
1709
1710    /// Create a `uv auth helper --protocol bazel get` command.
1711    pub fn auth_helper(&self) -> Command {
1712        let mut command = self.new_command();
1713        command.arg("auth").arg("helper");
1714        self.add_shared_options(&mut command, false);
1715        command
1716    }
1717
1718    /// Create a `uv auth token` command.
1719    pub fn auth_token(&self) -> Command {
1720        let mut command = self.new_command();
1721        command.arg("auth").arg("token");
1722        self.add_shared_options(&mut command, false);
1723        command
1724    }
1725
1726    /// Set `HOME` to the real home directory.
1727    ///
1728    /// We need this for testing commands which use the macOS keychain.
1729    #[must_use]
1730    pub fn with_real_home(mut self) -> Self {
1731        if let Some(home) = env::var_os(EnvVars::HOME) {
1732            self.extra_env
1733                .push((EnvVars::HOME.to_string().into(), home));
1734        }
1735        // Use the test's isolated config directory to avoid reading user
1736        // configuration files (like `.python-version`) that could interfere with tests.
1737        self.extra_env.push((
1738            EnvVars::XDG_CONFIG_HOME.into(),
1739            self.user_config_dir.as_os_str().into(),
1740        ));
1741        self
1742    }
1743
1744    /// Run the given python code and check whether it succeeds.
1745    pub fn assert_command(&self, command: &str) -> Assert {
1746        self.python_command()
1747            .arg("-c")
1748            .arg(command)
1749            .current_dir(&self.temp_dir)
1750            .assert()
1751    }
1752
1753    /// Run the given python file and check whether it succeeds.
1754    pub fn assert_file(&self, file: impl AsRef<Path>) -> Assert {
1755        self.python_command()
1756            .arg(file.as_ref())
1757            .current_dir(&self.temp_dir)
1758            .assert()
1759    }
1760
1761    /// Assert a package is installed with the given version.
1762    pub fn assert_installed(&self, package: &'static str, version: &'static str) {
1763        self.assert_command(
1764            format!("import {package} as package; print(package.__version__, end='')").as_str(),
1765        )
1766        .success()
1767        .stdout(version);
1768    }
1769
1770    /// Assert a package is not installed.
1771    pub fn assert_not_installed(&self, package: &'static str) {
1772        self.assert_command(format!("import {package}").as_str())
1773            .failure();
1774    }
1775
1776    /// Generate various escaped regex patterns for the given path.
1777    pub fn path_patterns(path: impl AsRef<Path>) -> Vec<String> {
1778        let mut patterns = Vec::new();
1779
1780        // We can only canonicalize paths that exist already
1781        if path.as_ref().exists() {
1782            patterns.push(Self::path_pattern(
1783                path.as_ref()
1784                    .canonicalize()
1785                    .expect("Failed to create canonical path"),
1786            ));
1787        }
1788
1789        // Include a non-canonicalized version
1790        patterns.push(Self::path_pattern(path));
1791
1792        patterns
1793    }
1794
1795    /// Generate an escaped regex pattern for the given path.
1796    fn path_pattern(path: impl AsRef<Path>) -> String {
1797        format!(
1798            // Trim the trailing separator for cross-platform directories filters
1799            r"{}\\?/?",
1800            regex::escape(&path.as_ref().simplified_display().to_string())
1801                // Make separators platform agnostic because on Windows we will display
1802                // paths with Unix-style separators sometimes
1803                .replace(r"\\", r"(\\|\/)")
1804        )
1805    }
1806
1807    pub fn python_path(&self) -> OsString {
1808        if cfg!(unix) {
1809            // On Unix, we needed to normalize the Python executable names to `python3` for the tests
1810            env::join_paths(
1811                self.python_versions
1812                    .iter()
1813                    .map(|(version, _)| self.python_dir.join(version.to_string())),
1814            )
1815            .unwrap()
1816        } else {
1817            // On Windows, just join the parent directories of the executables
1818            env::join_paths(
1819                self.python_versions
1820                    .iter()
1821                    .map(|(_, executable)| executable.parent().unwrap().to_path_buf()),
1822            )
1823            .unwrap()
1824        }
1825    }
1826
1827    /// Standard snapshot filters _plus_ those for this test context.
1828    pub fn filters(&self) -> Vec<(&str, &str)> {
1829        // Put test context snapshots before the default filters
1830        // This ensures we don't replace other patterns inside paths from the test context first
1831        self.filters
1832            .iter()
1833            .map(|(p, r)| (p.as_str(), r.as_str()))
1834            .chain(INSTA_FILTERS.iter().copied())
1835            .collect()
1836    }
1837
1838    /// Only the filters added to this test context.
1839    pub fn filters_without_standard_filters(&self) -> Vec<(&str, &str)> {
1840        self.filters
1841            .iter()
1842            .map(|(p, r)| (p.as_str(), r.as_str()))
1843            .collect()
1844    }
1845
1846    /// For when we add pypy to the test suite.
1847    #[allow(clippy::unused_self)]
1848    pub fn python_kind(&self) -> &'static str {
1849        "python"
1850    }
1851
1852    /// Returns the site-packages folder inside the venv.
1853    pub fn site_packages(&self) -> PathBuf {
1854        site_packages_path(
1855            &self.venv,
1856            &format!(
1857                "{}{}",
1858                self.python_kind(),
1859                self.python_version.as_ref().expect(
1860                    "A Python version must be provided to retrieve the test site packages path"
1861                )
1862            ),
1863        )
1864    }
1865
1866    /// Reset the virtual environment in the test context.
1867    pub fn reset_venv(&self) {
1868        self.create_venv();
1869    }
1870
1871    /// Create a new virtual environment named `.venv` in the test context.
1872    fn create_venv(&self) {
1873        let executable = get_python(
1874            self.python_version
1875                .as_ref()
1876                .expect("A Python version must be provided to create a test virtual environment"),
1877        );
1878        create_venv_from_executable(&self.venv, &self.cache_dir, &executable, &self.uv_bin);
1879    }
1880
1881    /// Copies the files from the ecosystem project given into this text
1882    /// context.
1883    ///
1884    /// This will almost always write at least a `pyproject.toml` into this
1885    /// test context.
1886    ///
1887    /// The given name should correspond to the name of a sub-directory (not a
1888    /// path to it) in the `test/ecosystem` directory.
1889    ///
1890    /// This panics (fails the current test) for any failure.
1891    pub fn copy_ecosystem_project(&self, name: &str) {
1892        let project_dir = PathBuf::from(format!("../../test/ecosystem/{name}"));
1893        self.temp_dir.copy_from(project_dir, &["*"]).unwrap();
1894        // If there is a (gitignore) lockfile, remove it.
1895        if let Err(err) = fs_err::remove_file(self.temp_dir.join("uv.lock")) {
1896            assert_eq!(
1897                err.kind(),
1898                io::ErrorKind::NotFound,
1899                "Failed to remove uv.lock: {err}"
1900            );
1901        }
1902    }
1903
1904    /// Creates a way to compare the changes made to a lock file.
1905    ///
1906    /// This routine starts by copying (not moves) the generated lock file to
1907    /// memory. It then calls the given closure with this test context to get a
1908    /// `Command` and runs the command. The diff between the old lock file and
1909    /// the new one is then returned.
1910    ///
1911    /// This assumes that a lock has already been performed.
1912    pub fn diff_lock(&self, change: impl Fn(&Self) -> Command) -> String {
1913        static TRIM_TRAILING_WHITESPACE: std::sync::LazyLock<Regex> =
1914            std::sync::LazyLock::new(|| Regex::new(r"(?m)^\s+$").unwrap());
1915
1916        let lock_path = ChildPath::new(self.temp_dir.join("uv.lock"));
1917        let old_lock = fs_err::read_to_string(&lock_path).unwrap();
1918        let (snapshot, _, status) = run_and_format_with_status(
1919            change(self),
1920            self.filters(),
1921            "diff_lock",
1922            Some(WindowsFilters::Platform),
1923            None,
1924        );
1925        assert!(status.success(), "{snapshot}");
1926        let new_lock = fs_err::read_to_string(&lock_path).unwrap();
1927        diff_snapshot(&old_lock, &new_lock)
1928    }
1929
1930    /// Read a file in the temporary directory
1931    pub fn read(&self, file: impl AsRef<Path>) -> String {
1932        fs_err::read_to_string(self.temp_dir.join(&file))
1933            .unwrap_or_else(|_| panic!("Missing file: `{}`", file.user_display()))
1934    }
1935
1936    /// Creates a new `Command` that is intended to be suitable for use in
1937    /// all tests.
1938    fn new_command(&self) -> Command {
1939        Self::new_command_with(&self.uv_bin)
1940    }
1941
1942    /// Creates a new `Command` that is intended to be suitable for use in
1943    /// all tests, but with the given binary.
1944    ///
1945    /// Clears environment variables defined in [`EnvVars`] to avoid reading
1946    /// test host settings.
1947    fn new_command_with(bin: &Path) -> Command {
1948        let mut command = Command::new(bin);
1949
1950        let passthrough = [
1951            // For linux distributions
1952            EnvVars::PATH,
1953            // For debugging tests.
1954            EnvVars::RUST_LOG,
1955            EnvVars::RUST_BACKTRACE,
1956            // Windows System configuration.
1957            EnvVars::SYSTEMDRIVE,
1958            // Work around small default stack sizes and large futures in debug builds.
1959            EnvVars::RUST_MIN_STACK,
1960            EnvVars::UV_STACK_SIZE,
1961            // Allow running tests with custom network settings.
1962            EnvVars::ALL_PROXY,
1963            EnvVars::HTTPS_PROXY,
1964            EnvVars::HTTP_PROXY,
1965            EnvVars::NO_PROXY,
1966            EnvVars::SSL_CERT_DIR,
1967            EnvVars::SSL_CERT_FILE,
1968            EnvVars::UV_NATIVE_TLS,
1969            EnvVars::UV_SYSTEM_CERTS,
1970        ];
1971
1972        for env_var in EnvVars::all_names()
1973            .iter()
1974            .filter(|name| !passthrough.contains(name))
1975        {
1976            command.env_remove(env_var);
1977        }
1978
1979        command
1980    }
1981}
1982
1983/// Creates a "unified" diff between the two line-oriented strings suitable
1984/// for snapshotting.
1985pub fn diff_snapshot(old: &str, new: &str) -> String {
1986    static TRIM_TRAILING_WHITESPACE: std::sync::LazyLock<Regex> =
1987        std::sync::LazyLock::new(|| Regex::new(r"(?m)^\s+$").unwrap());
1988
1989    let diff = similar::TextDiff::from_lines(old, new);
1990    let unified = diff
1991        .unified_diff()
1992        .context_radius(10)
1993        .header("old", "new")
1994        .to_string();
1995    // Not totally clear why, but some lines end up containing only
1996    // whitespace in the diff, even though they don't appear in the
1997    // original data. So just strip them here.
1998    TRIM_TRAILING_WHITESPACE
1999        .replace_all(&unified, "")
2000        .into_owned()
2001}
2002
2003pub fn site_packages_path(venv: &Path, python: &str) -> PathBuf {
2004    if cfg!(unix) {
2005        venv.join("lib").join(python).join("site-packages")
2006    } else if cfg!(windows) {
2007        venv.join("Lib").join("site-packages")
2008    } else {
2009        unimplemented!("Only Windows and Unix are supported")
2010    }
2011}
2012
2013pub fn venv_bin_path(venv: impl AsRef<Path>) -> PathBuf {
2014    if cfg!(unix) {
2015        venv.as_ref().join("bin")
2016    } else if cfg!(windows) {
2017        venv.as_ref().join("Scripts")
2018    } else {
2019        unimplemented!("Only Windows and Unix are supported")
2020    }
2021}
2022
2023/// Get the path to the python interpreter for a specific python version.
2024pub fn get_python(version: &PythonVersion) -> PathBuf {
2025    ManagedPythonInstallations::from_settings(None)
2026        .map(|installed_pythons| {
2027            installed_pythons
2028                .find_version(version)
2029                .expect("Tests are run on a supported platform")
2030                .next()
2031                .as_ref()
2032                .map(|python| python.executable(false))
2033        })
2034        // We'll search for the request Python on the PATH if not found in the python versions
2035        // We hack this into a `PathBuf` to satisfy the compiler but it's just a string
2036        .unwrap_or_default()
2037        .unwrap_or(PathBuf::from(version.to_string()))
2038}
2039
2040/// Create a virtual environment at the given path.
2041pub fn create_venv_from_executable<P: AsRef<Path>>(
2042    path: P,
2043    cache_dir: &ChildPath,
2044    python: &Path,
2045    uv_bin: &Path,
2046) {
2047    TestContext::new_command_with(uv_bin)
2048        .arg("venv")
2049        .arg(path.as_ref().as_os_str())
2050        .arg("--clear")
2051        .arg("--cache-dir")
2052        .arg(cache_dir.path())
2053        .arg("--python")
2054        .arg(python)
2055        .current_dir(path.as_ref().parent().unwrap())
2056        .assert()
2057        .success();
2058    ChildPath::new(path.as_ref()).assert(predicate::path::is_dir());
2059}
2060
2061/// Create a `PATH` with the requested Python versions available in order.
2062///
2063/// Generally this should be used with `UV_TEST_PYTHON_PATH`.
2064pub fn python_path_with_versions(
2065    temp_dir: &ChildPath,
2066    python_versions: &[&str],
2067) -> anyhow::Result<OsString> {
2068    let download_list = ManagedPythonDownloadList::new_only_embedded().unwrap();
2069    Ok(env::join_paths(
2070        python_installations_for_versions(temp_dir, python_versions, &download_list)?
2071            .into_iter()
2072            .map(|path| path.parent().unwrap().to_path_buf()),
2073    )?)
2074}
2075
2076/// Returns a list of Python executables for the given versions.
2077///
2078/// Generally this should be used with `UV_TEST_PYTHON_PATH`.
2079pub fn python_installations_for_versions(
2080    temp_dir: &ChildPath,
2081    python_versions: &[&str],
2082    download_list: &ManagedPythonDownloadList,
2083) -> anyhow::Result<Vec<PathBuf>> {
2084    let cache = Cache::from_path(temp_dir.child("cache").to_path_buf())
2085        .init_no_wait()?
2086        .expect("No cache contention when setting up Python in tests");
2087    let selected_pythons = python_versions
2088        .iter()
2089        .map(|python_version| {
2090            if let Ok(python) = PythonInstallation::find(
2091                &PythonRequest::parse(python_version),
2092                EnvironmentPreference::OnlySystem,
2093                PythonPreference::Managed,
2094                download_list,
2095                &cache,
2096                Preview::default(),
2097            ) {
2098                python.into_interpreter().sys_executable().to_owned()
2099            } else {
2100                panic!("Could not find Python {python_version} for test\nTry `cargo run python install` first, or refer to CONTRIBUTING.md");
2101            }
2102        })
2103        .collect::<Vec<_>>();
2104
2105    assert!(
2106        python_versions.is_empty() || !selected_pythons.is_empty(),
2107        "Failed to fulfill requested test Python versions: {selected_pythons:?}"
2108    );
2109
2110    Ok(selected_pythons)
2111}
2112
2113#[derive(Debug, Copy, Clone)]
2114pub enum WindowsFilters {
2115    Platform,
2116    Universal,
2117}
2118
2119/// Helper method to apply filters to a string. Useful when `!uv_snapshot` cannot be used.
2120pub fn apply_filters<T: AsRef<str>>(mut snapshot: String, filters: impl AsRef<[(T, T)]>) -> String {
2121    for (matcher, replacement) in filters.as_ref() {
2122        // TODO(konstin): Cache regex compilation
2123        let re = Regex::new(matcher.as_ref()).expect("Do you need to regex::escape your filter?");
2124        if re.is_match(&snapshot) {
2125            snapshot = re.replace_all(&snapshot, replacement.as_ref()).to_string();
2126        }
2127    }
2128    snapshot
2129}
2130
2131/// Execute the command and format its output status, stdout and stderr into a snapshot string.
2132///
2133/// This function is derived from `insta_cmd`s `spawn_with_info`.
2134pub fn run_and_format<T: AsRef<str>>(
2135    command: impl BorrowMut<Command>,
2136    filters: impl AsRef<[(T, T)]>,
2137    function_name: &str,
2138    windows_filters: Option<WindowsFilters>,
2139    input: Option<&str>,
2140) -> (String, Output) {
2141    let (snapshot, output, _) =
2142        run_and_format_with_status(command, filters, function_name, windows_filters, input);
2143    (snapshot, output)
2144}
2145
2146/// Execute the command and format its output status, stdout and stderr into a snapshot string.
2147///
2148/// This function is derived from `insta_cmd`s `spawn_with_info`.
2149#[expect(clippy::print_stderr)]
2150pub fn run_and_format_with_status<T: AsRef<str>>(
2151    mut command: impl BorrowMut<Command>,
2152    filters: impl AsRef<[(T, T)]>,
2153    function_name: &str,
2154    windows_filters: Option<WindowsFilters>,
2155    input: Option<&str>,
2156) -> (String, Output, ExitStatus) {
2157    let program = command
2158        .borrow_mut()
2159        .get_program()
2160        .to_string_lossy()
2161        .to_string();
2162
2163    // Support profiling test run commands with traces.
2164    if let Ok(root) = env::var(EnvVars::TRACING_DURATIONS_TEST_ROOT) {
2165        // We only want to fail if the variable is set at runtime.
2166        #[allow(clippy::assertions_on_constants)]
2167        {
2168            assert!(
2169                cfg!(feature = "tracing-durations-export"),
2170                "You need to enable the tracing-durations-export feature to use `TRACING_DURATIONS_TEST_ROOT`"
2171            );
2172        }
2173        command.borrow_mut().env(
2174            EnvVars::TRACING_DURATIONS_FILE,
2175            Path::new(&root).join(function_name).with_extension("jsonl"),
2176        );
2177    }
2178
2179    let output = if let Some(input) = input {
2180        let mut child = command
2181            .borrow_mut()
2182            .stdin(Stdio::piped())
2183            .stdout(Stdio::piped())
2184            .stderr(Stdio::piped())
2185            .spawn()
2186            .unwrap_or_else(|err| panic!("Failed to spawn {program}: {err}"));
2187        child
2188            .stdin
2189            .as_mut()
2190            .expect("Failed to open stdin")
2191            .write_all(input.as_bytes())
2192            .expect("Failed to write to stdin");
2193
2194        child
2195            .wait_with_output()
2196            .unwrap_or_else(|err| panic!("Failed to read output from {program}: {err}"))
2197    } else {
2198        command
2199            .borrow_mut()
2200            .output()
2201            .unwrap_or_else(|err| panic!("Failed to spawn {program}: {err}"))
2202    };
2203
2204    eprintln!("\n━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ Unfiltered output ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━");
2205    eprintln!(
2206        "----- stdout -----\n{}\n----- stderr -----\n{}",
2207        String::from_utf8_lossy(&output.stdout),
2208        String::from_utf8_lossy(&output.stderr),
2209    );
2210    eprintln!("────────────────────────────────────────────────────────────────────────────────\n");
2211
2212    let mut snapshot = apply_filters(
2213        format!(
2214            "success: {:?}\nexit_code: {}\n----- stdout -----\n{}\n----- stderr -----\n{}",
2215            output.status.success(),
2216            output.status.code().unwrap_or(!0),
2217            String::from_utf8_lossy(&output.stdout),
2218            String::from_utf8_lossy(&output.stderr),
2219        ),
2220        filters,
2221    );
2222
2223    // This is a heuristic filter meant to try and make *most* of our tests
2224    // pass whether it's on Windows or Unix. In particular, there are some very
2225    // common Windows-only dependencies that, when removed from a resolution,
2226    // cause the set of dependencies to be the same across platforms.
2227    if cfg!(windows) {
2228        if let Some(windows_filters) = windows_filters {
2229            // The optional leading +/-/~ is for install logs, the optional next line is for lockfiles
2230            let windows_only_deps = [
2231                (r"( ?[-+~] ?)?colorama==\d+(\.\d+)+( [\\]\n\s+--hash=.*)?\n(\s+# via .*\n)?"),
2232                (r"( ?[-+~] ?)?colorama==\d+(\.\d+)+(\s+[-+~]?\s+# via .*)?\n"),
2233                (r"( ?[-+~] ?)?tzdata==\d+(\.\d+)+( [\\]\n\s+--hash=.*)?\n(\s+# via .*\n)?"),
2234                (r"( ?[-+~] ?)?tzdata==\d+(\.\d+)+(\s+[-+~]?\s+# via .*)?\n"),
2235            ];
2236            let mut removed_packages = 0;
2237            for windows_only_dep in windows_only_deps {
2238                // TODO(konstin): Cache regex compilation
2239                let re = Regex::new(windows_only_dep).unwrap();
2240                if re.is_match(&snapshot) {
2241                    snapshot = re.replace(&snapshot, "").to_string();
2242                    removed_packages += 1;
2243                }
2244            }
2245            if removed_packages > 0 {
2246                for i in 1..20 {
2247                    for verb in match windows_filters {
2248                        WindowsFilters::Platform => [
2249                            "Resolved",
2250                            "Prepared",
2251                            "Installed",
2252                            "Checked",
2253                            "Uninstalled",
2254                        ]
2255                        .iter(),
2256                        WindowsFilters::Universal => {
2257                            ["Prepared", "Installed", "Checked", "Uninstalled"].iter()
2258                        }
2259                    } {
2260                        snapshot = snapshot.replace(
2261                            &format!("{verb} {} packages", i + removed_packages),
2262                            &format!("{verb} {} package{}", i, if i > 1 { "s" } else { "" }),
2263                        );
2264                    }
2265                }
2266            }
2267        }
2268    }
2269
2270    let status = output.status;
2271    (snapshot, output, status)
2272}
2273
2274/// Recursively copy a directory and its contents, skipping gitignored files.
2275pub fn copy_dir_ignore(src: impl AsRef<Path>, dst: impl AsRef<Path>) -> anyhow::Result<()> {
2276    for entry in ignore::Walk::new(&src) {
2277        let entry = entry?;
2278        let relative = entry.path().strip_prefix(&src)?;
2279        let ty = entry.file_type().unwrap();
2280        if ty.is_dir() {
2281            fs_err::create_dir(dst.as_ref().join(relative))?;
2282        } else {
2283            fs_err::copy(entry.path(), dst.as_ref().join(relative))?;
2284        }
2285    }
2286    Ok(())
2287}
2288
2289/// Create a stub package `name` in `dir` with the given `pyproject.toml` body.
2290pub fn make_project(dir: &Path, name: &str, body: &str) -> anyhow::Result<()> {
2291    let pyproject_toml = formatdoc! {r#"
2292        [project]
2293        name = "{name}"
2294        version = "0.1.0"
2295        requires-python = ">=3.11,<3.13"
2296        {body}
2297
2298        [build-system]
2299        requires = ["uv_build>=0.9.0,<10000"]
2300        build-backend = "uv_build"
2301        "#
2302    };
2303    fs_err::create_dir_all(dir)?;
2304    fs_err::write(dir.join("pyproject.toml"), pyproject_toml)?;
2305    fs_err::create_dir_all(dir.join("src").join(name))?;
2306    fs_err::write(dir.join("src").join(name).join("__init__.py"), "")?;
2307    Ok(())
2308}
2309
2310// This is a fine-grained token that only has read-only access to the `uv-private-pypackage` repository
2311pub const READ_ONLY_GITHUB_TOKEN: &[&str] = &[
2312    "Z2l0aHViCg==",
2313    "cGF0Cg==",
2314    "MTFBQlVDUjZBMERMUTQ3aVphN3hPdV9qQmhTMkZUeHZ4ZE13OHczakxuZndsV2ZlZjc2cE53eHBWS2tiRUFwdnpmUk8zV0dDSUhicDFsT01aago=",
2315];
2316
2317// This is a fine-grained token that only has read-only access to the `uv-private-pypackage-2` repository
2318#[cfg(not(windows))]
2319pub const READ_ONLY_GITHUB_TOKEN_2: &[&str] = &[
2320    "Z2l0aHViCg==",
2321    "cGF0Cg==",
2322    "MTFBQlVDUjZBMDJTOFYwMTM4YmQ0bV9uTXpueWhxZDBrcllROTQ5SERTeTI0dENKZ2lmdzIybDFSR2s1SE04QW8xTUVYQ1I0Q1YxYUdPRGpvZQo=",
2323];
2324
2325pub const READ_ONLY_GITHUB_SSH_DEPLOY_KEY: &str = "LS0tLS1CRUdJTiBPUEVOU1NIIFBSSVZBVEUgS0VZLS0tLS0KYjNCbGJuTnphQzFyWlhrdGRqRUFBQUFBQkc1dmJtVUFBQUFFYm05dVpRQUFBQUFBQUFBQkFBQUFNd0FBQUF0emMyZ3RaVwpReU5UVXhPUUFBQUNBeTF1SnNZK1JXcWp1NkdIY3Z6a3AwS21yWDEwdmo3RUZqTkpNTkRqSGZPZ0FBQUpqWUpwVnAyQ2FWCmFRQUFBQXR6YzJndFpXUXlOVFV4T1FBQUFDQXkxdUpzWStSV3FqdTZHSGN2emtwMEttclgxMHZqN0VGak5KTU5EakhmT2cKQUFBRUMwbzBnd1BxbGl6TFBJOEFXWDVaS2dVZHJyQ2ptMDhIQm9FenB4VDg3MXBqTFc0bXhqNUZhcU83b1lkeS9PU25RcQphdGZYUytQc1FXTTBrdzBPTWQ4NkFBQUFFR3R2Ym5OMGFVQmhjM1J5WVd3dWMyZ0JBZ01FQlE9PQotLS0tLUVORCBPUEVOU1NIIFBSSVZBVEUgS0VZLS0tLS0K";
2326
2327/// Decode a split, base64 encoded authentication token.
2328/// We split and encode the token to bypass revoke by GitHub's secret scanning
2329pub fn decode_token(content: &[&str]) -> String {
2330    content
2331        .iter()
2332        .map(|part| base64.decode(part).unwrap())
2333        .map(|decoded| {
2334            std::str::from_utf8(decoded.as_slice())
2335                .unwrap()
2336                .trim_end()
2337                .to_string()
2338        })
2339        .join("_")
2340}
2341
2342/// Simulates `reqwest::blocking::get` but returns bytes directly, and disables
2343/// certificate verification, passing through the `BaseClient`
2344#[tokio::main(flavor = "current_thread")]
2345pub async fn download_to_disk(url: &str, path: &Path) {
2346    let trusted_hosts: Vec<_> = env::var(EnvVars::UV_INSECURE_HOST)
2347        .unwrap_or_default()
2348        .split(' ')
2349        .map(|h| uv_configuration::TrustedHost::from_str(h).unwrap())
2350        .collect();
2351
2352    let client = uv_client::BaseClientBuilder::default()
2353        .allow_insecure_host(trusted_hosts)
2354        .build();
2355    let url = url.parse().unwrap();
2356    let response = client
2357        .for_host(&url)
2358        .get(reqwest::Url::from(url))
2359        .send()
2360        .await
2361        .unwrap();
2362
2363    let mut file = fs_err::tokio::File::create(path).await.unwrap();
2364    let mut stream = response.bytes_stream();
2365    while let Some(chunk) = stream.next().await {
2366        file.write_all(&chunk.unwrap()).await.unwrap();
2367    }
2368    file.sync_all().await.unwrap();
2369}
2370
2371/// A guard that sets a directory to read-only and restores original permissions when dropped.
2372///
2373/// This is useful for tests that need to make a directory read-only and ensure
2374/// the permissions are restored even if the test panics.
2375#[cfg(unix)]
2376pub struct ReadOnlyDirectoryGuard {
2377    path: PathBuf,
2378    original_mode: u32,
2379}
2380
2381#[cfg(unix)]
2382impl ReadOnlyDirectoryGuard {
2383    /// Sets the directory to read-only (removes write permission) and returns a guard
2384    /// that will restore the original permissions when dropped.
2385    pub fn new(path: impl Into<PathBuf>) -> std::io::Result<Self> {
2386        use std::os::unix::fs::PermissionsExt;
2387        let path = path.into();
2388        let metadata = fs_err::metadata(&path)?;
2389        let original_mode = metadata.permissions().mode();
2390        // Remove write permissions (keep read and execute)
2391        let readonly_mode = original_mode & !0o222;
2392        fs_err::set_permissions(&path, std::fs::Permissions::from_mode(readonly_mode))?;
2393        Ok(Self {
2394            path,
2395            original_mode,
2396        })
2397    }
2398}
2399
2400#[cfg(unix)]
2401impl Drop for ReadOnlyDirectoryGuard {
2402    fn drop(&mut self) {
2403        use std::os::unix::fs::PermissionsExt;
2404        let _ = fs_err::set_permissions(
2405            &self.path,
2406            std::fs::Permissions::from_mode(self.original_mode),
2407        );
2408    }
2409}
2410
2411/// Utility macro to return the name of the current function.
2412///
2413/// https://stackoverflow.com/a/40234666/3549270
2414#[doc(hidden)]
2415#[macro_export]
2416macro_rules! function_name {
2417    () => {{
2418        fn f() {}
2419        fn type_name_of_val<T>(_: T) -> &'static str {
2420            std::any::type_name::<T>()
2421        }
2422        let mut name = type_name_of_val(f).strip_suffix("::f").unwrap_or("");
2423        while let Some(rest) = name.strip_suffix("::{{closure}}") {
2424            name = rest;
2425        }
2426        name
2427    }};
2428}
2429
2430/// Run [`assert_cmd_snapshot!`], with default filters or with custom filters.
2431///
2432/// By default, the filters will search for the generally windows-only deps colorama and tzdata,
2433/// filter them out and decrease the package counts by one for each match.
2434#[macro_export]
2435macro_rules! uv_snapshot {
2436    ($spawnable:expr, @$snapshot:literal) => {{
2437        uv_snapshot!($crate::INSTA_FILTERS.to_vec(), $spawnable, @$snapshot)
2438    }};
2439    ($filters:expr, $spawnable:expr, @$snapshot:literal) => {{
2440        // Take a reference for backwards compatibility with the vec-expecting insta filters.
2441        let (snapshot, output) = $crate::run_and_format($spawnable, &$filters, $crate::function_name!(), Some($crate::WindowsFilters::Platform), None);
2442        ::insta::assert_snapshot!(snapshot, @$snapshot);
2443        output
2444    }};
2445    ($filters:expr, $spawnable:expr, input=$input:expr, @$snapshot:literal) => {{
2446        // Take a reference for backwards compatibility with the vec-expecting insta filters.
2447        let (snapshot, output) = $crate::run_and_format($spawnable, &$filters, $crate::function_name!(), Some($crate::WindowsFilters::Platform), Some($input));
2448        ::insta::assert_snapshot!(snapshot, @$snapshot);
2449        output
2450    }};
2451    ($filters:expr, windows_filters=false, $spawnable:expr, @$snapshot:literal) => {{
2452        // Take a reference for backwards compatibility with the vec-expecting insta filters.
2453        let (snapshot, output) = $crate::run_and_format($spawnable, &$filters, $crate::function_name!(), None, None);
2454        ::insta::assert_snapshot!(snapshot, @$snapshot);
2455        output
2456    }};
2457    ($filters:expr, universal_windows_filters=true, $spawnable:expr, @$snapshot:literal) => {{
2458        // Take a reference for backwards compatibility with the vec-expecting insta filters.
2459        let (snapshot, output) = $crate::run_and_format($spawnable, &$filters, $crate::function_name!(), Some($crate::WindowsFilters::Universal), None);
2460        ::insta::assert_snapshot!(snapshot, @$snapshot);
2461        output
2462    }};
2463}