Skip to content

Commit 4aea02d

Browse files
JS: Avoid unnecessary stringification
Instead, allow the user to pass in a writer and write the tokens incrementally. Or they can still stringify as before if they want. Also remove some unneeded `#[inline]`.
1 parent f7fa9f3 commit 4aea02d

File tree

6 files changed

+64
-49
lines changed

6 files changed

+64
-49
lines changed

src/js/token.rs

+14-5
Original file line numberDiff line numberDiff line change
@@ -1006,8 +1006,17 @@ pub fn tokenize(source: &str) -> Tokens<'_> {
10061006
#[derive(Debug, PartialEq)]
10071007
pub struct Tokens<'a>(pub Vec<Token<'a>>);
10081008

1009+
impl<'a> Tokens<'a> {
1010+
pub(super) fn write<W: std::io::Write>(self, mut w: W) -> std::io::Result<()> {
1011+
for token in self.0.iter() {
1012+
write!(w, "{}", token)?;
1013+
}
1014+
Ok(())
1015+
}
1016+
}
1017+
10091018
impl<'a> fmt::Display for Tokens<'a> {
1010-
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
1019+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
10111020
let tokens = &self.0;
10121021
for i in 0..tokens.len() {
10131022
if i > 0
@@ -1096,7 +1105,7 @@ impl<'a> From<&[Token<'a>]> for Tokens<'a> {
10961105
fn check_regex() {
10971106
let source = r#"var x = /"\.x/g;"#;
10981107
let expected_result = r#"var x=/"\.x/g"#;
1099-
assert_eq!(::js::minify(source), expected_result);
1108+
assert_eq!(::js::minify(source).to_string(), expected_result);
11001109

11011110
let v = tokenize(source).apply(::js::clean_tokens);
11021111
assert_eq!(
@@ -1110,7 +1119,7 @@ fn check_regex() {
11101119

11111120
let source = r#"var x = /"\.x/gigigigig;var x = "hello";"#;
11121121
let expected_result = r#"var x=/"\.x/gi;var x="hello""#;
1113-
assert_eq!(::js::minify(source), expected_result);
1122+
assert_eq!(::js::minify(source).to_string(), expected_result);
11141123

11151124
let v = tokenize(source).apply(::js::clean_tokens);
11161125
assert_eq!(
@@ -1127,7 +1136,7 @@ fn check_regex() {
11271136
fn more_regex() {
11281137
let source = r#"var x = /"\.x\/a/i;"#;
11291138
let expected_result = r#"var x=/"\.x\/a/i"#;
1130-
assert_eq!(::js::minify(source), expected_result);
1139+
assert_eq!(::js::minify(source).to_string(), expected_result);
11311140

11321141
let v = tokenize(source).apply(::js::clean_tokens);
11331142
assert_eq!(
@@ -1141,7 +1150,7 @@ fn more_regex() {
11411150

11421151
let source = r#"var x = /\\/i;"#;
11431152
let expected_result = r#"var x=/\\/i"#;
1144-
assert_eq!(::js::minify(source), expected_result);
1153+
assert_eq!(::js::minify(source).to_string(), expected_result);
11451154

11461155
let v = tokenize(source).apply(::js::clean_tokens);
11471156
assert_eq!(

src/js/tools.rs

+48-36
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
// Take a look at the license at the top of the repository in the LICENSE file.
22

3+
use std::{fmt, io};
4+
35
use js::token::{self, Keyword, ReservedChar, Token, Tokens};
46
use js::utils::{get_array, get_variable_name_and_value_positions, VariableNameGenerator};
57

@@ -178,14 +180,26 @@ fn build_ast<'a>(v: &[token::Token<'a>]) -> Result<Elem<'a>, String> {
178180
/// func(data[i]);
179181
/// }
180182
/// }"#.into();
181-
/// let js_minified = minify(js);
183+
/// let js_minified = minify(js).to_string();
182184
/// }
183185
/// ```
184186
#[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+
}
189203
}
190204

191205
// 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>(
219233
(ret, variables)
220234
}
221235

222-
#[inline]
223236
fn aggregate_strings_inner<'a, 'b: 'a>(
224237
mut tokens: Tokens<'a>,
225238
separation_token: Option<Token<'b>>,
@@ -386,7 +399,6 @@ pub fn aggregate_strings_with_separation<'a, 'b: 'a>(
386399
aggregate_strings_inner(tokens, Some(separation_token))
387400
}
388401

389-
#[inline]
390402
fn aggregate_strings_into_array_inner<'a, 'b: 'a, T: Fn(&Tokens<'a>, usize) -> bool>(
391403
mut tokens: Tokens<'a>,
392404
array_name: &str,
@@ -906,7 +918,7 @@ fn name_generator() {
906918
fn simple_quote() {
907919
let source = r#"var x = "\\";"#;
908920
let expected_result = r#"var x="\\""#;
909-
assert_eq!(minify(source), expected_result);
921+
assert_eq!(minify(source).to_string(), expected_result);
910922
}
911923

912924
#[test]
@@ -942,7 +954,7 @@ far_away(another_var, 12);
942954
let expected_result = "var foo=\"something\";var another_var=2348323;function far_away(x,y){\
943955
var x2=x+4;return x*x2+y}far_away(another_var,12);far_away(another_var,\
944956
12)";
945-
assert_eq!(minify(source), expected_result);
957+
assert_eq!(minify(source).to_string(), expected_result);
946958
}
947959

948960
#[test]
@@ -975,7 +987,7 @@ console.log('done!');
975987
*
976988
* right?
977989
*/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);
979991
}
980992

981993
#[test]
@@ -992,7 +1004,7 @@ search_input.onchange = function(e) {
9921004
"#;
9931005
let expected_result = "search_input.onchange=function(e){clearTimeout(searchTimeout);\
9941006
setTimeout(search,0)}";
995-
assert_eq!(minify(source), expected_result);
1007+
assert_eq!(minify(source).to_string(), expected_result);
9961008
}
9971009

