Skip to content

Commit

Permalink
Resolve crates.io workspace dependencies
Browse files Browse the repository at this point in the history
  • Loading branch information
messense committed Sep 25, 2022
1 parent 8564be1 commit 0356306
Show file tree
Hide file tree
Showing 4 changed files with 71 additions and 4 deletions.
68 changes: 64 additions & 4 deletions src/source_distribution.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,17 +19,19 @@ const LOCAL_DEPENDENCIES_FOLDER: &str = "local_dependencies";
/// This method is rather frail, but unfortunately I don't know a better solution.
fn rewrite_cargo_toml(
manifest_path: impl AsRef<Path>,
workspace_manifest: &toml_edit::Document,
known_path_deps: &HashMap<String, PathBuf>,
local_deps_folder: String,
root_crate: bool,
) -> Result<String> {
let manifest_path = manifest_path.as_ref();
let text = fs::read_to_string(&manifest_path).context(format!(
"Can't read Cargo.toml at {}",
manifest_path.as_ref().display(),
manifest_path.display(),
))?;
let mut data = text.parse::<toml_edit::Document>().context(format!(
"Failed to parse Cargo.toml at {}",
manifest_path.as_ref().display()
manifest_path.display()
))?;
let mut rewritten = false;
// ˇˇˇˇˇˇˇˇˇˇˇˇ dep_category
Expand All @@ -39,6 +41,10 @@ fn rewrite_cargo_toml(
// ^^^^^^^^^^^^^ dep_name
for dep_category in &["dependencies", "dev-dependencies", "build-dependencies"] {
if let Some(table) = data.get_mut(*dep_category).and_then(|x| x.as_table_mut()) {
let workspace_deps = workspace_manifest
.get("workspace")
.and_then(|x| x.get(dep_category))
.and_then(|x| x.as_table());
let dep_names: Vec<_> = table.iter().map(|(key, _)| key.to_string()).collect();
for dep_name in dep_names {
let workspace_inherit = table
Expand All @@ -55,19 +61,63 @@ fn rewrite_cargo_toml(
if !table[&dep_name]["path"].is_str() {
bail!(
"In {}, {} {} has a path value that is not a string",
manifest_path.as_ref().display(),
manifest_path.display(),
dep_category,
dep_name
)
}
} else {
// If a workspace inherited dependency isn't a path dep,
// we need to replace `workspace = true` with its full requirement spec.
if !known_path_deps.contains_key(&dep_name) {
if let Some(workspace_dep) = workspace_deps.and_then(|x| x.get(&dep_name)) {
let mut workspace_dep = workspace_dep.clone();
// Merge optional and features from the current Cargo.toml
if table[&dep_name].get("optional").is_some() {
workspace_dep["optional"] = table[&dep_name]["optional"].clone();
}
if let Some(features) =
table[&dep_name].get("features").and_then(|x| x.as_array())
{
let existing_features = workspace_dep
.as_table_like_mut()
.unwrap()
.entry("features")
.or_insert_with(|| {
toml_edit::Item::Value(toml_edit::Array::new().into())
})
.as_array_mut()
.with_context(|| {
format!(
"In {}, {} {} has a features value that is not an array",
manifest_path.display(),
dep_category,
dep_name
)
})?;
existing_features.extend(features);
}
table[&dep_name] = workspace_dep;
rewritten = true;
} else {
bail!(
"In {}, {} {} is marked as `workspace = true`, but it is found neither in \
the workspace manifest nor in the known path dependencies",
manifest_path.display(),
dep_category,
dep_name
)
}
continue;
}
}
if !known_path_deps.contains_key(&dep_name) {
bail!(
"cargo metadata does not know about the path for {}.{} present in {}, \
which should never happen ಠ_ಠ",
dep_category,
dep_name,
manifest_path.as_ref().display()
manifest_path.display()
);
}
// This is the location of the targeted crate in the source distribution
Expand Down Expand Up @@ -136,6 +186,7 @@ fn add_crate_to_source_distribution(
writer: &mut SDistWriter,
pyproject_toml_path: impl AsRef<Path>,
manifest_path: impl AsRef<Path>,
workspace_manifest: &toml_edit::Document,
prefix: impl AsRef<Path>,
known_path_deps: &HashMap<String, PathBuf>,
root_crate: bool,
Expand Down Expand Up @@ -230,6 +281,7 @@ fn add_crate_to_source_distribution(
};
let rewritten_cargo_toml = rewrite_cargo_toml(
&manifest_path,
workspace_manifest,
known_path_deps,
local_deps_folder,
root_crate,
Expand Down Expand Up @@ -296,6 +348,12 @@ pub fn source_distribution(
let metadata21 = &build_context.metadata21;
let manifest_path = &build_context.manifest_path;
let pyproject_toml_path = fs::canonicalize(&build_context.pyproject_toml_path)?;
let workspace_manifest_path = build_context
.cargo_metadata
.workspace_root
.join("Cargo.toml");
let workspace_manifest: toml_edit::Document =
fs::read_to_string(&workspace_manifest_path)?.parse()?;

let known_path_deps = find_path_deps(&build_context.cargo_metadata)?;

Expand All @@ -312,6 +370,7 @@ pub fn source_distribution(
&mut writer,
&pyproject_toml_path,
&path_dep,
&workspace_manifest,
&root_dir.join(LOCAL_DEPENDENCIES_FOLDER).join(name),
&known_path_deps,
false,
Expand All @@ -328,6 +387,7 @@ pub fn source_distribution(
&mut writer,
&pyproject_toml_path,
&manifest_path,
&workspace_manifest,
&root_dir,
&known_path_deps,
true,
Expand Down
1 change: 1 addition & 0 deletions test-crates/workspace-inheritance/Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions test-crates/workspace-inheritance/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -5,4 +5,5 @@ members = [
]

[workspace.dependencies]
libc = { version = "0.2", features = ["std"] }
generic_lib = { path = "generic_lib" }
5 changes: 5 additions & 0 deletions test-crates/workspace-inheritance/python/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -11,3 +11,8 @@ crate-type = ["cdylib"]
[dependencies]
pyo3 = { version = "0.16.5", features = ["extension-module"] }
generic_lib.workspace = true

[dependencies.libc]
workspace = true
optional = true
features = ["extra_traits"]

0 comments on commit 0356306

Please sign in to comment.