fix corner cases in unsafe_math (#3532)

fixes #3531
This commit is contained in:
Alex Lam S.L
2019-10-27 08:25:11 +08:00
committed by GitHub
parent 37f35e4ac2
commit a270ba6b59
2 changed files with 259 additions and 73 deletions

View File

@@ -5930,7 +5930,8 @@ merge(Compressor.prototype, {
// a - -b => a + b // a - -b => a + b
if (self.right instanceof AST_UnaryPrefix if (self.right instanceof AST_UnaryPrefix
&& self.right.operator == "-" && self.right.operator == "-"
&& self.left.is_number(compressor)) { && self.left.is_number(compressor)
&& self.right.expression.is_number(compressor)) {
self = make_node(AST_Binary, self, { self = make_node(AST_Binary, self, {
operator: "+", operator: "+",
left: self.left, left: self.left,
@@ -5979,6 +5980,7 @@ merge(Compressor.prototype, {
// a + (b + c) => (a + b) + c // a + (b + c) => (a + b) + c
if (self.right instanceof AST_Binary if (self.right instanceof AST_Binary
&& self.right.operator != "%" && self.right.operator != "%"
&& self.right.is_number(compressor)
&& PRECEDENCE[self.right.operator] == PRECEDENCE[self.operator]) { && PRECEDENCE[self.right.operator] == PRECEDENCE[self.operator]) {
self = make_node(AST_Binary, self, { self = make_node(AST_Binary, self, {
operator: align(self.operator, self.right.operator), operator: align(self.operator, self.right.operator),
@@ -5991,6 +5993,14 @@ merge(Compressor.prototype, {
}), }),
right: self.right.right right: self.right.right
}); });
if (self.operator == "+"
&& !self.right.is_boolean(compressor)
&& !self.right.is_number(compressor)) {
self.right = make_node(AST_UnaryPrefix, self.right, {
operator: "+",
expression: self.right
});
}
} }
// (2 * n) * 3 => 6 * n // (2 * n) * 3 => 6 * n
// (n + 2) + 3 => n + 5 // (n + 2) + 3 => n + 5
@@ -5998,7 +6008,8 @@ merge(Compressor.prototype, {
&& self.left instanceof AST_Binary && self.left instanceof AST_Binary
&& self.left.operator != "%" && self.left.operator != "%"
&& PRECEDENCE[self.left.operator] == PRECEDENCE[self.operator]) { && PRECEDENCE[self.left.operator] == PRECEDENCE[self.operator]) {
if (self.left.left instanceof AST_Constant) { if (self.left.left instanceof AST_Constant
&& (self.left.operator != "+" || self.left.right.is_number(compressor))) {
self = make_node(AST_Binary, self, { self = make_node(AST_Binary, self, {
operator: self.left.operator, operator: self.left.operator,
left: make_node(AST_Binary, self.left, { left: make_node(AST_Binary, self.left, {
@@ -6010,7 +6021,8 @@ merge(Compressor.prototype, {
}), }),
right: self.left.right right: self.left.right
}); });
} else if (self.left.right instanceof AST_Constant) { } else if (self.left.right instanceof AST_Constant
&& (self.left.operator != "+" || self.left.left.is_number(compressor))) {
self = make_node(AST_Binary, self, { self = make_node(AST_Binary, self, {
operator: self.left.operator, operator: self.left.operator,
left: self.left.left, left: self.left.left,

View File

@@ -152,7 +152,7 @@ evaluate_2: {
} }
input: { input: {
var x = "42", y = null; var x = "42", y = null;
console.log( [
x + 1 + 2, x + 1 + 2,
x * 1 * 2, x * 1 * 2,
+x + 1 + 2, +x + 1 + 2,
@@ -164,12 +164,14 @@ evaluate_2: {
1 + (2 + ~x + 3), 1 + (2 + ~x + 3),
-y + (2 + ~x + 3), -y + (2 + ~x + 3),
1 & (2 & x & 3), 1 & (2 & x & 3),
1 + (2 + (x |= 0) + 3) 1 + (2 + (x |= 0) + 3),
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect: { expect: {
var x = "42", y = null; var x = "42", y = null;
console.log( [
x + 1 + 2, x + 1 + 2,
1 * x * 2, 1 * x * 2,
+x + 1 + 2, +x + 1 + 2,
@@ -181,10 +183,25 @@ evaluate_2: {
2 + ~x + 3 + 1, 2 + ~x + 3 + 1,
2 + ~x + 3 - y, 2 + ~x + 3 - y,
0 & x, 0 & x,
2 + (x |= 0) + 3 + 1 2 + (x |= 0) + 3 + 1,
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect_stdout: "4212 84 45 14223 43 48 6 47 -36 -37 0 47" expect_stdout: [
"string 4212",
"number 84",
"number 45",
"string 14223",
"number 43",
"number 48",
"number 6",
"number 47",
"number -36",
"number -37",
"number 0",
"number 47",
]
} }
evaluate_2_unsafe_math: { evaluate_2_unsafe_math: {
@@ -194,7 +211,7 @@ evaluate_2_unsafe_math: {
} }
input: { input: {
var x = "42", y = null; var x = "42", y = null;
console.log( [
x + 1 + 2, x + 1 + 2,
x * 1 * 2, x * 1 * 2,
+x + 1 + 2, +x + 1 + 2,
@@ -206,12 +223,14 @@ evaluate_2_unsafe_math: {
1 + (2 + ~x + 3), 1 + (2 + ~x + 3),
-y + (2 + ~x + 3), -y + (2 + ~x + 3),
1 & (2 & x & 3), 1 & (2 & x & 3),
1 + (2 + (x |= 0) + 3) 1 + (2 + (x |= 0) + 3),
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect: { expect: {
var x = "42", y = null; var x = "42", y = null;
console.log( [
x + 1 + 2, x + 1 + 2,
2 * x, 2 * x,
+x + 3, +x + 3,
@@ -223,10 +242,25 @@ evaluate_2_unsafe_math: {
6 + ~x, 6 + ~x,
5 + ~x - y, 5 + ~x - y,
0 & x, 0 & x,
6 + (x |= 0) 6 + (x |= 0),
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect_stdout: "4212 84 45 14223 43 48 6 47 -36 -37 0 47" expect_stdout: [
"string 4212",
"number 84",
"number 45",
"string 14223",
"number 43",
"number 48",
"number 6",
"number 47",
"number -36",
"number -37",
"number 0",
"number 47",
]
} }
evaluate_3: { evaluate_3: {
@@ -279,8 +313,8 @@ evaluate_5: {
unsafe_math: false, unsafe_math: false,
} }
input: { input: {
var a = true; var a = "1";
console.log( [
+a + 2 + 3, +a + 2 + 3,
+a + 2 - 3, +a + 2 - 3,
+a - 2 + 3, +a - 2 + 3,
@@ -292,12 +326,14 @@ evaluate_5: {
2 + 3 + +a, 2 + 3 + +a,
2 + 3 - +a, 2 + 3 - +a,
2 - 3 + +a, 2 - 3 + +a,
2 - 3 - +a 2 - 3 - +a,
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect: { expect: {
var a = true; var a = "1";
console.log( [
+a + 2 + 3, +a + 2 + 3,
+a + 2 - 3, +a + 2 - 3,
a - 2 + 3, a - 2 + 3,
@@ -309,10 +345,25 @@ evaluate_5: {
+a + 5, +a + 5,
5 - a, 5 - a,
+a - 1, +a - 1,
-1 - a -1 - a,
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect_stdout: "6 0 2 -4 6 0 4 -2 6 4 0 -2" expect_stdout: [
"number 6",
"number 0",
"number 2",
"number -4",
"number 6",
"number 0",
"number 4",
"number -2",
"number 6",
"number 4",
"number 0",
"number -2",
]
} }
evaluate_5_unsafe_math: { evaluate_5_unsafe_math: {
@@ -321,8 +372,8 @@ evaluate_5_unsafe_math: {
unsafe_math: true, unsafe_math: true,
} }
input: { input: {
var a = true; var a = "1";
console.log( [
+a + 2 + 3, +a + 2 + 3,
+a + 2 - 3, +a + 2 - 3,
+a - 2 + 3, +a - 2 + 3,
@@ -334,12 +385,14 @@ evaluate_5_unsafe_math: {
2 + 3 + +a, 2 + 3 + +a,
2 + 3 - +a, 2 + 3 - +a,
2 - 3 + +a, 2 - 3 + +a,
2 - 3 - +a 2 - 3 - +a,
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect: { expect: {
var a = true; var a = "1";
console.log( [
+a + 5, +a + 5,
+a + -1, +a + -1,
a - -1, a - -1,
@@ -351,10 +404,25 @@ evaluate_5_unsafe_math: {
+a + 5, +a + 5,
5 - a, 5 - a,
+a - 1, +a - 1,
-1 - a -1 - a,
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect_stdout: "6 0 2 -4 6 0 4 -2 6 4 0 -2" expect_stdout: [
"number 6",
"number 0",
"number 2",
"number -4",
"number 6",
"number 0",
"number 4",
"number -2",
"number 6",
"number 4",
"number 0",
"number -2",
]
} }
evaluate_6: { evaluate_6: {
@@ -363,8 +431,8 @@ evaluate_6: {
unsafe_math: false, unsafe_math: false,
} }
input: { input: {
var a = true; var a = "1";
console.log( [
-a + 2 + 3, -a + 2 + 3,
-a + 2 - 3, -a + 2 - 3,
-a - 2 + 3, -a - 2 + 3,
@@ -376,27 +444,44 @@ evaluate_6: {
2 + 3 + -a, 2 + 3 + -a,
2 + 3 - -a, 2 + 3 - -a,
2 - 3 + -a, 2 - 3 + -a,
2 - 3 - -a 2 - 3 - -a,
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect: { expect: {
var a = true; var a = "1";
console.log( [
2 - a + 3, 2 - a + 3,
2 - a - 3, 2 - a - 3,
-a - 2 + 3, -a - 2 + 3,
-a - 2 - 3, -a - 2 - 3,
2 - a + 3, 2 - a + 3,
2 - a - 3, 2 - a - 3,
2 + a + 3, 2 - -a + 3,
2 + a - 3, 2 - -a - 3,
5 - a, 5 - a,
5 + a, 5 - -a,
-1 - a, -1 - a,
-1 + a -1 - -a,
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect_stdout: "4 -2 0 -6 4 -2 6 0 4 6 -2 0" expect_stdout: [
"number 4",
"number -2",
"number 0",
"number -6",
"number 4",
"number -2",
"number 6",
"number 0",
"number 4",
"number 6",
"number -2",
"number 0",
]
} }
evaluate_6_unsafe_math: { evaluate_6_unsafe_math: {
@@ -405,8 +490,8 @@ evaluate_6_unsafe_math: {
unsafe_math: true, unsafe_math: true,
} }
input: { input: {
var a = true; var a = "1";
console.log( [
-a + 2 + 3, -a + 2 + 3,
-a + 2 - 3, -a + 2 - 3,
-a - 2 + 3, -a - 2 + 3,
@@ -418,27 +503,44 @@ evaluate_6_unsafe_math: {
2 + 3 + -a, 2 + 3 + -a,
2 + 3 - -a, 2 + 3 - -a,
2 - 3 + -a, 2 - 3 + -a,
2 - 3 - -a 2 - 3 - -a,
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect: { expect: {
var a = true; var a = "1";
console.log( [
5 - a, 5 - a,
-1 - a, -1 - a,
-a - -1, -a - -1,
-a - 5, -a - 5,
5 - a, 5 - a,
-1 - a, -1 - a,
2 + a + 3, 5 - -a,
-1 + a, -1 - -a,
5 - a, 5 - a,
5 + a, 5 - -a,
-1 - a, -1 - a,
-1 + a -1 - -a,
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect_stdout: "4 -2 0 -6 4 -2 6 0 4 6 -2 0" expect_stdout: [
"number 4",
"number -2",
"number 0",
"number -6",
"number 4",
"number -2",
"number 6",
"number 0",
"number 4",
"number 6",
"number -2",
"number 0",
]
} }
evaluate_7: { evaluate_7: {
@@ -448,7 +550,7 @@ evaluate_7: {
} }
input: { input: {
var x = "42", y; var x = "42", y;
console.log( [
+x + 2 + (3 + !y), +x + 2 + (3 + !y),
+x + 2 + (3 - !y), +x + 2 + (3 - !y),
+x + 2 - (3 + !y), +x + 2 - (3 + !y),
@@ -456,12 +558,14 @@ evaluate_7: {
+x - 2 + (3 + !y), +x - 2 + (3 + !y),
+x - 2 + (3 - !y), +x - 2 + (3 - !y),
+x - 2 - (3 + !y), +x - 2 - (3 + !y),
+x - 2 - (3 - !y) +x - 2 - (3 - !y),
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect: { expect: {
var x = "42", y; var x = "42", y;
console.log( [
+x + 2 + (3 + !y), +x + 2 + (3 + !y),
+x + 2 + (3 - !y), +x + 2 + (3 - !y),
+x + 2 - (3 + !y), +x + 2 - (3 + !y),
@@ -469,10 +573,21 @@ evaluate_7: {
x - 2 + (3 + !y), x - 2 + (3 + !y),
x - 2 + (3 - !y), x - 2 + (3 - !y),
x - 2 - (3 + !y), x - 2 - (3 + !y),
x - 2 - (3 - !y) x - 2 - (3 - !y),
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect_stdout: "48 46 40 42 44 42 36 38" expect_stdout: [
"number 48",
"number 46",
"number 40",
"number 42",
"number 44",
"number 42",
"number 36",
"number 38",
]
} }
evaluate_7_unsafe_math: { evaluate_7_unsafe_math: {
@@ -482,7 +597,7 @@ evaluate_7_unsafe_math: {
} }
input: { input: {
var x = "42", y; var x = "42", y;
console.log( [
+x + 2 + (3 + !y), +x + 2 + (3 + !y),
+x + 2 + (3 - !y), +x + 2 + (3 - !y),
+x + 2 - (3 + !y), +x + 2 - (3 + !y),
@@ -490,12 +605,14 @@ evaluate_7_unsafe_math: {
+x - 2 + (3 + !y), +x - 2 + (3 + !y),
+x - 2 + (3 - !y), +x - 2 + (3 - !y),
+x - 2 - (3 + !y), +x - 2 - (3 + !y),
+x - 2 - (3 - !y) +x - 2 - (3 - !y),
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect: { expect: {
var x = "42", y; var x = "42", y;
console.log( [
+x + 5 + !y, +x + 5 + !y,
+x + 5 - !y, +x + 5 - !y,
+x + -1 - !y, +x + -1 - !y,
@@ -503,10 +620,21 @@ evaluate_7_unsafe_math: {
x - -1 + !y, x - -1 + !y,
x - -1 - !y, x - -1 - !y,
x - 5 - !y, x - 5 - !y,
x - 5 + !y x - 5 + !y,
); ].forEach(function(n) {
console.log(typeof n, n);
});
} }
expect_stdout: "48 46 40 42 44 42 36 38" expect_stdout: [
"number 48",
"number 46",
"number 40",
"number 42",
"number 44",
"number 42",
"number 36",
"number 38",
]
} }
NaN_redefined: { NaN_redefined: {
@@ -573,3 +701,49 @@ unary_binary_parenthesis: {
} }
expect_stdout: true expect_stdout: true
} }
issue_3531_1: {
options = {
evaluate: true,
unsafe_math: true,
}
input: {
var a = "1";
console.log(typeof (a + 1 - .1 - .1 - .1));
}
expect: {
var a = "1";
console.log(typeof (a + 1 - (.2 + .1)));
}
expect_stdout: "number"
}
issue_3531_2: {
options = {
evaluate: true,
unsafe_math: true,
}
input: {
console.log(1 - (2 - {}));
}
expect: {
console.log(-1 + +{});
}
expect_stdout: "NaN"
}
issue_3531_3: {
options = {
evaluate: true,
unsafe_math: true,
}
input: {
var a = "3";
console.log(1 - (2 + a));
}
expect: {
var a = "3";
console.log(1 - (2 + a));
}
expect_stdout: "-22"
}