9981010
#[test]
@@ -1005,7 +1017,7 @@ for (var entry in results) {
10051017
}"#;
10061018
let expected_result = "for(var entry in results){if(results.hasOwnProperty(entry)){\
10071019
ar.push(results[entry])}}";
1008-
assert_eq!(minify(source), expected_result);
1020+
assert_eq!(minify(source).to_string(), expected_result);
10091021
}
10101022

10111023
#[test]
@@ -1015,37 +1027,37 @@ val = val.replace(/\_/g, "");
10151027
10161028
var valGenerics = extractGenerics(val);"#;
10171029
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);
10191031
}
10201032

10211033
#[test]
10221034
fn keep_space() {
10231035
let source = "return 12;return x;";
10241036

10251037
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());
10411053
}
10421054

10431055
#[test]
10441056
fn test_remove_extra_whitespace_before_typeof() {
10451057
let source = "var x = typeof 'foo';var y = typeof x;case typeof 'foo': 'bla'";
10461058

10471059
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);
10491061
}
10501062

10511063
#[test]
@@ -1055,29 +1067,29 @@ if (x in ev && typeof ev) { return true; }
10551067
if (true in ev) { return true; }"#;
10561068

10571069
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);
10591071
}
10601072

10611073
#[test]
10621074
fn test_remove_extra_whitespace_before_operator() {
10631075
let source = "( x ) / 2; x / y;x /= y";
10641076

10651077
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);
10671079
}
10681080

10691081
#[test]
10701082
fn check_regex_syntax() {
10711083
let source = "console.log(/MSIE|Trident|Edge/.test(window.navigator.userAgent));";
10721084
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);
10741086
}
10751087

10761088
#[test]
10771089
fn minify_minified() {
10781090
let source = "function (i, n, a) { i[n].type.replace(/ *;(.|\\s)*/,\"\")===t&&a.push(i[n].MathJax.elementJax);return a}";
10791091
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);
10811093
}
10821094

