Skip to content

Commit 21c0928

Browse files
committed
fix(fix): Ensure optional dep is available for dep-features
1 parent 078383b commit 21c0928

File tree

2 files changed

+40
-7
lines changed

2 files changed

+40
-7
lines changed

src/cargo/ops/fix.rs

+36-3
Original file line numberDiff line numberDiff line change
@@ -285,7 +285,7 @@ fn migrate_manifests(ws: &Workspace<'_>, pkgs: &[&Package]) -> CargoResult<()> {
285285
fixes += rename_dep_fields_2024(workspace, "dependencies");
286286
}
287287

288-
fixes += add_feature_for_unused_deps(pkg, root);
288+
fixes += add_feature_for_unused_deps(pkg, root, ws.gctx());
289289
fixes += rename_table(root, "project", "package");
290290
if let Some(target) = root.get_mut("lib").and_then(|t| t.as_table_like_mut()) {
291291
fixes += rename_target_fields_2024(target);
@@ -435,7 +435,11 @@ fn rename_table(parent: &mut dyn toml_edit::TableLike, old: &str, new: &str) ->
435435
1
436436
}
437437

438-
fn add_feature_for_unused_deps(pkg: &Package, parent: &mut dyn toml_edit::TableLike) -> usize {
438+
fn add_feature_for_unused_deps(
439+
pkg: &Package,
440+
parent: &mut dyn toml_edit::TableLike,
441+
gctx: &GlobalContext,
442+
) -> usize {
439443
let manifest = pkg.manifest();
440444

441445
let activated_opt_deps = manifest
@@ -461,14 +465,43 @@ fn add_feature_for_unused_deps(pkg: &Package, parent: &mut dyn toml_edit::TableL
461465
.or_insert(toml_edit::table())
462466
.as_table_like_mut()
463467
{
468+
let activate_dep = format!("dep:{dep_name_in_toml}");
469+
let strong_dep_feature_prefix = format!("{dep_name_in_toml}/");
464470
features
465471
.entry(dep_name_in_toml.as_str())
466472
.or_insert_with(|| {
467473
fixes += 1;
468474
toml_edit::Item::Value(toml_edit::Value::Array(
469-
toml_edit::Array::from_iter(&[format!("dep:{}", dep_name_in_toml)]),
475+
toml_edit::Array::from_iter([&activate_dep]),
470476
))
471477
});
478+
// Ensure `dep:dep_name` is present for `dep_name/feature_name` since `dep:` is the
479+
// only way to guarantee an optional dependency is available for use.
480+
//
481+
// The way we avoid implicitly creating features in Edition2024 is we remove the
482+
// dependency from `resolved_toml` if there is no `dep:` syntax as that is the only
483+
// syntax that suppresses the creation of the implicit feature.
484+
for (feature_name, activations) in features.iter_mut() {
485+
let Some(activations) = activations.as_array_mut() else {
486+
let _ = gctx.shell().warn(format_args!("skipping fix of feature `{feature_name}` in package `{}`: unsupported feature schema", pkg.name()));
487+
continue;
488+
};
489+
if activations
490+
.iter()
491+
.any(|a| a.as_str().map(|a| a == activate_dep).unwrap_or(false))
492+
{
493+
continue;
494+
}
495+
let Some(activate_dep_pos) = activations.iter().position(|a| {
496+
a.as_str()
497+
.map(|a| a.starts_with(&strong_dep_feature_prefix))
498+
.unwrap_or(false)
499+
}) else {
500+
continue;
501+
};
502+
fixes += 1;
503+
activations.insert(activate_dep_pos, &activate_dep);
504+
}
472505
}
473506
}
474507
}

tests/testsuite/fix.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -2705,7 +2705,7 @@ unrelated-dep-feature = ["unrelated-feature/a", "unrelated-feature/b"]
27052705
.with_stderr(
27062706
"\
27072707
[MIGRATING] Cargo.toml from 2021 edition to 2024
2708-
[FIXED] Cargo.toml (1 fix)
2708+
[FIXED] Cargo.toml (4 fixes)
27092709
[UPDATING] `dummy-registry` index
27102710
[LOCKING] 5 packages to latest compatible versions
27112711
[CHECKING] foo v0.1.0 ([CWD])
@@ -2730,11 +2730,11 @@ renamed-feature = { version = "0.1.0", optional = true }
27302730
unrelated-feature = { version = "0.1.0", optional = true }
27312731
27322732
[features]
2733-
dep-feature = ["dep-feature/a", "dep-feature/b"]
2733+
dep-feature = [ "dep:dep-feature","dep-feature/a", "dep-feature/b"]
27342734
dep-and-dep-feature = ["dep:dep-and-dep-feature", "dep-and-dep-feature/a", "dep-and-dep-feature/b"]
2735-
renamed = ["renamed-feature/a", "renamed-feature/b"]
2735+
renamed = [ "dep:renamed-feature","renamed-feature/a", "renamed-feature/b"]
27362736
unrelated-feature = []
2737-
unrelated-dep-feature = ["unrelated-feature/a", "unrelated-feature/b"]
2737+
unrelated-dep-feature = [ "dep:unrelated-feature","unrelated-feature/a", "unrelated-feature/b"]
27382738
renamed-feature = ["dep:renamed-feature"]
27392739
27402740
"#]],

0 commit comments

Comments
 (0)