Skip to main content

uv_settings/
lib.rs

1use std::num::NonZeroUsize;
2use std::ops::Deref;
3use std::path::{Path, PathBuf};
4use std::str::FromStr;
5use std::time::Duration;
6use tracing::info_span;
7use uv_client::{DEFAULT_CONNECT_TIMEOUT, DEFAULT_READ_TIMEOUT, DEFAULT_READ_TIMEOUT_UPLOAD};
8use uv_configuration::RequiredVersion;
9use uv_dirs::{system_config_file, user_config_dir};
10use uv_distribution_types::Origin;
11use uv_flags::EnvironmentFlags;
12use uv_fs::Simplified;
13use uv_normalize::{GroupName, PackageName};
14use uv_pep440::Version;
15use uv_redacted::DisplaySafeUrl;
16use uv_static::{EnvVars, InvalidEnvironmentVariable, parse_boolish_environment_variable};
17use uv_warnings::warn_user;
18
19pub use crate::combine::*;
20pub use crate::settings::*;
21
22mod combine;
23mod settings;
24
25/// The [`Options`] as loaded from a configuration file on disk.
26#[derive(Debug, Clone)]
27pub struct FilesystemOptions(Options);
28
29impl FilesystemOptions {
30    /// Convert the [`FilesystemOptions`] into [`Options`].
31    pub fn into_options(self) -> Options {
32        self.0
33    }
34
35    /// Set the [`Origin`] on all indexes without an existing origin.
36    #[must_use]
37    pub fn with_origin(self, origin: Origin) -> Self {
38        Self(self.0.with_origin(origin))
39    }
40}
41
42impl Deref for FilesystemOptions {
43    type Target = Options;
44
45    fn deref(&self) -> &Self::Target {
46        &self.0
47    }
48}
49
50impl FilesystemOptions {
51    /// Load the user [`FilesystemOptions`].
52    pub fn user() -> Result<Option<Self>, Error> {
53        let Some(dir) = user_config_dir() else {
54            return Ok(None);
55        };
56        let root = dir.join("uv");
57        let file = root.join("uv.toml");
58
59        tracing::debug!("Searching for user configuration in: `{}`", file.display());
60        match read_file(&file) {
61            Ok(options) => {
62                tracing::debug!("Found user configuration in: `{}`", file.display());
63                validate_uv_toml(&file, &options)?;
64                Ok(Some(Self(options.with_origin(Origin::User))))
65            }
66            Err(Error::Io(err))
67                if matches!(
68                    err.kind(),
69                    std::io::ErrorKind::NotFound
70                        | std::io::ErrorKind::NotADirectory
71                        | std::io::ErrorKind::PermissionDenied
72                ) =>
73            {
74                Ok(None)
75            }
76            Err(err) => Err(err),
77        }
78    }
79
80    pub fn system() -> Result<Option<Self>, Error> {
81        if parse_boolish_environment_variable(EnvVars::UV_NO_SYSTEM_CONFIG)? == Some(true) {
82            return Ok(None);
83        }
84
85        let Some(file) = system_config_file() else {
86            return Ok(None);
87        };
88
89        tracing::debug!("Found system configuration in: `{}`", file.display());
90        let options = read_file(&file)?;
91        validate_uv_toml(&file, &options)?;
92        Ok(Some(Self(options.with_origin(Origin::System))))
93    }
94
95    /// Find the [`FilesystemOptions`] for the given path.
96    ///
97    /// The search starts at the given path and goes up the directory tree until a `uv.toml` file or
98    /// `pyproject.toml` file is found.
99    pub fn find(path: &Path) -> Result<Option<Self>, Error> {
100        for ancestor in path.ancestors() {
101            match Self::from_directory(ancestor) {
102                Ok(Some(options)) => {
103                    return Ok(Some(options));
104                }
105                Ok(None) => {
106                    // Continue traversing the directory tree.
107                }
108                Err(Error::PyprojectToml(path, err)) => {
109                    // If we see an invalid `pyproject.toml`, warn but continue.
110                    warn_user!(
111                        "Failed to parse `{}` during settings discovery:\n{}",
112                        path.user_display().cyan(),
113                        textwrap::indent(&err.to_string(), "  ")
114                    );
115                }
116                Err(err) => {
117                    // Otherwise, warn and stop.
118                    return Err(err);
119                }
120            }
121        }
122        Ok(None)
123    }
124
125    /// Load a [`FilesystemOptions`] from a directory, preferring a `uv.toml` file over a
126    /// `pyproject.toml` file.
127    pub fn from_directory(dir: &Path) -> Result<Option<Self>, Error> {
128        // Read a `uv.toml` file in the current directory.
129        let path = dir.join("uv.toml");
130        match fs_err::read_to_string(&path) {
131            Ok(content) => {
132                let options =
133                    info_span!("toml::from_str filesystem options uv.toml", path = %path.display())
134                        .in_scope(|| toml::from_str::<Options>(&content))
135                        .map_err(|err| {
136                            check_uv_toml_required_version(
137                                &path,
138                                &content,
139                                Error::UvToml(path.clone(), Box::new(err)),
140                            )
141                        })?
142                        .relative_to(&std::path::absolute(dir)?)?;
143
144                // If the directory also contains a `[tool.uv]` table in a `pyproject.toml` file,
145                // warn.
146                let pyproject = dir.join("pyproject.toml");
147                if let Ok(content) = fs_err::read_to_string(&pyproject) {
148                    let result = info_span!("toml::from_str filesystem options pyproject.toml", path = %pyproject.display())
149                        .in_scope(|| toml::from_str::<PyProjectToml>(&content)).ok();
150                    if let Some(options) =
151                        result.and_then(|pyproject| pyproject.tool.and_then(|tool| tool.uv))
152                    {
153                        warn_uv_toml_masked_fields(&options);
154                    }
155                }
156
157                tracing::debug!("Found workspace configuration at `{}`", path.display());
158                validate_uv_toml(&path, &options)?;
159                return Ok(Some(Self(options.with_origin(Origin::Project))));
160            }
161            Err(err) if err.kind() == std::io::ErrorKind::NotFound => {}
162            Err(err) => return Err(err.into()),
163        }
164
165        // Read a `pyproject.toml` file in the current directory.
166        let path = dir.join("pyproject.toml");
167        match fs_err::read_to_string(&path) {
168            Ok(content) => {
169                // Parse, but skip any `pyproject.toml` that doesn't have a `[tool.uv]` section.
170                let pyproject =
171                    info_span!("toml::from_str filesystem options pyproject.toml", path = %path.display())
172                        .in_scope(|| toml::from_str::<PyProjectToml>(&content))
173                        .map_err(|err| {
174                            check_pyproject_required_version(&path, &content, err)
175                        })?;
176                let Some(tool) = pyproject.tool else {
177                    tracing::debug!(
178                        "Skipping `pyproject.toml` in `{}` (no `[tool]` section)",
179                        dir.display()
180                    );
181                    return Ok(None);
182                };
183                let Some(options) = tool.uv else {
184                    tracing::debug!(
185                        "Skipping `pyproject.toml` in `{}` (no `[tool.uv]` section)",
186                        dir.display()
187                    );
188                    return Ok(None);
189                };
190
191                let options = options.relative_to(&std::path::absolute(dir)?)?;
192
193                tracing::debug!("Found workspace configuration at `{}`", path.display());
194                return Ok(Some(Self(options)));
195            }
196            Err(err) if err.kind() == std::io::ErrorKind::NotFound => {}
197            Err(err) => return Err(err.into()),
198        }
199
200        Ok(None)
201    }
202
203    /// Load a [`FilesystemOptions`] from a `uv.toml` file.
204    pub fn from_file(path: impl AsRef<Path>) -> Result<Self, Error> {
205        let path = path.as_ref();
206        tracing::debug!("Reading user configuration from: `{}`", path.display());
207
208        let options = read_file(path)?;
209        validate_uv_toml(path, &options)?;
210        Ok(Self(options))
211    }
212}
213
214impl From<Options> for FilesystemOptions {
215    fn from(options: Options) -> Self {
216        Self(options)
217    }
218}
219
220/// Load [`Options`] from a `uv.toml` file.
221fn read_file(path: &Path) -> Result<Options, Error> {
222    let content = fs_err::read_to_string(path)?;
223    let options = info_span!("toml::from_str filesystem options uv.toml", path = %path.display())
224        .in_scope(|| toml::from_str::<Options>(&content))
225        .map_err(|err| {
226            check_uv_toml_required_version(
227                path,
228                &content,
229                Error::UvToml(path.to_path_buf(), Box::new(err)),
230            )
231        })?;
232    let options = if let Some(parent) = std::path::absolute(path)?.parent() {
233        options.relative_to(parent)?
234    } else {
235        options
236    };
237    Ok(options)
238}
239
240/// If `required_version` is set and incompatible with the running uv, return the corresponding
241/// [`Error::RequiredVersion`].
242fn required_version_mismatch(required_version: Option<RequiredVersion>) -> Option<Error> {
243    let required_version = required_version?;
244    let package_version = Version::from_str(uv_version::version())
245        .expect("uv crate version to be a valid PEP 440 version");
246    if required_version.contains(&package_version) {
247        None
248    } else {
249        Some(Error::RequiredVersion {
250            required_version,
251            package_version,
252        })
253    }
254}
255
256/// On a `pyproject.toml` settings parse error, check whether `tool.uv.required-version` should
257/// take precedence over that error.
258fn check_pyproject_required_version(path: &Path, content: &str, source: toml::de::Error) -> Error {
259    let fallback = || Error::PyprojectToml(path.to_path_buf(), Box::new(source));
260    let Ok(pyproject) = info_span!(
261        "toml::from_str filesystem required-version pyproject.toml",
262        path = %path.display()
263    )
264    .in_scope(|| toml::from_str::<PyProjectRequiredVersionToml>(content)) else {
265        return fallback();
266    };
267
268    let required_version = pyproject
269        .tool
270        .and_then(|tool| tool.uv)
271        .and_then(|uv| uv.required_version);
272    required_version_mismatch(required_version).unwrap_or_else(fallback)
273}
274
275/// On a `uv.toml` settings parse or schema error, check whether top-level `required-version`
276/// should take precedence over that error.
277fn check_uv_toml_required_version(path: &Path, content: &str, source: Error) -> Error {
278    let Ok(uv_toml) = info_span!(
279        "toml::from_str filesystem required-version uv.toml",
280        path = %path.display()
281    )
282    .in_scope(|| toml::from_str::<UvRequiredVersionToml>(content)) else {
283        return source;
284    };
285    required_version_mismatch(uv_toml.required_version).unwrap_or(source)
286}
287
288/// Validate that an [`Options`] schema is compatible with `uv.toml`.
289fn validate_uv_toml(path: &Path, options: &Options) -> Result<(), Error> {
290    // A `required-version` mismatch takes precedence over a schema error.
291    if let Some(err) = required_version_mismatch(options.globals.required_version.clone()) {
292        return Err(err);
293    }
294    let Options {
295        globals: _,
296        top_level: _,
297        install_mirrors: _,
298        publish: _,
299        add: _,
300        audit: _,
301        pip: _,
302        cache_keys: _,
303        override_dependencies: _,
304        exclude_dependencies: _,
305        constraint_dependencies: _,
306        build_constraint_dependencies: _,
307        environments,
308        required_environments,
309        conflicts,
310        workspace,
311        sources,
312        dev_dependencies,
313        default_groups,
314        dependency_groups,
315        managed,
316        package,
317        build_backend,
318    } = options;
319    // The `uv.toml` format is not allowed to include any of the following, which are
320    // permitted by the schema since they _can_ be included in `pyproject.toml` files
321    // (and we want to use `deny_unknown_fields`).
322    if conflicts.is_some() {
323        return Err(Error::PyprojectOnlyField(path.to_path_buf(), "conflicts"));
324    }
325    if workspace.is_some() {
326        return Err(Error::PyprojectOnlyField(path.to_path_buf(), "workspace"));
327    }
328    if sources.is_some() {
329        return Err(Error::PyprojectOnlyField(path.to_path_buf(), "sources"));
330    }
331    if dev_dependencies.is_some() {
332        return Err(Error::PyprojectOnlyField(
333            path.to_path_buf(),
334            "dev-dependencies",
335        ));
336    }
337    if default_groups.is_some() {
338        return Err(Error::PyprojectOnlyField(
339            path.to_path_buf(),
340            "default-groups",
341        ));
342    }
343    if dependency_groups.is_some() {
344        return Err(Error::PyprojectOnlyField(
345            path.to_path_buf(),
346            "dependency-groups",
347        ));
348    }
349    if managed.is_some() {
350        return Err(Error::PyprojectOnlyField(path.to_path_buf(), "managed"));
351    }
352    if package.is_some() {
353        return Err(Error::PyprojectOnlyField(path.to_path_buf(), "package"));
354    }
355    if build_backend.is_some() {
356        return Err(Error::PyprojectOnlyField(
357            path.to_path_buf(),
358            "build-backend",
359        ));
360    }
361    if environments.is_some() {
362        return Err(Error::PyprojectOnlyField(
363            path.to_path_buf(),
364            "environments",
365        ));
366    }
367    if required_environments.is_some() {
368        return Err(Error::PyprojectOnlyField(
369            path.to_path_buf(),
370            "required-environments",
371        ));
372    }
373    Ok(())
374}
375
376/// Validate that an [`Options`] contains no fields that `uv.toml` would mask
377///
378/// This is essentially the inverse of [`validate_uv_toml`].
379#[allow(deprecated)]
380fn warn_uv_toml_masked_fields(options: &Options) {
381    let Options {
382        globals:
383            GlobalOptions {
384                required_version,
385                system_certs,
386                native_tls,
387                offline,
388                no_cache,
389                cache_dir,
390                preview,
391                python_preference,
392                python_downloads,
393                concurrent_downloads,
394                concurrent_builds,
395                concurrent_installs,
396                allow_insecure_host,
397                http_proxy,
398                https_proxy,
399                no_proxy,
400            },
401        top_level:
402            ResolverInstallerSchema {
403                index,
404                index_url,
405                extra_index_url,
406                no_index,
407                find_links,
408                index_strategy,
409                keyring_provider,
410                resolution,
411                prerelease,
412                fork_strategy,
413                dependency_metadata,
414                config_settings,
415                config_settings_package,
416                no_build_isolation,
417                no_build_isolation_package,
418                extra_build_dependencies,
419                extra_build_variables,
420                exclude_newer,
421                exclude_newer_package,
422                link_mode,
423                compile_bytecode,
424                no_sources,
425                no_sources_package: _,
426                upgrade,
427                upgrade_package,
428                reinstall,
429                reinstall_package,
430                no_build,
431                no_build_package,
432                no_binary,
433                no_binary_package,
434                torch_backend,
435            },
436        install_mirrors:
437            PythonInstallMirrors {
438                python_install_mirror,
439                pypy_install_mirror,
440                python_downloads_json_url,
441            },
442        publish:
443            PublishOptions {
444                publish_url,
445                trusted_publishing,
446                check_url,
447            },
448        add: AddOptions { add_bounds },
449        audit: _,
450        pip,
451        cache_keys,
452        override_dependencies,
453        exclude_dependencies,
454        constraint_dependencies,
455        build_constraint_dependencies,
456        environments: _,
457        required_environments: _,
458        conflicts: _,
459        workspace: _,
460        sources: _,
461        dev_dependencies: _,
462        default_groups: _,
463        dependency_groups: _,
464        managed: _,
465        package: _,
466        build_backend: _,
467    } = options;
468
469    let mut masked_fields = vec![];
470
471    if required_version.is_some() {
472        masked_fields.push("required-version");
473    }
474    if system_certs.is_some() {
475        masked_fields.push("system-certs");
476    }
477    if native_tls.is_some() {
478        masked_fields.push("native-tls");
479    }
480    if offline.is_some() {
481        masked_fields.push("offline");
482    }
483    if no_cache.is_some() {
484        masked_fields.push("no-cache");
485    }
486    if cache_dir.is_some() {
487        masked_fields.push("cache-dir");
488    }
489    if preview.is_some() {
490        masked_fields.push("preview");
491    }
492    if python_preference.is_some() {
493        masked_fields.push("python-preference");
494    }
495    if python_downloads.is_some() {
496        masked_fields.push("python-downloads");
497    }
498    if concurrent_downloads.is_some() {
499        masked_fields.push("concurrent-downloads");
500    }
501    if concurrent_builds.is_some() {
502        masked_fields.push("concurrent-builds");
503    }
504    if concurrent_installs.is_some() {
505        masked_fields.push("concurrent-installs");
506    }
507    if allow_insecure_host.is_some() {
508        masked_fields.push("allow-insecure-host");
509    }
510    if http_proxy.is_some() {
511        masked_fields.push("http-proxy");
512    }
513    if https_proxy.is_some() {
514        masked_fields.push("https-proxy");
515    }
516    if no_proxy.is_some() {
517        masked_fields.push("no-proxy");
518    }
519    if index.is_some() {
520        masked_fields.push("index");
521    }
522    if index_url.is_some() {
523        masked_fields.push("index-url");
524    }
525    if extra_index_url.is_some() {
526        masked_fields.push("extra-index-url");
527    }
528    if no_index.is_some() {
529        masked_fields.push("no-index");
530    }
531    if find_links.is_some() {
532        masked_fields.push("find-links");
533    }
534    if index_strategy.is_some() {
535        masked_fields.push("index-strategy");
536    }
537    if keyring_provider.is_some() {
538        masked_fields.push("keyring-provider");
539    }
540    if resolution.is_some() {
541        masked_fields.push("resolution");
542    }
543    if prerelease.is_some() {
544        masked_fields.push("prerelease");
545    }
546    if fork_strategy.is_some() {
547        masked_fields.push("fork-strategy");
548    }
549    if dependency_metadata.is_some() {
550        masked_fields.push("dependency-metadata");
551    }
552    if config_settings.is_some() {
553        masked_fields.push("config-settings");
554    }
555    if config_settings_package.is_some() {
556        masked_fields.push("config-settings-package");
557    }
558    if no_build_isolation.is_some() {
559        masked_fields.push("no-build-isolation");
560    }
561    if no_build_isolation_package.is_some() {
562        masked_fields.push("no-build-isolation-package");
563    }
564    if extra_build_dependencies.is_some() {
565        masked_fields.push("extra-build-dependencies");
566    }
567    if extra_build_variables.is_some() {
568        masked_fields.push("extra-build-variables");
569    }
570    if exclude_newer.is_some() {
571        masked_fields.push("exclude-newer");
572    }
573    if exclude_newer_package.is_some() {
574        masked_fields.push("exclude-newer-package");
575    }
576    if link_mode.is_some() {
577        masked_fields.push("link-mode");
578    }
579    if compile_bytecode.is_some() {
580        masked_fields.push("compile-bytecode");
581    }
582    if no_sources.is_some() {
583        masked_fields.push("no-sources");
584    }
585    if upgrade.is_some() {
586        masked_fields.push("upgrade");
587    }
588    if upgrade_package.is_some() {
589        masked_fields.push("upgrade-package");
590    }
591    if reinstall.is_some() {
592        masked_fields.push("reinstall");
593    }
594    if reinstall_package.is_some() {
595        masked_fields.push("reinstall-package");
596    }
597    if no_build.is_some() {
598        masked_fields.push("no-build");
599    }
600    if no_build_package.is_some() {
601        masked_fields.push("no-build-package");
602    }
603    if no_binary.is_some() {
604        masked_fields.push("no-binary");
605    }
606    if no_binary_package.is_some() {
607        masked_fields.push("no-binary-package");
608    }
609    if torch_backend.is_some() {
610        masked_fields.push("torch-backend");
611    }
612    if python_install_mirror.is_some() {
613        masked_fields.push("python-install-mirror");
614    }
615    if pypy_install_mirror.is_some() {
616        masked_fields.push("pypy-install-mirror");
617    }
618    if python_downloads_json_url.is_some() {
619        masked_fields.push("python-downloads-json-url");
620    }
621    if publish_url.is_some() {
622        masked_fields.push("publish-url");
623    }
624    if trusted_publishing.is_some() {
625        masked_fields.push("trusted-publishing");
626    }
627    if check_url.is_some() {
628        masked_fields.push("check-url");
629    }
630    if add_bounds.is_some() {
631        masked_fields.push("add-bounds");
632    }
633    if pip.is_some() {
634        masked_fields.push("pip");
635    }
636    if cache_keys.is_some() {
637        masked_fields.push("cache_keys");
638    }
639    if override_dependencies.is_some() {
640        masked_fields.push("override-dependencies");
641    }
642    if exclude_dependencies.is_some() {
643        masked_fields.push("exclude-dependencies");
644    }
645    if constraint_dependencies.is_some() {
646        masked_fields.push("constraint-dependencies");
647    }
648    if build_constraint_dependencies.is_some() {
649        masked_fields.push("build-constraint-dependencies");
650    }
651    if !masked_fields.is_empty() {
652        let field_listing = masked_fields.join("\n- ");
653        warn_user!(
654            "Found both a `uv.toml` file and a `[tool.uv]` section in an adjacent `pyproject.toml`. The following fields from `[tool.uv]` will be ignored in favor of the `uv.toml` file:\n- {}",
655            field_listing,
656        );
657    }
658}
659
660#[derive(thiserror::Error, Debug)]
661pub enum Error {
662    #[error(transparent)]
663    Io(#[from] std::io::Error),
664
665    #[error(transparent)]
666    Index(#[from] uv_distribution_types::IndexUrlError),
667
668    #[error("Failed to parse: `{}`", _0.user_display())]
669    PyprojectToml(PathBuf, #[source] Box<toml::de::Error>),
670
671    #[error("Failed to parse: `{}`", _0.user_display())]
672    UvToml(PathBuf, #[source] Box<toml::de::Error>),
673
674    #[error("Failed to parse: `{}`. The `{}` field is not allowed in a `uv.toml` file. `{}` is only applicable in the context of a project, and should be placed in a `pyproject.toml` file instead.", _0.user_display(), _1, _1
675    )]
676    PyprojectOnlyField(PathBuf, &'static str),
677
678    #[error(
679        "Required uv version `{required_version}` does not match the running version `{package_version}`"
680    )]
681    RequiredVersion {
682        required_version: RequiredVersion,
683        package_version: Version,
684    },
685
686    #[error(transparent)]
687    InvalidEnvironmentVariable(#[from] InvalidEnvironmentVariable),
688}
689
690#[derive(Copy, Clone, Debug)]
691pub struct Concurrency {
692    pub downloads: Option<NonZeroUsize>,
693    pub builds: Option<NonZeroUsize>,
694    pub installs: Option<NonZeroUsize>,
695}
696
697/// A boolean flag parsed from an environment variable.
698///
699/// Stores both the value and the environment variable name for use in error messages.
700#[derive(Debug, Clone, Copy)]
701pub struct EnvFlag {
702    pub value: Option<bool>,
703    pub env_var: &'static str,
704}
705
706impl EnvFlag {
707    /// Create a new [`EnvFlag`] by parsing the given environment variable.
708    pub fn new(env_var: &'static str) -> Result<Self, Error> {
709        Ok(Self {
710            value: parse_boolish_environment_variable(env_var)?,
711            env_var,
712        })
713    }
714}
715
716/// Options loaded from environment variables.
717///
718/// This is currently a subset of all respected environment variables, most are parsed via Clap at
719/// the CLI level, however there are limited semantics in that context.
720#[derive(Debug, Clone)]
721pub struct EnvironmentOptions {
722    pub skip_wheel_filename_check: Option<bool>,
723    pub hide_build_output: Option<bool>,
724    pub python_install_bin: Option<bool>,
725    pub python_install_registry: Option<bool>,
726    pub python_no_registry: EnvFlag,
727    pub install_mirrors: PythonInstallMirrors,
728    pub log_context: Option<bool>,
729    pub lfs: Option<bool>,
730    pub http_connect_timeout: Duration,
731    pub http_read_timeout: Duration,
732    /// There's no upload timeout in reqwest, instead we have to use a read timeout as upload
733    /// timeout.
734    pub http_read_timeout_upload: Duration,
735    pub http_retries: u32,
736    pub concurrency: Concurrency,
737    #[cfg(feature = "tracing-durations-export")]
738    pub tracing_durations_file: Option<PathBuf>,
739    pub frozen: EnvFlag,
740    pub locked: EnvFlag,
741    pub offline: EnvFlag,
742    pub no_sync: EnvFlag,
743    pub managed_python: EnvFlag,
744    pub no_managed_python: EnvFlag,
745    pub native_tls: EnvFlag,
746    pub system_certs: EnvFlag,
747    pub preview: EnvFlag,
748    pub isolated: EnvFlag,
749    pub no_progress: EnvFlag,
750    pub no_installer_metadata: EnvFlag,
751    pub dev: EnvFlag,
752    pub no_dev: EnvFlag,
753    pub show_resolution: EnvFlag,
754    pub no_editable: EnvFlag,
755    pub no_env_file: EnvFlag,
756    pub no_group: Option<Vec<GroupName>>,
757    pub no_binary_package: Option<Vec<PackageName>>,
758    pub no_build_package: Option<Vec<PackageName>>,
759    pub no_sources_package: Option<Vec<PackageName>>,
760    pub venv_seed: EnvFlag,
761    pub venv_clear: EnvFlag,
762    pub venv_relocatable: EnvFlag,
763    pub init_bare: EnvFlag,
764    pub malware_check: EnvFlag,
765    pub malware_check_url: Option<DisplaySafeUrl>,
766}
767
768impl EnvironmentOptions {
769    /// Create a new [`EnvironmentOptions`] from environment variables.
770    pub fn new() -> Result<Self, Error> {
771        // Timeout options, matching https://doc.rust-lang.org/nightly/cargo/reference/config.html#httptimeout
772        // `UV_REQUEST_TIMEOUT` is provided for backwards compatibility with v0.1.6
773        let http_read_timeout = parse_integer_environment_variable(
774            EnvVars::UV_HTTP_TIMEOUT,
775            Some("value should be an integer number of seconds"),
776        )?
777        .or(parse_integer_environment_variable(
778            EnvVars::UV_REQUEST_TIMEOUT,
779            Some("value should be an integer number of seconds"),
780        )?)
781        .or(parse_integer_environment_variable(
782            EnvVars::HTTP_TIMEOUT,
783            Some("value should be an integer number of seconds"),
784        )?)
785        .map(Duration::from_secs);
786
787        Ok(Self {
788            skip_wheel_filename_check: parse_boolish_environment_variable(
789                EnvVars::UV_SKIP_WHEEL_FILENAME_CHECK,
790            )?,
791            hide_build_output: parse_boolish_environment_variable(EnvVars::UV_HIDE_BUILD_OUTPUT)?,
792            python_install_bin: parse_boolish_environment_variable(EnvVars::UV_PYTHON_INSTALL_BIN)?,
793            python_install_registry: parse_boolish_environment_variable(
794                EnvVars::UV_PYTHON_INSTALL_REGISTRY,
795            )?,
796            python_no_registry: EnvFlag::new(EnvVars::UV_PYTHON_NO_REGISTRY)?,
797            concurrency: Concurrency {
798                downloads: parse_integer_environment_variable(
799                    EnvVars::UV_CONCURRENT_DOWNLOADS,
800                    None,
801                )?,
802                builds: parse_integer_environment_variable(EnvVars::UV_CONCURRENT_BUILDS, None)?,
803                installs: parse_integer_environment_variable(
804                    EnvVars::UV_CONCURRENT_INSTALLS,
805                    None,
806                )?,
807            },
808            install_mirrors: PythonInstallMirrors {
809                python_install_mirror: parse_string_environment_variable(
810                    EnvVars::UV_PYTHON_INSTALL_MIRROR,
811                )?,
812                pypy_install_mirror: parse_string_environment_variable(
813                    EnvVars::UV_PYPY_INSTALL_MIRROR,
814                )?,
815                python_downloads_json_url: parse_string_environment_variable(
816                    EnvVars::UV_PYTHON_DOWNLOADS_JSON_URL,
817                )?,
818            },
819            log_context: parse_boolish_environment_variable(EnvVars::UV_LOG_CONTEXT)?,
820            lfs: parse_boolish_environment_variable(EnvVars::UV_GIT_LFS)?,
821            http_read_timeout_upload: parse_integer_environment_variable(
822                EnvVars::UV_UPLOAD_HTTP_TIMEOUT,
823                Some("value should be an integer number of seconds"),
824            )?
825            .map(Duration::from_secs)
826            .or(http_read_timeout)
827            .unwrap_or(DEFAULT_READ_TIMEOUT_UPLOAD),
828            http_read_timeout: http_read_timeout.unwrap_or(DEFAULT_READ_TIMEOUT),
829            http_connect_timeout: parse_integer_environment_variable(
830                EnvVars::UV_HTTP_CONNECT_TIMEOUT,
831                Some("value should be an integer number of seconds"),
832            )?
833            .map(Duration::from_secs)
834            .unwrap_or(DEFAULT_CONNECT_TIMEOUT),
835            http_retries: parse_integer_environment_variable(EnvVars::UV_HTTP_RETRIES, None)?
836                .unwrap_or(uv_client::DEFAULT_RETRIES),
837            #[cfg(feature = "tracing-durations-export")]
838            tracing_durations_file: parse_path_environment_variable(
839                EnvVars::TRACING_DURATIONS_FILE,
840            ),
841            frozen: EnvFlag::new(EnvVars::UV_FROZEN)?,
842            locked: EnvFlag::new(EnvVars::UV_LOCKED)?,
843            offline: EnvFlag::new(EnvVars::UV_OFFLINE)?,
844            no_sync: EnvFlag::new(EnvVars::UV_NO_SYNC)?,
845            managed_python: EnvFlag::new(EnvVars::UV_MANAGED_PYTHON)?,
846            no_managed_python: EnvFlag::new(EnvVars::UV_NO_MANAGED_PYTHON)?,
847            native_tls: EnvFlag::new(EnvVars::UV_NATIVE_TLS)?,
848            system_certs: EnvFlag::new(EnvVars::UV_SYSTEM_CERTS)?,
849            preview: EnvFlag::new(EnvVars::UV_PREVIEW)?,
850            isolated: EnvFlag::new(EnvVars::UV_ISOLATED)?,
851            no_progress: EnvFlag::new(EnvVars::UV_NO_PROGRESS)?,
852            no_installer_metadata: EnvFlag::new(EnvVars::UV_NO_INSTALLER_METADATA)?,
853            dev: EnvFlag::new(EnvVars::UV_DEV)?,
854            no_dev: EnvFlag::new(EnvVars::UV_NO_DEV)?,
855            show_resolution: EnvFlag::new(EnvVars::UV_SHOW_RESOLUTION)?,
856            no_editable: EnvFlag::new(EnvVars::UV_NO_EDITABLE)?,
857            no_env_file: EnvFlag::new(EnvVars::UV_NO_ENV_FILE)?,
858            no_group: parse_name_list_environment_variable(EnvVars::UV_NO_GROUP)?,
859            no_binary_package: parse_name_list_environment_variable(EnvVars::UV_NO_BINARY_PACKAGE)?,
860            no_build_package: parse_name_list_environment_variable(EnvVars::UV_NO_BUILD_PACKAGE)?,
861            no_sources_package: parse_name_list_environment_variable(
862                EnvVars::UV_NO_SOURCES_PACKAGE,
863            )?,
864            venv_seed: EnvFlag::new(EnvVars::UV_VENV_SEED)?,
865            venv_clear: EnvFlag::new(EnvVars::UV_VENV_CLEAR)?,
866            venv_relocatable: EnvFlag::new(EnvVars::UV_VENV_RELOCATABLE)?,
867            init_bare: EnvFlag::new(EnvVars::UV_INIT_BARE)?,
868            malware_check: EnvFlag::new(EnvVars::UV_MALWARE_CHECK)?,
869            malware_check_url: parse_string_environment_variable(EnvVars::UV_MALWARE_CHECK_URL)?
870                .map(|value| {
871                    value.parse::<DisplaySafeUrl>().map_err(|err| {
872                        Error::InvalidEnvironmentVariable(InvalidEnvironmentVariable {
873                            name: EnvVars::UV_MALWARE_CHECK_URL.to_string(),
874                            value,
875                            err: err.to_string(),
876                        })
877                    })
878                })
879                .transpose()?,
880        })
881    }
882}
883
884/// Parse a string environment variable.
885fn parse_string_environment_variable(name: &'static str) -> Result<Option<String>, Error> {
886    match std::env::var(name) {
887        Ok(v) => {
888            if v.is_empty() {
889                Ok(None)
890            } else {
891                Ok(Some(v))
892            }
893        }
894        Err(e) => match e {
895            std::env::VarError::NotPresent => Ok(None),
896            std::env::VarError::NotUnicode(err) => Err(Error::InvalidEnvironmentVariable(
897                InvalidEnvironmentVariable {
898                    name: name.to_string(),
899                    value: err.to_string_lossy().to_string(),
900                    err: "expected a valid UTF-8 string".to_string(),
901                },
902            )),
903        },
904    }
905}
906
907/// Parse an environment variable containing a whitespace-delimited list of names.
908fn parse_name_list_environment_variable<T>(name: &'static str) -> Result<Option<Vec<T>>, Error>
909where
910    T: FromStr,
911    <T as FromStr>::Err: std::fmt::Display,
912{
913    let Some(value) = parse_string_environment_variable(name)? else {
914        return Ok(None);
915    };
916
917    let names = value
918        .split_whitespace()
919        .map(|entry| {
920            entry.parse::<T>().map_err(|err| {
921                Error::InvalidEnvironmentVariable(InvalidEnvironmentVariable {
922                    name: name.to_string(),
923                    value: value.clone(),
924                    err: err.to_string(),
925                })
926            })
927        })
928        .collect::<Result<Vec<_>, _>>()?;
929
930    if names.is_empty() {
931        Ok(None)
932    } else {
933        Ok(Some(names))
934    }
935}
936
937fn parse_integer_environment_variable<T>(
938    name: &'static str,
939    help: Option<&str>,
940) -> Result<Option<T>, Error>
941where
942    T: std::str::FromStr + Copy,
943    <T as std::str::FromStr>::Err: std::fmt::Display,
944{
945    let value = match std::env::var(name) {
946        Ok(v) => v,
947        Err(e) => {
948            return match e {
949                std::env::VarError::NotPresent => Ok(None),
950                std::env::VarError::NotUnicode(err) => Err(Error::InvalidEnvironmentVariable(
951                    InvalidEnvironmentVariable {
952                        name: name.to_string(),
953                        value: err.to_string_lossy().to_string(),
954                        err: "expected a valid UTF-8 string".to_string(),
955                    },
956                )),
957            };
958        }
959    };
960    if value.is_empty() {
961        return Ok(None);
962    }
963
964    match value.parse::<T>() {
965        Ok(v) => Ok(Some(v)),
966        Err(err) => Err(Error::InvalidEnvironmentVariable(
967            InvalidEnvironmentVariable {
968                name: name.to_string(),
969                value,
970                err: if let Some(help) = help {
971                    format!("{err}; {help}")
972                } else {
973                    err.to_string()
974                },
975            },
976        )),
977    }
978}
979
980#[cfg(feature = "tracing-durations-export")]
981/// Parse a path environment variable.
982fn parse_path_environment_variable(name: &'static str) -> Option<PathBuf> {
983    let value = std::env::var_os(name)?;
984
985    if value.is_empty() {
986        return None;
987    }
988
989    Some(PathBuf::from(value))
990}
991
992/// Populate the [`EnvironmentFlags`] from the given [`EnvironmentOptions`].
993impl From<&EnvironmentOptions> for EnvironmentFlags {
994    fn from(options: &EnvironmentOptions) -> Self {
995        let mut flags = Self::empty();
996        if options.skip_wheel_filename_check == Some(true) {
997            flags.insert(Self::SKIP_WHEEL_FILENAME_CHECK);
998        }
999        if options.hide_build_output == Some(true) {
1000            flags.insert(Self::HIDE_BUILD_OUTPUT);
1001        }
1002        flags
1003    }
1004}