10831095
#[test]
@@ -1090,7 +1102,7 @@ fn check_string() {
10901102
"###;
10911103
let expected = "const a=123;const b=\"123\";const c=`the number is ${a} <-- note the spaces \
10921104
here`;const d=` ${a} ${b} `";
1093-
assert_eq!(minify(source), expected);
1105+
assert_eq!(minify(source).to_string(), expected);
10941106
}
10951107

10961108
// TODO: requires AST to fix this issue!
@@ -1102,7 +1114,7 @@ console.log(2)
11021114
var x = 12;
11031115
"#;
11041116
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);
11061118
}*/
11071119

11081120
// TODO: requires AST to fix this issue!
@@ -1115,5 +1127,5 @@ function foo() {
11151127
}
11161128
"#;
11171129
let expected_result = r#"function foo(){return 12;}"#;
1118-
assert_eq!(minify(source), expected_result);
1130+
assert_eq!(minify(source).to_string(), expected_result);
11191131
}*/

src/js/utils.rs

-5
Original file line numberDiff line numberDiff line change
@@ -174,7 +174,6 @@ pub fn replace_token_with<'a, 'b: 'a, F: Fn(&Token<'a>) -> Option<Token<'b>>>(
174174
/// assert_eq!(result, vec![(2, Some(6)), (10, None), (14, Some(22))]);
175175
/// }
176176
/// ```
177-
#[inline]
178177
pub fn get_variable_name_and_value_positions<'a>(
179178
tokens: &'a Tokens<'a>,
180179
pos: usize,
@@ -275,7 +274,6 @@ fn get_next<'a>(it: &mut IntoIter<Token<'a>>) -> Option<Token<'a>> {
275274
/// println!("result: {:?}", s);
276275
/// }
277276
/// ```
278-
#[inline]
279277
pub fn clean_tokens(tokens: Tokens<'_>) -> Tokens<'_> {
280278
let mut v = Vec::with_capacity(tokens.len() / 3 * 2);
281279
let mut it = tokens.0.into_iter();
@@ -307,7 +305,6 @@ pub fn clean_tokens(tokens: Tokens<'_>) -> Tokens<'_> {
307305

308306
/// Returns true if the token is a "useful" one (so not a comment or a "useless"
309307
/// character).
310-
#[inline]
311308
pub fn clean_token(token: &Token<'_>, next_token: &Option<&Token<'_>>) -> bool {
312309
!token.is_comment() && {
313310
if let Some(x) = token.get_char() {
@@ -358,7 +355,6 @@ fn get_next_except<'a, F: Fn(&Token<'a>) -> bool>(
358355
/// println!("result: {:?}", s);
359356
/// }
360357
/// ```
361-
#[inline]
362358
pub fn clean_tokens_except<'a, F: Fn(&Token<'a>) -> bool>(tokens: Tokens<'a>, f: F) -> Tokens<'a> {
363359
let mut v = Vec::with_capacity(tokens.len() / 3 * 2);
364360
let mut it = tokens.0.into_iter();
@@ -410,7 +406,6 @@ pub fn clean_token_except<'a, F: Fn(&Token<'a>) -> bool>(
410406
}
411407
}
412408

413-
#[inline]
414409
pub(crate) fn get_array<'a>(
415410
tokens: &'a Tokens<'a>,
416411
array_name: &str,

src/json/read/byte_to_char.rs

-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,6 @@ impl<R: Read + fmt::Debug> fmt::Debug for ByteToChar<R> {
3939
impl<R: Read> Iterator for ByteToChar<R> {
4040
type Item = Result<char, CharsError>;
4141

42-
#[inline]
4342
fn next(&mut self) -> Option<Result<char, CharsError>> {
4443
let first_byte = match self.get_next() {
4544
Err(err) => return Some(Err(err)),

src/main.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -85,7 +85,7 @@ fn main() {
8585
"css" => call_minifier(arg, |s| {
8686
css::minify(s).expect("css minification failed").to_string()
8787
}),
88-
"js" => call_minifier(arg, js::minify),
88+
"js" => call_minifier(arg, |s| js::minify(s).to_string()),
8989
"json" => call_minifier(arg, json::minify),
9090
// "html" | "htm" => call_minifier(arg, html::minify),
9191
x => println!("\"{}\": this format isn't supported", x),

tests/js_minify.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -57,5 +57,5 @@ fn test_minification() {
5757

5858
let source = include_str!("./files/main.js");
5959
let expected_result = include_str!("./files/minified_main.js");
60-
compare_strs(&minify(source), expected_result);
60+
compare_strs(&minify(source).to_string(), expected_result);
6161
}

0 commit comments

Comments
 (0)