@@ -57,6 +57,7 @@ use url::Url;
57
57
58
58
use core:: { PackageId , Registry , SourceId , Summary , Dependency } ;
59
59
use core:: PackageIdSpec ;
60
+ use util:: config:: Config ;
60
61
use util:: Graph ;
61
62
use util:: errors:: { CargoResult , CargoError } ;
62
63
use util:: profile;
@@ -330,20 +331,25 @@ struct Context<'a> {
330
331
resolve_replacements : RcList < ( PackageId , PackageId ) > ,
331
332
332
333
replacements : & ' a [ ( PackageIdSpec , Dependency ) ] ,
334
+
335
+ // These warnings are printed after resolution.
336
+ warnings : RcList < String > ,
333
337
}
334
338
335
339
type Activations = HashMap < String , HashMap < SourceId , Vec < Summary > > > ;
336
340
337
341
/// Builds the list of all packages required to build the first argument.
338
342
pub fn resolve ( summaries : & [ ( Summary , Method ) ] ,
339
343
replacements : & [ ( PackageIdSpec , Dependency ) ] ,
340
- registry : & mut Registry ) -> CargoResult < Resolve > {
344
+ registry : & mut Registry ,
345
+ config : Option < & Config > ) -> CargoResult < Resolve > {
341
346
let cx = Context {
342
347
resolve_graph : RcList :: new ( ) ,
343
348
resolve_features : HashMap :: new ( ) ,
344
349
resolve_replacements : RcList :: new ( ) ,
345
350
activations : HashMap :: new ( ) ,
346
351
replacements : replacements,
352
+ warnings : RcList :: new ( ) ,
347
353
} ;
348
354
let _p = profile:: start ( format ! ( "resolving" ) ) ;
349
355
let cx = activate_deps_loop ( cx, registry, summaries) ?;
@@ -368,8 +374,18 @@ pub fn resolve(summaries: &[(Summary, Method)],
368
374
}
369
375
370
376
check_cycles ( & resolve, & cx. activations ) ?;
371
-
372
377
trace ! ( "resolved: {:?}" , resolve) ;
378
+
379
+ // If we have a shell, emit warnings about required deps used as feature.
380
+ if let Some ( config) = config {
381
+ let mut shell = config. shell ( ) ;
382
+ let mut warnings = & cx. warnings ;
383
+ while let Some ( ref head) = warnings. head {
384
+ shell. warn ( & head. 0 ) ?;
385
+ warnings = & head. 1 ;
386
+ }
387
+ }
388
+
373
389
Ok ( resolve)
374
390
}
375
391
@@ -827,13 +843,15 @@ fn compatible(a: &semver::Version, b: &semver::Version) -> bool {
827
843
//
828
844
// The feature dependencies map is a mapping of package name to list of features
829
845
// enabled. Each package should be enabled, and each package should have the
830
- // specified set of features enabled.
846
+ // specified set of features enabled. The boolean indicates whether this
847
+ // package was specifically requested (rather than just requesting features
848
+ // *within* this package).
831
849
//
832
850
// The all used features set is the set of features which this local package had
833
851
// enabled, which is later used when compiling to instruct the code what
834
852
// features were enabled.
835
853
fn build_features < ' a > ( s : & ' a Summary , method : & ' a Method )
836
- -> CargoResult < ( HashMap < & ' a str , Vec < String > > , HashSet < & ' a str > ) > {
854
+ -> CargoResult < ( HashMap < & ' a str , ( bool , Vec < String > ) > , HashSet < & ' a str > ) > {
837
855
let mut deps = HashMap :: new ( ) ;
838
856
let mut used = HashSet :: new ( ) ;
839
857
let mut visited = HashSet :: new ( ) ;
@@ -867,7 +885,7 @@ fn build_features<'a>(s: &'a Summary, method: &'a Method)
867
885
868
886
fn add_feature < ' a > ( s : & ' a Summary ,
869
887
feat : & ' a str ,
870
- deps : & mut HashMap < & ' a str , Vec < String > > ,
888
+ deps : & mut HashMap < & ' a str , ( bool , Vec < String > ) > ,
871
889
used : & mut HashSet < & ' a str > ,
872
890
visited : & mut HashSet < & ' a str > ) -> CargoResult < ( ) > {
873
891
if feat. is_empty ( ) { return Ok ( ( ) ) }
@@ -884,8 +902,8 @@ fn build_features<'a>(s: &'a Summary, method: &'a Method)
884
902
let package = feat_or_package;
885
903
used. insert ( package) ;
886
904
deps. entry ( package)
887
- . or_insert ( Vec :: new ( ) )
888
- . push ( feat. to_string ( ) ) ;
905
+ . or_insert ( ( false , Vec :: new ( ) ) )
906
+ . 1 . push ( feat. to_string ( ) ) ;
889
907
}
890
908
None => {
891
909
let feat = feat_or_package;
@@ -896,12 +914,14 @@ fn build_features<'a>(s: &'a Summary, method: &'a Method)
896
914
used. insert ( feat) ;
897
915
match s. features ( ) . get ( feat) {
898
916
Some ( recursive) => {
917
+ // This is a feature, add it recursively.
899
918
for f in recursive {
900
919
add_feature ( s, f, deps, used, visited) ?;
901
920
}
902
921
}
903
922
None => {
904
- deps. entry ( feat) . or_insert ( Vec :: new ( ) ) ;
923
+ // This is a dependency, mark it as explicitly requested.
924
+ deps. entry ( feat) . or_insert ( ( false , Vec :: new ( ) ) ) . 0 = true ;
905
925
}
906
926
}
907
927
visited. remove ( feat) ;
@@ -1057,8 +1077,9 @@ impl<'a> Context<'a> {
1057
1077
. unwrap_or ( & [ ] )
1058
1078
}
1059
1079
1080
+ /// Return all dependencies and the features we want from them.
1060
1081
fn resolve_features < ' b > ( & mut self ,
1061
- candidate : & ' b Summary ,
1082
+ s : & ' b Summary ,
1062
1083
method : & ' b Method )
1063
1084
-> CargoResult < Vec < ( Dependency , Vec < String > ) > > {
1064
1085
let dev_deps = match * method {
@@ -1067,21 +1088,31 @@ impl<'a> Context<'a> {
1067
1088
} ;
1068
1089
1069
1090
// First, filter by dev-dependencies
1070
- let deps = candidate . dependencies ( ) ;
1091
+ let deps = s . dependencies ( ) ;
1071
1092
let deps = deps. iter ( ) . filter ( |d| d. is_transitive ( ) || dev_deps) ;
1072
1093
1073
- let ( mut feature_deps, used_features) = build_features ( candidate,
1074
- method) ?;
1094
+ let ( mut feature_deps, used_features) = build_features ( s, method) ?;
1075
1095
let mut ret = Vec :: new ( ) ;
1076
1096
1077
- // Next, sanitize all requested features by whitelisting all the
1078
- // requested features that correspond to optional dependencies
1097
+ // Next, collect all actually enabled dependencies and their features.
1079
1098
for dep in deps {
1080
- // weed out optional dependencies, but not those required
1099
+ // Skip optional dependencies, but not those enabled through a feature
1081
1100
if dep. is_optional ( ) && !feature_deps. contains_key ( dep. name ( ) ) {
1082
1101
continue
1083
1102
}
1084
- let mut base = feature_deps. remove ( dep. name ( ) ) . unwrap_or ( vec ! [ ] ) ;
1103
+ // So we want this dependency. Move the features we want from `feature_deps`
1104
+ // to `ret`.
1105
+ let base = feature_deps. remove ( dep. name ( ) ) . unwrap_or ( ( false , vec ! [ ] ) ) ;
1106
+ if !dep. is_optional ( ) && base. 0 {
1107
+ self . warnings . push (
1108
+ format ! ( "Package `{}` does not have feature `{}`. It has a required dependency \
1109
+ with that name, but only optional dependencies can be used as features. \
1110
+ This is currently a warning to ease the transition, but it will become an \
1111
+ error in the future.",
1112
+ s. package_id( ) , dep. name( ) )
1113
+ ) ;
1114
+ }
1115
+ let mut base = base. 1 ;
1085
1116
base. extend ( dep. features ( ) . iter ( ) . cloned ( ) ) ;
1086
1117
for feature in base. iter ( ) {
1087
1118
if feature. contains ( "/" ) {
@@ -1091,23 +1122,20 @@ impl<'a> Context<'a> {
1091
1122
ret. push ( ( dep. clone ( ) , base) ) ;
1092
1123
}
1093
1124
1094
- // All features can only point to optional dependencies, in which case
1095
- // they should have all been weeded out by the above iteration. Any
1096
- // remaining features are bugs in that the package does not actually
1097
- // have those features.
1125
+ // Any remaining entries in feature_deps are bugs in that the package does not actually
1126
+ // have those dependencies. We classified them as dependencies in the first place
1127
+ // because there is no such feature, either.
1098
1128
if !feature_deps. is_empty ( ) {
1099
1129
let unknown = feature_deps. keys ( ) . map ( |s| & s[ ..] )
1100
1130
. collect :: < Vec < & str > > ( ) ;
1101
- if !unknown. is_empty ( ) {
1102
- let features = unknown. join ( ", " ) ;
1103
- bail ! ( "Package `{}` does not have these features: `{}`" ,
1104
- candidate. package_id( ) , features)
1105
- }
1131
+ let features = unknown. join ( ", " ) ;
1132
+ bail ! ( "Package `{}` does not have these features: `{}`" ,
1133
+ s. package_id( ) , features)
1106
1134
}
1107
1135
1108
1136
// Record what list of features is active for this package.
1109
1137
if !used_features. is_empty ( ) {
1110
- let pkgid = candidate . package_id ( ) ;
1138
+ let pkgid = s . package_id ( ) ;
1111
1139
1112
1140
let set = self . resolve_features . entry ( pkgid. clone ( ) )
1113
1141
. or_insert_with ( HashSet :: new) ;
0 commit comments