@@ -285,7 +285,7 @@ fn migrate_manifests(ws: &Workspace<'_>, pkgs: &[&Package]) -> CargoResult<()> {
285
285
fixes += rename_dep_fields_2024 ( workspace, "dependencies" ) ;
286
286
}
287
287
288
- fixes += add_feature_for_unused_deps ( pkg, root) ;
288
+ fixes += add_feature_for_unused_deps ( pkg, root, ws . gctx ( ) ) ;
289
289
fixes += rename_table ( root, "project" , "package" ) ;
290
290
if let Some ( target) = root. get_mut ( "lib" ) . and_then ( |t| t. as_table_like_mut ( ) ) {
291
291
fixes += rename_target_fields_2024 ( target) ;
@@ -435,7 +435,11 @@ fn rename_table(parent: &mut dyn toml_edit::TableLike, old: &str, new: &str) ->
435
435
1
436
436
}
437
437
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 {
439
443
let manifest = pkg. manifest ( ) ;
440
444
441
445
let activated_opt_deps = manifest
@@ -461,14 +465,43 @@ fn add_feature_for_unused_deps(pkg: &Package, parent: &mut dyn toml_edit::TableL
461
465
. or_insert ( toml_edit:: table ( ) )
462
466
. as_table_like_mut ( )
463
467
{
468
+ let activate_dep = format ! ( "dep:{dep_name_in_toml}" ) ;
469
+ let strong_dep_feature_prefix = format ! ( "{dep_name_in_toml}/" ) ;
464
470
features
465
471
. entry ( dep_name_in_toml. as_str ( ) )
466
472
. or_insert_with ( || {
467
473
fixes += 1 ;
468
474
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 ] ) ,
470
476
) )
471
477
} ) ;
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
+ }
472
505
}
473
506
}
474
507
}
0 commit comments