support optional chaining operator (#4899)
This commit is contained in:
196
test/compress/optional-chains.js
Normal file
196
test/compress/optional-chains.js
Normal file
@@ -0,0 +1,196 @@
|
||||
call: {
|
||||
input: {
|
||||
console.log?.(undefined?.(console.log("FAIL")));
|
||||
}
|
||||
expect_exact: 'console.log?.((void 0)?.(console.log("FAIL")));'
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
dot: {
|
||||
input: {
|
||||
console?.log((void 0)?.p);
|
||||
}
|
||||
expect_exact: "console?.log((void 0)?.p);"
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
dot_in: {
|
||||
input: {
|
||||
var o = { in: 42 };
|
||||
console.log(o.in, o?.in);
|
||||
}
|
||||
expect_exact: "var o={in:42};console.log(o.in,o?.in);"
|
||||
expect_stdout: "42 42"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
sub: {
|
||||
input: {
|
||||
console?.["log"](null?.[console.log("FAIL")]);
|
||||
}
|
||||
expect_exact: 'console?.["log"](null?.[console.log("FAIL")]);'
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
ternary_decimal: {
|
||||
input: {
|
||||
null ? .42 : console.log("PASS");
|
||||
}
|
||||
expect_exact: 'null?.42:console.log("PASS");'
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
collapse_vars_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
A = 42;
|
||||
a?.[42];
|
||||
console.log(typeof A);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
A = 42;
|
||||
a?.[42];
|
||||
console.log(typeof A);
|
||||
}
|
||||
expect_stdout: "number"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
collapse_vars_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
A = 42;
|
||||
a?.(42);
|
||||
console.log(typeof A);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
A = 42;
|
||||
a?.(42);
|
||||
console.log(typeof A);
|
||||
}
|
||||
expect_stdout: "number"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
properties: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
properties: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
console.log(a?.["FAIL"]);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
console.log(a?.FAIL);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
reduce_vars_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 1;
|
||||
null?.[a = 0];
|
||||
console.log(a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
null?.[a = 0];
|
||||
console.log(a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
reduce_vars_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 1;
|
||||
null?.(a = 0);
|
||||
console.log(a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
null?.(a = 0);
|
||||
console.log(a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
side_effects: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
a?.[a = "FAIL"];
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
a?.[a = "FAIL"];
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
trim_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
optional_chains: true,
|
||||
reduce_vars: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
(function(a, b) {
|
||||
console?.log?.(a?.p, b?.[console.log("FAIL")]);
|
||||
})?.({ p: "PASS" });
|
||||
}
|
||||
expect: {
|
||||
(function(a, b) {
|
||||
console?.log?.(a.p, void 0);
|
||||
})({ p: "PASS" });
|
||||
}
|
||||
expect_stdout: "PASS undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
trim_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
optional_chains: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(void console.log("PASS"))?.[console.log("FAIL")];
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
@@ -1 +1 @@
|
||||
++null
|
||||
console.log(4 || (null = 4));
|
||||
|
||||
1
test/input/invalid/assign_5.js
Normal file
1
test/input/invalid/assign_5.js
Normal file
@@ -0,0 +1 @@
|
||||
console.log(5 || ([]?.length ^= 5));
|
||||
@@ -427,16 +427,30 @@ describe("bin/uglifyjs", function() {
|
||||
done();
|
||||
});
|
||||
});
|
||||
it("Should throw syntax error (++null)", function(done) {
|
||||
it("Should throw syntax error (null = 4)", function(done) {
|
||||
var command = uglifyjscmd + " test/input/invalid/assign_4.js";
|
||||
exec(command, function(err, stdout, stderr) {
|
||||
assert.ok(err);
|
||||
assert.strictEqual(stdout, "");
|
||||
assert.strictEqual(stderr.split(/\n/).slice(0, 4).join("\n"), [
|
||||
"Parse error at test/input/invalid/assign_4.js:1,0",
|
||||
"++null",
|
||||
"^",
|
||||
"ERROR: Invalid use of ++ operator",
|
||||
"Parse error at test/input/invalid/assign_4.js:1,23",
|
||||
"console.log(4 || (null = 4));",
|
||||
" ^",
|
||||
"ERROR: Invalid assignment",
|
||||
].join("\n"));
|
||||
done();
|
||||
});
|
||||
});
|
||||
it("Should throw syntax error ([]?.length ^= 5)", function(done) {
|
||||
var command = uglifyjscmd + " test/input/invalid/assign_5.js";
|
||||
exec(command, function(err, stdout, stderr) {
|
||||
assert.ok(err);
|
||||
assert.strictEqual(stdout, "");
|
||||
assert.strictEqual(stderr.split(/\n/).slice(0, 4).join("\n"), [
|
||||
"Parse error at test/input/invalid/assign_5.js:1,29",
|
||||
"console.log(5 || ([]?.length ^= 5));",
|
||||
" ^",
|
||||
"ERROR: Invalid assignment",
|
||||
].join("\n"));
|
||||
done();
|
||||
});
|
||||
|
||||
@@ -11,7 +11,7 @@ minify_in_situ() {
|
||||
do
|
||||
echo "$i"
|
||||
CODE=`cat "$i"`
|
||||
node_modules/.bin/esbuild --loader=ts --target=es2019 > "$i" <<EOF
|
||||
node_modules/.bin/esbuild --loader=ts --target=esnext > "$i" <<EOF
|
||||
$CODE
|
||||
EOF
|
||||
ARGS="$ARGS $i"
|
||||
|
||||
@@ -11,7 +11,7 @@ minify_in_situ() {
|
||||
do
|
||||
echo "$i"
|
||||
CODE=`cat "$i"`
|
||||
node_modules/.bin/esbuild --loader=ts --target=es2019 > "$i" <<EOF
|
||||
node_modules/.bin/esbuild --loader=ts --target=esnext > "$i" <<EOF
|
||||
$CODE
|
||||
EOF
|
||||
ARGS="$ARGS $i"
|
||||
|
||||
@@ -152,6 +152,7 @@ var SUPPORT = function(matrix) {
|
||||
logical_assignment: "[].p ??= 0;",
|
||||
new_target: "function f() { new.target; }",
|
||||
nullish: "0 ?? 0",
|
||||
optional_chaining: "0?.p",
|
||||
rest: "var [...a] = [];",
|
||||
rest_object: "var {...a} = {};",
|
||||
spread: "[...[]];",
|
||||
@@ -1487,12 +1488,16 @@ function _createExpression(recurmax, noComma, stmtDepth, canThrow) {
|
||||
return createValue() + " in " + createObjectLiteral(recurmax, stmtDepth, canThrow);
|
||||
case p++:
|
||||
var name = getVarName();
|
||||
var s = name + "[" + createExpression(recurmax, COMMA_OK, stmtDepth, canThrow) + "]";
|
||||
return canThrow && rng(20) == 0 ? s : name + " && " + s;
|
||||
var prop = "[" + createExpression(recurmax, COMMA_OK, stmtDepth, canThrow) + "]";
|
||||
if (SUPPORT.optional_chaining && rng(50) == 0) return name + "?." + prop;
|
||||
if (canThrow && rng(20) == 0) return name + prop;
|
||||
return name + " && " + name + prop;
|
||||
case p++:
|
||||
var name = getVarName();
|
||||
var s = name + "." + getDotKey();
|
||||
return canThrow && rng(20) == 0 ? s : name + " && " + s;
|
||||
var prop = getDotKey();
|
||||
if (SUPPORT.optional_chaining && rng(50) == 0) return name + "?." + prop;
|
||||
if (canThrow && rng(20) == 0) return name + "." + prop;
|
||||
return name + " && " + name + "." + prop;
|
||||
case p++:
|
||||
case p++:
|
||||
var name = getVarName();
|
||||
@@ -1534,7 +1539,16 @@ function _createExpression(recurmax, noComma, stmtDepth, canThrow) {
|
||||
name = rng(3) == 0 ? getVarName() : "f" + rng(funcs + 2);
|
||||
} while (name in called && !called[name]);
|
||||
called[name] = true;
|
||||
return mayDefer("typeof " + name + ' == "function" && --_calls_ >= 0 && ' + name + createArgs(recurmax, stmtDepth, canThrow));
|
||||
var args = createArgs(recurmax, stmtDepth, canThrow);
|
||||
var call = "typeof " + name + ' == "function" && --_calls_ >= 0 && ' + name + args;
|
||||
if (canThrow) {
|
||||
if (SUPPORT.optional_chaining && args[0] != "`" && rng(50) == 0) {
|
||||
call = name + "?." + args;
|
||||
} else if (rng(20) == 0) {
|
||||
call = name + args;
|
||||
}
|
||||
}
|
||||
return mayDefer(call);
|
||||
}
|
||||
_createExpression.N = p;
|
||||
return _createExpression(recurmax, noComma, stmtDepth, canThrow);
|
||||
|
||||
Reference in New Issue
Block a user