aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAlex Lam S.L <alexlamsl@gmail.com>2018-06-19 18:20:11 +0800
committerGitHub <noreply@github.com>2018-06-19 18:20:11 +0800
commite54ddcbb8a675940f77e05a5f04c3a208268b41a (patch)
tree4e879bd33c839320bab9e0ef58190e13b4f3d729
parent9e19e63551907be19c8c6b5b75217d7650504436 (diff)
downloadtracifyjs-e54ddcbb8a675940f77e05a5f04c3a208268b41a.tar.gz
tracifyjs-e54ddcbb8a675940f77e05a5f04c3a208268b41a.zip
fix corner cases in `properties` (#3189)
fixes #3188
-rw-r--r--lib/compress.js19
-rw-r--r--test/compress/properties.js103
2 files changed, 116 insertions, 6 deletions
diff --git a/lib/compress.js b/lib/compress.js
index ad01e48c..56c8fc47 100644
--- a/lib/compress.js
+++ b/lib/compress.js
@@ -6171,6 +6171,16 @@ merge(Compressor.prototype, {
});
});
+ function safe_to_flatten(value, compressor) {
+ if (value instanceof AST_SymbolRef) {
+ value = value.fixed_value();
+ }
+ if (!value) return false;
+ return !(value instanceof AST_Lambda)
+ || compressor.parent() instanceof AST_New
+ || !value.contains_this();
+ }
+
OPT(AST_Sub, function(self, compressor) {
var expr = self.expression;
var prop = self.property;
@@ -6210,7 +6220,8 @@ merge(Compressor.prototype, {
&& prop instanceof AST_Number && expr instanceof AST_Array) {
var index = prop.getValue();
var elements = expr.elements;
- if (index in elements) {
+ var retValue = elements[index];
+ if (safe_to_flatten(retValue, compressor)) {
var flatten = true;
var values = [];
for (var i = elements.length; --i > index;) {
@@ -6220,7 +6231,6 @@ merge(Compressor.prototype, {
if (flatten && value.has_side_effects(compressor)) flatten = false;
}
}
- var retValue = elements[index];
retValue = retValue instanceof AST_Hole ? make_node(AST_Undefined, retValue) : retValue;
if (!flatten) values.unshift(retValue);
while (--i >= 0) {
@@ -6296,10 +6306,7 @@ merge(Compressor.prototype, {
if (!all(props, function(prop) {
return prop instanceof AST_ObjectKeyVal;
})) break;
- var value = prop.value;
- if (value instanceof AST_Function
- && !(compressor.parent() instanceof AST_New)
- && value.contains_this()) break;
+ if (!safe_to_flatten(prop.value, compressor)) break;
return make_node(AST_Sub, this, {
expression: make_node(AST_Array, expr, {
elements: props.map(function(prop) {
diff --git a/test/compress/properties.js b/test/compress/properties.js
index efe0d491..d36ae074 100644
--- a/test/compress/properties.js
+++ b/test/compress/properties.js
@@ -1729,3 +1729,106 @@ issue_869_2: {
}
expect_stdout: "PASS"
}
+
+issue_3188_1: {
+ options = {
+ collapse_vars: true,
+ inline: true,
+ properties: true,
+ reduce_vars: true,
+ side_effects: true,
+ }
+ input: {
+ (function() {
+ function f() {
+ console.log(this.p);
+ }
+ (function() {
+ var o = {
+ p: "PASS",
+ f: f
+ };
+ o.f();
+ })();
+ })();
+ }
+ expect: {
+ (function() {
+ function f() {
+ console.log(this.p);
+ }
+ ({
+ p: "PASS",
+ f: f
+ }).f();
+ var o;
+ })();
+ }
+ expect_stdout: "PASS"
+}
+
+issue_3188_2: {
+ options = {
+ collapse_vars: true,
+ inline: true,
+ properties: true,
+ reduce_vars: true,
+ side_effects: true,
+ unused: true,
+ }
+ input: {
+ (function() {
+ var f = function() {
+ console.log(this.p);
+ };
+ function g() {
+ var o = {
+ p: "PASS",
+ f: f
+ };
+ o.f();
+ }
+ g();
+ })();
+ }
+ expect: {
+ ({
+ p: "PASS",
+ f: function() {
+ console.log(this.p);
+ }
+ }).f();
+ }
+ expect_stdout: "PASS"
+}
+
+issue_3188_3: {
+ options = {
+ collapse_vars: true,
+ inline: true,
+ properties: true,
+ reduce_vars: true,
+ side_effects: true,
+ }
+ input: {
+ (function() {
+ function f() {
+ console.log(this[0]);
+ }
+ (function() {
+ var o = ["PASS", f];
+ o[1]();
+ })();
+ })();
+ }
+ expect: {
+ (function() {
+ function f() {
+ console.log(this[0]);
+ }
+ ["PASS", f][1]();
+ var o;
+ })();
+ }
+ expect_stdout: "PASS"
+}