mirror of
				https://github.com/astral-sh/uv.git
				synced 2025-10-26 01:48:20 +00:00 
			
		
		
		
	Avoid including non-excluded members in parent workspaces (#6735)
## Summary If you're in a directory, and there's workspace above it, we check if the directory is excluded from the workspace members... But not if it's _included_ in the first place. Closes https://github.com/astral-sh/uv/issues/6732.
This commit is contained in:
		
							parent
							
								
									56cc0c9b3c
								
							
						
					
					
						commit
						485e0d2748
					
				
					 2 changed files with 85 additions and 1 deletions
				
			
		|  | @ -1006,6 +1006,14 @@ async fn find_workspace( | ||||||
|             .and_then(|tool| tool.uv.as_ref()) |             .and_then(|tool| tool.uv.as_ref()) | ||||||
|             .and_then(|uv| uv.workspace.as_ref()) |             .and_then(|uv| uv.workspace.as_ref()) | ||||||
|         { |         { | ||||||
|  |             if !is_included_in_workspace(project_root, workspace_root, workspace)? { | ||||||
|  |                 debug!( | ||||||
|  |                     "Found workspace root `{}`, but project is not included", | ||||||
|  |                     workspace_root.simplified_display() | ||||||
|  |                 ); | ||||||
|  |                 return Ok(None); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             if is_excluded_from_workspace(project_root, workspace_root, workspace)? { |             if is_excluded_from_workspace(project_root, workspace_root, workspace)? { | ||||||
|                 debug!( |                 debug!( | ||||||
|                     "Found workspace root `{}`, but project is excluded", |                     "Found workspace root `{}`, but project is excluded", | ||||||
|  | @ -1101,6 +1109,21 @@ pub fn check_nested_workspaces(inner_workspace_root: &Path, options: &DiscoveryO | ||||||
|             .and_then(|tool| tool.uv.as_ref()) |             .and_then(|tool| tool.uv.as_ref()) | ||||||
|             .and_then(|uv| uv.workspace.as_ref()) |             .and_then(|uv| uv.workspace.as_ref()) | ||||||
|         { |         { | ||||||
|  |             let is_included = match is_included_in_workspace( | ||||||
|  |                 inner_workspace_root, | ||||||
|  |                 outer_workspace_root, | ||||||
|  |                 workspace, | ||||||
|  |             ) { | ||||||
|  |                 Ok(contents) => contents, | ||||||
|  |                 Err(err) => { | ||||||
|  |                     warn!( | ||||||
|  |                         "Invalid pyproject.toml `{}`: {err}", | ||||||
|  |                         pyproject_toml_path.simplified_display() | ||||||
|  |                     ); | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|             let is_excluded = match is_excluded_from_workspace( |             let is_excluded = match is_excluded_from_workspace( | ||||||
|                 inner_workspace_root, |                 inner_workspace_root, | ||||||
|                 outer_workspace_root, |                 outer_workspace_root, | ||||||
|  | @ -1115,7 +1138,8 @@ pub fn check_nested_workspaces(inner_workspace_root: &Path, options: &DiscoveryO | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|             }; |             }; | ||||||
|             if !is_excluded { | 
 | ||||||
|  |             if is_included && !is_excluded { | ||||||
|                 warn_user!( |                 warn_user!( | ||||||
|                     "Nested workspaces are not supported, but outer workspace (`{}`) includes `{}`", |                     "Nested workspaces are not supported, but outer workspace (`{}`) includes `{}`", | ||||||
|                     outer_workspace_root.simplified_display().cyan(), |                     outer_workspace_root.simplified_display().cyan(), | ||||||
|  |  | ||||||
|  | @ -948,6 +948,66 @@ fn workspace_hidden_member() -> Result<()> { | ||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Ensure that workspace discovery accepts valid hidden directories.
 | ||||||
|  | #[test] | ||||||
|  | fn workspace_non_included_member() -> Result<()> { | ||||||
|  |     let context = TestContext::new("3.12"); | ||||||
|  | 
 | ||||||
|  |     // Build the main workspace ...
 | ||||||
|  |     let workspace = context.temp_dir.child("workspace"); | ||||||
|  |     workspace.child("pyproject.toml").write_str(indoc! {r#" | ||||||
|  |         [tool.uv.workspace] | ||||||
|  |         members = ["packages/*"] | ||||||
|  |     "#})?;
 | ||||||
|  | 
 | ||||||
|  |     // ... with a  ...
 | ||||||
|  |     let deps = indoc! {r#" | ||||||
|  |         dependencies = ["b"] | ||||||
|  | 
 | ||||||
|  |         [tool.uv.sources] | ||||||
|  |         b = { workspace = true } | ||||||
|  |     "#};
 | ||||||
|  |     make_project(&workspace.join("packages").join("a"), "a", deps)?; | ||||||
|  | 
 | ||||||
|  |     // ... and b.
 | ||||||
|  |     let deps = indoc! {r" | ||||||
|  |         dependencies = [] | ||||||
|  |     "};
 | ||||||
|  |     make_project(&workspace.join("packages").join("b"), "b", deps)?; | ||||||
|  | 
 | ||||||
|  |     // ... and c, which is _not_ a member, but also isn't explicitly excluded.
 | ||||||
|  |     let deps = indoc! {r" | ||||||
|  |         dependencies = [] | ||||||
|  |     "};
 | ||||||
|  |     make_project(&workspace.join("c"), "c", deps)?; | ||||||
|  | 
 | ||||||
|  |     // Locking from `c` should not include any workspace members.
 | ||||||
|  |     uv_snapshot!(context.filters(), context.lock().current_dir(workspace.join("c")), @r###" | ||||||
|  |     success: true | ||||||
|  |     exit_code: 0 | ||||||
|  |     ----- stdout ----- | ||||||
|  | 
 | ||||||
|  |     ----- stderr ----- | ||||||
|  |     Using Python 3.12.[X] interpreter at: [PYTHON-3.12] | ||||||
|  |     Resolved 1 package in [TIME] | ||||||
|  |     "###
 | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     let lock: SourceLock = toml::from_str(&fs_err::read_to_string( | ||||||
|  |         workspace.join("c").join("uv.lock"), | ||||||
|  |     )?)?; | ||||||
|  | 
 | ||||||
|  |     assert_json_snapshot!(lock.sources(), @r###" | ||||||
|  |     { | ||||||
|  |       "c": { | ||||||
|  |         "editable": "." | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     "###);
 | ||||||
|  | 
 | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /// Ensure workspace members inherit sources from the root, if not specified in the member.
 | /// Ensure workspace members inherit sources from the root, if not specified in the member.
 | ||||||
| ///
 | ///
 | ||||||
| /// In such cases, relative paths should be resolved relative to the workspace root, rather than
 | /// In such cases, relative paths should be resolved relative to the workspace root, rather than
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Charlie Marsh
						Charlie Marsh