1
1
// Take a look at the license at the top of the repository in the LICENSE file.
2
2
3
+ use std:: { fmt, io} ;
4
+
3
5
use js:: token:: { self , Keyword , ReservedChar , Token , Tokens } ;
4
6
use js:: utils:: { get_array, get_variable_name_and_value_positions, VariableNameGenerator } ;
5
7
@@ -178,14 +180,26 @@ fn build_ast<'a>(v: &[token::Token<'a>]) -> Result<Elem<'a>, String> {
178
180
/// func(data[i]);
179
181
/// }
180
182
/// }"#.into();
181
- /// let js_minified = minify(js);
183
+ /// let js_minified = minify(js).to_string() ;
182
184
/// }
183
185
/// ```
184
186
#[ inline]
185
- pub fn minify ( source : & str ) -> String {
186
- token:: tokenize ( source)
187
- . apply ( :: js:: clean_tokens)
188
- . to_string ( )
187
+ pub fn minify < ' a > ( source : & ' a str ) -> Minified < ' a > {
188
+ Minified ( token:: tokenize ( source) . apply ( :: js:: clean_tokens) )
189
+ }
190
+
191
+ pub struct Minified < ' a > ( token:: Tokens < ' a > ) ;
192
+
193
+ impl < ' a > Minified < ' a > {
194
+ pub fn write < W : io:: Write > ( self , w : W ) -> io:: Result < ( ) > {
195
+ self . 0 . write ( w)
196
+ }
197
+ }
198
+
199
+ impl < ' a > fmt:: Display for Minified < ' a > {
200
+ fn fmt ( & self , f : & mut fmt:: Formatter < ' _ > ) -> fmt:: Result {
201
+ self . 0 . fmt ( f)
202
+ }
189
203
}
190
204
191
205
// TODO: No scope handling or anything. Might be nice as a second step to add it...
@@ -219,7 +233,6 @@ fn get_variables_name<'a>(
219
233
( ret, variables)
220
234
}
221
235
222
- #[ inline]
223
236
fn aggregate_strings_inner < ' a , ' b : ' a > (
224
237
mut tokens : Tokens < ' a > ,
225
238
separation_token : Option < Token < ' b > > ,
@@ -386,7 +399,6 @@ pub fn aggregate_strings_with_separation<'a, 'b: 'a>(
386
399
aggregate_strings_inner ( tokens, Some ( separation_token) )
387
400
}
388
401
389
- #[ inline]
390
402
fn aggregate_strings_into_array_inner < ' a , ' b : ' a , T : Fn ( & Tokens < ' a > , usize ) -> bool > (
391
403
mut tokens : Tokens < ' a > ,
392
404
array_name : & str ,
@@ -906,7 +918,7 @@ fn name_generator() {
906
918
fn simple_quote ( ) {
907
919
let source = r#"var x = "\\";"# ;
908
920
let expected_result = r#"var x="\\""# ;
909
- assert_eq ! ( minify( source) , expected_result) ;
921
+ assert_eq ! ( minify( source) . to_string ( ) , expected_result) ;
910
922
}
911
923
912
924
#[ test]
@@ -942,7 +954,7 @@ far_away(another_var, 12);
942
954
let expected_result = "var foo=\" something\" ;var another_var=2348323;function far_away(x,y){\
943
955
var x2=x+4;return x*x2+y}far_away(another_var,12);far_away(another_var,\
944
956
12)";
945
- assert_eq ! ( minify( source) , expected_result) ;
957
+ assert_eq ! ( minify( source) . to_string ( ) , expected_result) ;
946
958
}
947
959
948
960
#[ test]
@@ -975,7 +987,7 @@ console.log('done!');
975
987
*
976
988
* right?
977
989
*/function forEach(data,func){for(var i=0;i<data.length;++i){func(data[i])}}forEach([0,1,2,3,4,5,6,7,8,9],function(x){console.log(x)});console.log('done!')"# ;
978
- assert_eq ! ( minify( source) , expected_result) ;
990
+ assert_eq ! ( minify( source) . to_string ( ) , expected_result) ;
979
991
}
980
992
981
993
#[ test]
@@ -992,7 +1004,7 @@ search_input.onchange = function(e) {
992
1004
"# ;
993
1005
let expected_result = "search_input.onchange=function(e){clearTimeout(searchTimeout);\
994
1006
setTimeout(search,0)}";
995
- assert_eq ! ( minify( source) , expected_result) ;
1007
+ assert_eq ! ( minify( source) . to_string ( ) , expected_result) ;
996
1008
}
997
1009
998
1010
#[ test]
@@ -1005,7 +1017,7 @@ for (var entry in results) {
1005
1017
}"# ;
1006
1018
let expected_result = "for(var entry in results){if(results.hasOwnProperty(entry)){\
1007
1019
ar.push(results[entry])}}";
1008
- assert_eq ! ( minify( source) , expected_result) ;
1020
+ assert_eq ! ( minify( source) . to_string ( ) , expected_result) ;
1009
1021
}
1010
1022
1011
1023
#[ test]
@@ -1015,37 +1027,37 @@ val = val.replace(/\_/g, "");
1015
1027
1016
1028
var valGenerics = extractGenerics(val);"# ;
1017
1029
let expected_result = "val=val.replace(/\\ _/g,\" \" );var valGenerics=extractGenerics(val)" ;
1018
- assert_eq ! ( minify( source) , expected_result) ;
1030
+ assert_eq ! ( minify( source) . to_string ( ) , expected_result) ;
1019
1031
}
1020
1032
1021
1033
#[ test]
1022
1034
fn keep_space ( ) {
1023
1035
let source = "return 12;return x;" ;
1024
1036
1025
1037
let expected_result = "return 12;return x" ;
1026
- assert_eq ! ( minify( source) , expected_result) ;
1027
-
1028
- assert_eq ! ( "t in e" , minify( "t in e" ) ) ;
1029
- assert_eq ! ( "t+1 in e" , minify( "t + 1 in e" ) ) ;
1030
- assert_eq ! ( "t-1 in e" , minify( "t - 1 in e" ) ) ;
1031
- assert_eq ! ( "'a'in e" , minify( "'a' in e" ) ) ;
1032
- assert_eq ! ( "/a/g in e" , minify( "/a/g in e" ) ) ;
1033
- assert_eq ! ( "/a/i in e" , minify( "/a/i in e" ) ) ;
1034
-
1035
- assert_eq ! ( "t instanceof e" , minify( "t instanceof e" ) ) ;
1036
- assert_eq ! ( "t+1 instanceof e" , minify( "t + 1 instanceof e" ) ) ;
1037
- assert_eq ! ( "t-1 instanceof e" , minify( "t - 1 instanceof e" ) ) ;
1038
- assert_eq ! ( "'a'instanceof e" , minify( "'a' instanceof e" ) ) ;
1039
- assert_eq ! ( "/a/g instanceof e" , minify( "/a/g instanceof e" ) ) ;
1040
- assert_eq ! ( "/a/i instanceof e" , minify( "/a/i instanceof e" ) ) ;
1038
+ assert_eq ! ( minify( source) . to_string ( ) , expected_result) ;
1039
+
1040
+ assert_eq ! ( "t in e" , minify( "t in e" ) . to_string ( ) ) ;
1041
+ assert_eq ! ( "t+1 in e" , minify( "t + 1 in e" ) . to_string ( ) ) ;
1042
+ assert_eq ! ( "t-1 in e" , minify( "t - 1 in e" ) . to_string ( ) ) ;
1043
+ assert_eq ! ( "'a'in e" , minify( "'a' in e" ) . to_string ( ) ) ;
1044
+ assert_eq ! ( "/a/g in e" , minify( "/a/g in e" ) . to_string ( ) ) ;
1045
+ assert_eq ! ( "/a/i in e" , minify( "/a/i in e" ) . to_string ( ) ) ;
1046
+
1047
+ assert_eq ! ( "t instanceof e" , minify( "t instanceof e" ) . to_string ( ) ) ;
1048
+ assert_eq ! ( "t+1 instanceof e" , minify( "t + 1 instanceof e" ) . to_string ( ) ) ;
1049
+ assert_eq ! ( "t-1 instanceof e" , minify( "t - 1 instanceof e" ) . to_string ( ) ) ;
1050
+ assert_eq ! ( "'a'instanceof e" , minify( "'a' instanceof e" ) . to_string ( ) ) ;
1051
+ assert_eq ! ( "/a/g instanceof e" , minify( "/a/g instanceof e" ) . to_string ( ) ) ;
1052
+ assert_eq ! ( "/a/i instanceof e" , minify( "/a/i instanceof e" ) . to_string ( ) ) ;
1041
1053
}
1042
1054
1043
1055
#[ test]
1044
1056
fn test_remove_extra_whitespace_before_typeof ( ) {
1045
1057
let source = "var x = typeof 'foo';var y = typeof x;case typeof 'foo': 'bla'" ;
1046
1058
1047
1059
let expected_result = "var x=typeof'foo';var y=typeof x;case typeof'foo':'bla'" ;
1048
- assert_eq ! ( minify( source) , expected_result) ;
1060
+ assert_eq ! ( minify( source) . to_string ( ) , expected_result) ;
1049
1061
}
1050
1062
1051
1063
#[ test]
@@ -1055,29 +1067,29 @@ if (x in ev && typeof ev) { return true; }
1055
1067
if (true in ev) { return true; }"# ;
1056
1068
1057
1069
let expected_result = r#"if("key"in ev&&typeof ev){return true}if(x in ev&&typeof ev){return true}if(true in ev){return true}"# ;
1058
- assert_eq ! ( minify( source) , expected_result) ;
1070
+ assert_eq ! ( minify( source) . to_string ( ) , expected_result) ;
1059
1071
}
1060
1072
1061
1073
#[ test]
1062
1074
fn test_remove_extra_whitespace_before_operator ( ) {
1063
1075
let source = "( x ) / 2; x / y;x /= y" ;
1064
1076
1065
1077
let expected_result = "(x)/2;x/y;x/=y" ;
1066
- assert_eq ! ( minify( source) , expected_result) ;
1078
+ assert_eq ! ( minify( source) . to_string ( ) , expected_result) ;
1067
1079
}
1068
1080
1069
1081
#[ test]
1070
1082
fn check_regex_syntax ( ) {
1071
1083
let source = "console.log(/MSIE|Trident|Edge/.test(window.navigator.userAgent));" ;
1072
1084
let expected = "console.log(/MSIE|Trident|Edge/.test(window.navigator.userAgent))" ;
1073
- assert_eq ! ( minify( source) , expected) ;
1085
+ assert_eq ! ( minify( source) . to_string ( ) , expected) ;
1074
1086
}
1075
1087
1076
1088
#[ test]
1077
1089
fn minify_minified ( ) {
1078
1090
let source = "function (i, n, a) { i[n].type.replace(/ *;(.|\\ s)*/,\" \" )===t&&a.push(i[n].MathJax.elementJax);return a}" ;
1079
1091
let expected = "function(i,n,a){i[n].type.replace(/ *;(.|\\ s)*/,\" \" )===t&&a.push(i[n].MathJax.elementJax);return a}" ;
1080
- assert_eq ! ( minify( source) , expected) ;
1092
+ assert_eq ! ( minify( source) . to_string ( ) , expected) ;
1081
1093
}
1082
1094
1083
1095
#[ test]
@@ -1090,7 +1102,7 @@ fn check_string() {
1090
1102
"### ;
1091
1103
let expected = "const a=123;const b=\" 123\" ;const c=`the number is ${a} <-- note the spaces \
1092
1104
here`;const d=` ${a} ${b} `";
1093
- assert_eq ! ( minify( source) , expected) ;
1105
+ assert_eq ! ( minify( source) . to_string ( ) , expected) ;
1094
1106
}
1095
1107
1096
1108
// TODO: requires AST to fix this issue!
@@ -1102,7 +1114,7 @@ console.log(2)
1102
1114
var x = 12;
1103
1115
"#;
1104
1116
let expected_result = r#"console.log(1);console.log(2);var x=12;"#;
1105
- assert_eq!(minify(source), expected_result);
1117
+ assert_eq!(minify(source).to_string() , expected_result);
1106
1118
}*/
1107
1119
1108
1120
// TODO: requires AST to fix this issue!
@@ -1115,5 +1127,5 @@ function foo() {
1115
1127
}
1116
1128
"#;
1117
1129
let expected_result = r#"function foo(){return 12;}"#;
1118
- assert_eq!(minify(source), expected_result);
1130
+ assert_eq!(minify(source).to_string() , expected_result);
1119
1131
}*/
0 commit comments