1
1
//! Checks that all Flunt files have messages in alphabetical order
2
2
3
3
use crate :: walk:: { filter_dirs, walk} ;
4
+ use std:: collections:: HashMap ;
4
5
use std:: { fs:: OpenOptions , io:: Write , path:: Path } ;
5
6
6
7
use regex:: Regex ;
@@ -13,11 +14,27 @@ fn filter_fluent(path: &Path) -> bool {
13
14
if let Some ( ext) = path. extension ( ) { ext. to_str ( ) != Some ( "ftl" ) } else { true }
14
15
}
15
16
16
- fn check_alphabetic ( filename : & str , fluent : & str , bad : & mut bool ) {
17
+ fn check_alphabetic (
18
+ filename : & str ,
19
+ fluent : & str ,
20
+ bad : & mut bool ,
21
+ all_defined_msgs : & mut HashMap < String , String > ,
22
+ ) {
17
23
let mut matches = MESSAGE . captures_iter ( fluent) . peekable ( ) ;
18
24
while let Some ( m) = matches. next ( ) {
25
+ let name = m. get ( 1 ) . unwrap ( ) ;
26
+ if let Some ( defined_filename) = all_defined_msgs. get ( name. as_str ( ) ) {
27
+ tidy_error ! (
28
+ bad,
29
+ "{filename}: message `{}` is already defined in {}" ,
30
+ name. as_str( ) ,
31
+ defined_filename,
32
+ ) ;
33
+ }
34
+
35
+ all_defined_msgs. insert ( name. as_str ( ) . to_owned ( ) , filename. to_owned ( ) ) ;
36
+
19
37
if let Some ( next) = matches. peek ( ) {
20
- let name = m. get ( 1 ) . unwrap ( ) ;
21
38
let next = next. get ( 1 ) . unwrap ( ) ;
22
39
if name. as_str ( ) > next. as_str ( ) {
23
40
tidy_error ! (
@@ -34,13 +51,29 @@ run `./x.py test tidy --bless` to sort the file correctly",
34
51
}
35
52
}
36
53
37
- fn sort_messages ( fluent : & str ) -> String {
54
+ fn sort_messages (
55
+ filename : & str ,
56
+ fluent : & str ,
57
+ bad : & mut bool ,
58
+ all_defined_msgs : & mut HashMap < String , String > ,
59
+ ) -> String {
38
60
let mut chunks = vec ! [ ] ;
39
61
let mut cur = String :: new ( ) ;
40
62
for line in fluent. lines ( ) {
41
- if MESSAGE . is_match ( line) {
63
+ if let Some ( name) = MESSAGE . find ( line) {
64
+ if let Some ( defined_filename) = all_defined_msgs. get ( name. as_str ( ) ) {
65
+ tidy_error ! (
66
+ bad,
67
+ "{filename}: message `{}` is already defined in {}" ,
68
+ name. as_str( ) ,
69
+ defined_filename,
70
+ ) ;
71
+ }
72
+
73
+ all_defined_msgs. insert ( name. as_str ( ) . to_owned ( ) , filename. to_owned ( ) ) ;
42
74
chunks. push ( std:: mem:: take ( & mut cur) ) ;
43
75
}
76
+
44
77
cur += line;
45
78
cur. push ( '\n' ) ;
46
79
}
@@ -53,20 +86,33 @@ fn sort_messages(fluent: &str) -> String {
53
86
}
54
87
55
88
pub fn check ( path : & Path , bless : bool , bad : & mut bool ) {
89
+ let mut all_defined_msgs = HashMap :: new ( ) ;
56
90
walk (
57
91
path,
58
92
|path, is_dir| filter_dirs ( path) || ( !is_dir && filter_fluent ( path) ) ,
59
93
& mut |ent, contents| {
60
94
if bless {
61
- let sorted = sort_messages ( contents) ;
95
+ let sorted = sort_messages (
96
+ ent. path ( ) . to_str ( ) . unwrap ( ) ,
97
+ contents,
98
+ bad,
99
+ & mut all_defined_msgs,
100
+ ) ;
62
101
if sorted != contents {
63
102
let mut f =
64
103
OpenOptions :: new ( ) . write ( true ) . truncate ( true ) . open ( ent. path ( ) ) . unwrap ( ) ;
65
104
f. write ( sorted. as_bytes ( ) ) . unwrap ( ) ;
66
105
}
67
106
} else {
68
- check_alphabetic ( ent. path ( ) . to_str ( ) . unwrap ( ) , contents, bad) ;
107
+ check_alphabetic (
108
+ ent. path ( ) . to_str ( ) . unwrap ( ) ,
109
+ contents,
110
+ bad,
111
+ & mut all_defined_msgs,
112
+ ) ;
69
113
}
70
114
} ,
71
115
) ;
116
+
117
+ crate :: fluent_used:: check ( path, all_defined_msgs, bad) ;
72
118
}
0 commit comments