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