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#[derive(Debug, Clone)]
27pub struct FilesystemOptions(Options);
28
29impl FilesystemOptions {
30 pub fn into_options(self) -> Options {
32 self.0
33 }
34
35 #[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 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 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 }
108 Err(Error::PyprojectToml(path, err)) => {
109 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 return Err(err);
119 }
120 }
121 }
122 Ok(None)
123 }
124
125 pub fn from_directory(dir: &Path) -> Result<Option<Self>, Error> {
128 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 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 let path = dir.join("pyproject.toml");
167 match fs_err::read_to_string(&path) {
168 Ok(content) => {
169 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 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
220fn 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
240fn 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
256fn 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
275fn 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
288fn validate_uv_toml(path: &Path, options: &Options) -> Result<(), Error> {
290 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 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#[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#[derive(Debug, Clone, Copy)]
701pub struct EnvFlag {
702 pub value: Option<bool>,
703 pub env_var: &'static str,
704}
705
706impl EnvFlag {
707 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#[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 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 pub fn new() -> Result<Self, Error> {
771 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
884fn 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
907fn 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")]
981fn 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
992impl 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}