Skip to content

Commit

Permalink
workspace: Pay attention to members
Browse files Browse the repository at this point in the history
Previously we were ignoring this field and always doing auto-discovery.

Cargo only does auto-discovery if this field is not set.
  • Loading branch information
illicitonion committed Dec 29, 2024
1 parent bf82d09 commit 3710c6e
Show file tree
Hide file tree
Showing 26 changed files with 243 additions and 20 deletions.
141 changes: 123 additions & 18 deletions crate_universe/src/metadata/workspace_discoverer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -50,14 +50,15 @@ fn discover_workspaces_with_cache(
};

// First pass: Discover workspace parents.
for cargo_toml_path in cargo_toml_paths {
if let Some(workspace_parent) = discover_workspace_parent(&cargo_toml_path, manifest_cache)
{
for cargo_toml_path in &cargo_toml_paths {
if let Some(workspace_parent) = discover_workspace_parent(cargo_toml_path, manifest_cache) {
discovered_workspaces
.workspaces_to_members
.insert(workspace_parent, BTreeSet::new());
} else {
discovered_workspaces.non_workspaces.insert(cargo_toml_path);
discovered_workspaces
.non_workspaces
.insert(cargo_toml_path.clone());
}
}

Expand All @@ -69,6 +70,24 @@ fn discover_workspaces_with_cache(
.collect::<BTreeSet<_>>()
{
let workspace_manifest = manifest_cache.get(&workspace_path).unwrap();

let workspace_explicit_members = workspace_manifest
.workspace
.as_ref()
.and_then(|workspace| {
// Unfortunately cargo_toml doesn't preserve presence/absence of this field, so we infer empty means absent.
if workspace.members.is_empty() {
return None;
}
let members = workspace
.members
.iter()
.map(|member| glob::Pattern::new(member).map_err(anyhow::Error::from))
.collect::<Result<BTreeSet<_>>>();
Some(members)
})
.transpose()?;

'per_child: for entry in walkdir::WalkDir::new(workspace_path.parent().unwrap())
.follow_links(true)
.follow_root_links(true)
Expand Down Expand Up @@ -141,19 +160,53 @@ fn discover_workspaces_with_cache(
bail!("Found manifest at {} which is a member of the workspace at {} which isn't included in the crates_universe", child_path, actual_workspace_path);
}

for exclude in &workspace_manifest.workspace.as_ref().unwrap().exclude {
let exclude_path = workspace_path.parent().unwrap().join(exclude);
if exclude_path == child_path.parent().unwrap() {
discovered_workspaces.non_workspaces.insert(child_path);
let dir_relative_to_workspace_dir = child_path
.parent()
.unwrap()
.strip_prefix(workspace_path.parent().unwrap());

if let Ok(dir_relative_to_workspace_dir) = dir_relative_to_workspace_dir {
if workspace_manifest
.workspace
.as_ref()
.unwrap()
.exclude
.contains(&dir_relative_to_workspace_dir.to_string())
{
continue 'per_child;
}
}

discovered_workspaces
.workspaces_to_members
.get_mut(&actual_workspace_path)
.unwrap()
.insert(child_path.clone());
if let (Ok(dir_relative_to_workspace_dir), Some(workspace_explicit_members)) = (
dir_relative_to_workspace_dir,
workspace_explicit_members.as_ref(),
) {
if workspace_explicit_members
.iter()
.any(|glob| glob.matches(dir_relative_to_workspace_dir.as_str()))
{
discovered_workspaces
.workspaces_to_members
.get_mut(&actual_workspace_path)
.unwrap()
.insert(child_path.clone());
}
} else {
discovered_workspaces
.workspaces_to_members
.get_mut(&actual_workspace_path)
.unwrap()
.insert(child_path.clone());
}
}
}

for cargo_toml_path in cargo_toml_paths {
if !discovered_workspaces
.all_workspaces_and_members()
.contains(&cargo_toml_path)
{
discovered_workspaces.non_workspaces.insert(cargo_toml_path);
}
}

Expand Down Expand Up @@ -247,17 +300,13 @@ mod test {
expected.non_workspaces.extend([
root_dir.join("non-ws").join("Cargo.toml"),
root_dir.join("ws2").join("ws2excluded").join("Cargo.toml"),
root_dir
.join("ws2")
.join("ws2excluded")
.join("ws2excluded2")
.join("Cargo.toml"),
]);

let actual = discover_workspaces(
vec![
root_dir.join("ws1/ws1c1/Cargo.toml"),
root_dir.join("ws2/Cargo.toml"),
root_dir.join("ws2/ws2excluded/Cargo.toml"),
root_dir.join("non-ws/Cargo.toml"),
]
.into_iter()
Expand Down Expand Up @@ -334,6 +383,62 @@ mod test {
assert_eq!(expected, actual);
}

#[test]
fn test_discover_explicit_members() {
let r = runfiles::Runfiles::create().unwrap();
let root_dir =
runfiles::rlocation!(r, "rules_rust/crate_universe/test_data/workspace_examples")
.unwrap();
let root_dir = Utf8PathBuf::from_path_buf(root_dir).unwrap();

let mut workspaces_to_members = BTreeMap::<Utf8PathBuf, BTreeSet<Utf8PathBuf>>::new();

let mut includes_members = BTreeSet::new();
let includes_root = root_dir.join("includes");
for child in [
vec!["explicit-child"],
vec!["glob-char1"],
vec!["glob-char2"],
vec!["glob-direct-children", "grandchild1"],
vec!["glob-direct-children", "grandchild2"],
vec!["glob-transitive-children", "level1", "anchor"],
vec!["glob-transitive-children", "level1", "level2", "anchor"],
vec![
"glob-transitive-children",
"level1",
"level2",
"level3",
"anchor",
],
] {
let mut path = includes_root.clone();
for path_part in child {
path.push(path_part);
}
path.push("Cargo.toml");
includes_members.insert(path);
}

workspaces_to_members.insert(includes_root.join("Cargo.toml"), includes_members);
let non_workspaces = BTreeSet::<Utf8PathBuf>::new();

let expected = DiscoveredWorkspaces {
workspaces_to_members,
non_workspaces,
};

let actual = discover_workspaces(
vec![root_dir.join("includes/explicit-child/Cargo.toml")]
.into_iter()
.collect(),
&BTreeMap::new(),
root_dir.as_std_path(),
)
.unwrap();

assert_eq!(expected, actual);
}

fn ws1_discovered_workspaces(root_dir: &Utf8Path) -> DiscoveredWorkspaces {
let mut workspaces_to_members = BTreeMap::new();
workspaces_to_members.insert(
Expand Down
18 changes: 18 additions & 0 deletions crate_universe/test_data/workspace_examples/includes/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
[workspace]
members = [
"explicit-child",
"glob-direct-children/*",
"glob-transitive-children/**/anchor",
"glob-char?",
]
exclude = [
"glob-char3",
"glob-direct-children/excluded",
]

[package]
name = "includes"
version = "0.1.0"
edition = "2021"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "explicit-child"
version = "0.1.0"
edition = "2021"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
fn main() {
println!("Hello, world!");
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "glob-char1"
version = "0.1.0"
edition = "2021"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
fn main() {
println!("Hello, world!");
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "glob-char2"
version = "0.1.0"
edition = "2021"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
fn main() {
println!("Hello, world!");
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "glob-char3"
version = "0.1.0"
edition = "2021"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
fn main() {
println!("Hello, world!");
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "excluded"
version = "0.1.0"
edition = "2021"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
fn main() {
println!("Hello, world!");
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "grandchild1"
version = "0.1.0"
edition = "2021"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
fn main() {
println!("Hello, world!");
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "grandchild2"
version = "0.1.0"
edition = "2021"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
fn main() {
println!("Hello, world!");
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "anchor1"
version = "0.1.0"
edition = "2021"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
fn main() {
println!("Hello, world!");
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "anchor2"
version = "0.1.0"
edition = "2021"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
fn main() {
println!("Hello, world!");
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "anchor3"
version = "0.1.0"
edition = "2021"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
fn main() {
println!("Hello, world!");
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
[package]
name = "non-explicit-child"
version = "0.1.0"
edition = "2021"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
fn main() {
println!("Hello, world!");
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
fn main() {
println!("Hello, world!");
}
2 changes: 0 additions & 2 deletions crate_universe/test_data/workspace_examples/ws1/Cargo.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,4 @@
[workspace]
members = ["ws1c1", "ws1c2"]

[package]
name = "ws1"
version = "0.1.0"
Expand Down

0 comments on commit 3710c6e

Please sign in to comment.