@@ -718,9 +718,15 @@ impl Config {
718
718
let mut should_update = false ;
719
719
720
720
if let Some ( change) = change. user_config_change {
721
- if let Ok ( change) = toml:: from_str ( & change) {
721
+ if let Ok ( table) = toml:: from_str ( & change) {
722
+ validate_toml_table (
723
+ GlobalLocalConfigInput :: FIELDS ,
724
+ & table,
725
+ & mut String :: new ( ) ,
726
+ & mut toml_errors,
727
+ ) ;
722
728
config. user_config =
723
- Some ( GlobalLocalConfigInput :: from_toml ( change , & mut toml_errors) ) ;
729
+ Some ( GlobalLocalConfigInput :: from_toml ( table , & mut toml_errors) ) ;
724
730
should_update = true ;
725
731
}
726
732
}
@@ -748,21 +754,39 @@ impl Config {
748
754
}
749
755
750
756
if let Some ( change) = change. root_ratoml_change {
751
- if let Ok ( change) = toml:: from_str ( & change) {
752
- config. root_ratoml =
753
- Some ( GlobalLocalConfigInput :: from_toml ( change, & mut toml_errors) ) ;
754
- should_update = true ;
757
+ match toml:: from_str ( & change) {
758
+ Ok ( table) => {
759
+ validate_toml_table (
760
+ GlobalLocalConfigInput :: FIELDS ,
761
+ & table,
762
+ & mut String :: new ( ) ,
763
+ & mut toml_errors,
764
+ ) ;
765
+ config. root_ratoml =
766
+ Some ( GlobalLocalConfigInput :: from_toml ( table, & mut toml_errors) ) ;
767
+ should_update = true ;
768
+ }
769
+ Err ( e) => toml_errors. push ( ( "invalid toml file" . to_owned ( ) , e) ) ,
755
770
}
756
771
}
757
772
758
773
if let Some ( change) = change. ratoml_file_change {
759
774
for ( source_root_id, ( _, text) ) in change {
760
775
if let Some ( text) = text {
761
- if let Ok ( change) = toml:: from_str ( & text) {
762
- config. ratoml_files . insert (
763
- source_root_id,
764
- LocalConfigInput :: from_toml ( & change, & mut toml_errors) ,
765
- ) ;
776
+ match toml:: from_str ( & text) {
777
+ Ok ( table) => {
778
+ validate_toml_table (
779
+ & [ LocalConfigInput :: FIELDS ] ,
780
+ & table,
781
+ & mut String :: new ( ) ,
782
+ & mut toml_errors,
783
+ ) ;
784
+ config. ratoml_files . insert (
785
+ source_root_id,
786
+ LocalConfigInput :: from_toml ( & table, & mut toml_errors) ,
787
+ ) ;
788
+ }
789
+ Err ( e) => toml_errors. push ( ( "invalid toml file" . to_owned ( ) , e) ) ,
766
790
}
767
791
}
768
792
}
@@ -792,7 +816,7 @@ impl Config {
792
816
scope,
793
817
) {
794
818
Some ( snippet) => config. snippets . push ( snippet) ,
795
- None => error_sink. 0 . push ( ConfigErrorInner :: JsonError (
819
+ None => error_sink. 0 . push ( ConfigErrorInner :: Json (
796
820
format ! ( "snippet {name} is invalid" ) ,
797
821
<serde_json:: Error as serde:: de:: Error >:: custom (
798
822
"snippet path is invalid or triggers are missing" ,
@@ -801,8 +825,11 @@ impl Config {
801
825
}
802
826
}
803
827
828
+ error_sink. 0 . extend ( json_errors. into_iter ( ) . map ( |( a, b) | ConfigErrorInner :: Json ( a, b) ) ) ;
829
+ error_sink. 0 . extend ( toml_errors. into_iter ( ) . map ( |( a, b) | ConfigErrorInner :: Toml ( a, b) ) ) ;
830
+
804
831
if config. check_command ( ) . is_empty ( ) {
805
- error_sink. 0 . push ( ConfigErrorInner :: JsonError (
832
+ error_sink. 0 . push ( ConfigErrorInner :: Json (
806
833
"/check/command" . to_owned ( ) ,
807
834
serde_json:: Error :: custom ( "expected a non-empty string" ) ,
808
835
) ) ;
@@ -836,14 +863,9 @@ impl ConfigChange {
836
863
vfs_path : VfsPath ,
837
864
content : Option < String > ,
838
865
) -> Option < ( VfsPath , Option < String > ) > {
839
- if let Some ( changes) = self . ratoml_file_change . as_mut ( ) {
840
- changes. insert ( source_root, ( vfs_path, content) )
841
- } else {
842
- let mut map = FxHashMap :: default ( ) ;
843
- map. insert ( source_root, ( vfs_path, content) ) ;
844
- self . ratoml_file_change = Some ( map) ;
845
- None
846
- }
866
+ self . ratoml_file_change
867
+ . get_or_insert_with ( Default :: default)
868
+ . insert ( source_root, ( vfs_path, content) )
847
869
}
848
870
849
871
pub fn change_user_config ( & mut self , content : Option < String > ) {
@@ -1058,7 +1080,7 @@ pub struct ClientCommandsConfig {
1058
1080
1059
1081
#[ derive( Debug ) ]
1060
1082
pub enum ConfigErrorInner {
1061
- JsonError ( String , serde_json:: Error ) ,
1083
+ Json ( String , serde_json:: Error ) ,
1062
1084
Toml ( String , toml:: de:: Error ) ,
1063
1085
}
1064
1086
@@ -1074,7 +1096,7 @@ impl ConfigError {
1074
1096
impl fmt:: Display for ConfigError {
1075
1097
fn fmt ( & self , f : & mut fmt:: Formatter < ' _ > ) -> fmt:: Result {
1076
1098
let errors = self . 0 . iter ( ) . format_with ( "\n " , |inner, f| match inner {
1077
- ConfigErrorInner :: JsonError ( key, e) => {
1099
+ ConfigErrorInner :: Json ( key, e) => {
1078
1100
f ( key) ?;
1079
1101
f ( & ": " ) ?;
1080
1102
f ( e)
@@ -2607,6 +2629,8 @@ macro_rules! _config_data {
2607
2629
2608
2630
#[ allow( unused, clippy:: ptr_arg) ]
2609
2631
impl $input {
2632
+ const FIELDS : & ' static [ & ' static str ] = & [ $( stringify!( $field) ) ,* ] ;
2633
+
2610
2634
fn from_json( json: & mut serde_json:: Value , error_sink: & mut Vec <( String , serde_json:: Error ) >) -> Self {
2611
2635
Self { $(
2612
2636
$field: get_field(
@@ -2645,8 +2669,7 @@ macro_rules! _config_data {
2645
2669
mod $modname {
2646
2670
#[ test]
2647
2671
fn fields_are_sorted( ) {
2648
- let field_names: & ' static [ & ' static str ] = & [ $( stringify!( $field) ) ,* ] ;
2649
- field_names. windows( 2 ) . for_each( |w| assert!( w[ 0 ] <= w[ 1 ] , "{} <= {} does not hold" , w[ 0 ] , w[ 1 ] ) ) ;
2672
+ super :: $input:: FIELDS . windows( 2 ) . for_each( |w| assert!( w[ 0 ] <= w[ 1 ] , "{} <= {} does not hold" , w[ 0 ] , w[ 1 ] ) ) ;
2650
2673
}
2651
2674
}
2652
2675
} ;
@@ -2711,6 +2734,8 @@ struct GlobalLocalConfigInput {
2711
2734
}
2712
2735
2713
2736
impl GlobalLocalConfigInput {
2737
+ const FIELDS : & ' static [ & ' static [ & ' static str ] ] =
2738
+ & [ GlobalConfigInput :: FIELDS , LocalConfigInput :: FIELDS ] ;
2714
2739
fn from_toml (
2715
2740
toml : toml:: Table ,
2716
2741
error_sink : & mut Vec < ( String , toml:: de:: Error ) > ,
@@ -3148,6 +3173,35 @@ fn field_props(field: &str, ty: &str, doc: &[&str], default: &str) -> serde_json
3148
3173
map. into ( )
3149
3174
}
3150
3175
3176
+ fn validate_toml_table (
3177
+ known_ptrs : & [ & [ & ' static str ] ] ,
3178
+ toml : & toml:: Table ,
3179
+ ptr : & mut String ,
3180
+ error_sink : & mut Vec < ( String , toml:: de:: Error ) > ,
3181
+ ) {
3182
+ let verify = |ptr : & String | known_ptrs. iter ( ) . any ( |ptrs| ptrs. contains ( & ptr. as_str ( ) ) ) ;
3183
+
3184
+ let l = ptr. len ( ) ;
3185
+ for ( k, v) in toml {
3186
+ if !ptr. is_empty ( ) {
3187
+ ptr. push ( '_' ) ;
3188
+ }
3189
+ ptr. push_str ( k) ;
3190
+
3191
+ match v {
3192
+ // This is a table config, any entry in it is therefor valid
3193
+ toml:: Value :: Table ( _) if verify ( ptr) => ( ) ,
3194
+ toml:: Value :: Table ( table) => validate_toml_table ( known_ptrs, table, ptr, error_sink) ,
3195
+ _ if !verify ( ptr) => {
3196
+ error_sink. push ( ( ptr. clone ( ) , toml:: de:: Error :: custom ( "unexpected field" ) ) )
3197
+ }
3198
+ _ => ( ) ,
3199
+ }
3200
+
3201
+ ptr. truncate ( l) ;
3202
+ }
3203
+ }
3204
+
3151
3205
#[ cfg( test) ]
3152
3206
fn manual ( fields : & [ SchemaField ] ) -> String {
3153
3207
fields. iter ( ) . fold ( String :: new ( ) , |mut acc, ( field, _ty, doc, default) | {
@@ -3387,4 +3441,60 @@ mod tests {
3387
3441
matches!( config. flycheck( ) , FlycheckConfig :: CargoCommand { options, .. } if options. target_dir == Some ( Utf8PathBuf :: from( "other_folder" ) ) )
3388
3442
) ;
3389
3443
}
3444
+
3445
+ #[ test]
3446
+ fn toml_unknown_key ( ) {
3447
+ let config = Config :: new (
3448
+ AbsPathBuf :: try_from ( project_root ( ) ) . unwrap ( ) ,
3449
+ Default :: default ( ) ,
3450
+ vec ! [ ] ,
3451
+ None ,
3452
+ None ,
3453
+ ) ;
3454
+
3455
+ let mut change = ConfigChange :: default ( ) ;
3456
+
3457
+ change. change_root_ratoml ( Some (
3458
+ toml:: toml! {
3459
+ [ cargo. cfgs]
3460
+ these = "these"
3461
+ should = "should"
3462
+ be = "be"
3463
+ valid = "valid"
3464
+
3465
+ [ invalid. config]
3466
+ err = "error"
3467
+
3468
+ [ cargo]
3469
+ target = "ok"
3470
+
3471
+ // FIXME: This should be an error
3472
+ [ cargo. sysroot]
3473
+ non-table = "expected"
3474
+ }
3475
+ . to_string ( ) ,
3476
+ ) ) ;
3477
+
3478
+ let ( _, e, _) = config. apply_change ( change) ;
3479
+ expect_test:: expect![ [ r#"
3480
+ ConfigError(
3481
+ [
3482
+ Toml(
3483
+ "invalid_config_err",
3484
+ Error {
3485
+ inner: Error {
3486
+ inner: TomlError {
3487
+ message: "unexpected field",
3488
+ raw: None,
3489
+ keys: [],
3490
+ span: None,
3491
+ },
3492
+ },
3493
+ },
3494
+ ),
3495
+ ],
3496
+ )
3497
+ "# ] ]
3498
+ . assert_debug_eq ( & e) ;
3499
+ }
3390
3500
}
0 commit comments