Implement logical assignment operators (#4834)
JerryScript-DCO-1.0-Signed-off-by: Robert Fancsik robert.fancsik@h-lab.eu
This commit is contained in:
@@ -0,0 +1,216 @@
|
||||
// Copyright JS Foundation and other contributors, http://js.foundation
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
function testLogicalOr(truish, falsish, value) {
|
||||
let a = truish;
|
||||
a ||= value;
|
||||
expectSame(a, truish);
|
||||
|
||||
a = falsish;
|
||||
a ||= value;
|
||||
expectSame(a, value);
|
||||
|
||||
a = {
|
||||
a: truish
|
||||
};
|
||||
a.a ||= value
|
||||
expectSame(a.a, truish);
|
||||
|
||||
a = {
|
||||
a: falsish
|
||||
};
|
||||
a.a ||= value;
|
||||
expectSame(a.a, value);
|
||||
|
||||
let called = false;
|
||||
let setter = function (_) {
|
||||
called = true
|
||||
};
|
||||
a = Object.defineProperty({}, "a", {
|
||||
set: setter
|
||||
});
|
||||
let obj = {
|
||||
a: value
|
||||
}
|
||||
a ||= obj;
|
||||
expectSame(called, false);
|
||||
expectSame(Object.getOwnPropertyDescriptor(a, "a").set, setter);
|
||||
|
||||
a = falsish;
|
||||
a ||= function () {};
|
||||
expectSame(typeof a, 'function');
|
||||
expectSame(a.name, 'a');
|
||||
|
||||
a = falsish;
|
||||
a ||= class {};
|
||||
expectSame(typeof a, 'function');
|
||||
expectSame(a.name, 'a');
|
||||
|
||||
const b = truish;
|
||||
b ||= value;
|
||||
expectSame(b, truish);
|
||||
|
||||
expectSame(this.b, undefined);
|
||||
this.b ||= value;
|
||||
expectSame(this.b, value);
|
||||
}
|
||||
|
||||
function testLogicalAnd(truish, falsish, value) {
|
||||
let a = falsish;
|
||||
a &&= value;
|
||||
expectSame(a, falsish);
|
||||
|
||||
a = truish;
|
||||
a &&= value;
|
||||
expectSame(a, value);
|
||||
|
||||
a = {
|
||||
a: falsish
|
||||
};
|
||||
a.a &&= value
|
||||
expectSame(a.a, falsish);
|
||||
|
||||
a = {
|
||||
a: truish
|
||||
};
|
||||
a.a &&= value;
|
||||
expectSame(a.a, value);
|
||||
|
||||
let called = false;
|
||||
let setter = function (_) {
|
||||
called = true
|
||||
};
|
||||
a = Object.defineProperty({}, "a", {
|
||||
set: setter
|
||||
});
|
||||
let obj = {
|
||||
a: value
|
||||
};
|
||||
a &&= obj;
|
||||
expectSame(called, false);
|
||||
expectSame(Object.getOwnPropertyDescriptor(a, "a").value, obj.a);
|
||||
|
||||
a = truish;
|
||||
a &&= function () {};
|
||||
expectSame(typeof a, 'function');
|
||||
expectSame(a.name, 'a');
|
||||
|
||||
a = truish;
|
||||
a &&= class {};
|
||||
expectSame(typeof a, 'function');
|
||||
expectSame(a.name, 'a');
|
||||
|
||||
const b = falsish;
|
||||
b &&= value;
|
||||
expectSame(b, falsish);
|
||||
|
||||
expectSame(this.b, undefined);
|
||||
this.b &&= value;
|
||||
expectSame(this.b, undefined);
|
||||
}
|
||||
|
||||
function testNullish(truish, falsish, value) {
|
||||
let a = falsish;
|
||||
a ??= value;
|
||||
expectSame(a, value);
|
||||
|
||||
a = truish;
|
||||
a ??= value;
|
||||
expectSame(a, truish);
|
||||
|
||||
a = {
|
||||
a: falsish
|
||||
};
|
||||
a.a ??= value
|
||||
expectSame(a.a, value);
|
||||
|
||||
a = {
|
||||
a: truish
|
||||
};
|
||||
a.a ??= value;
|
||||
expectSame(a.a, truish);
|
||||
|
||||
let called = false;
|
||||
let setter = function (_) {
|
||||
called = true
|
||||
};
|
||||
a = Object.defineProperty({}, "a", {
|
||||
set: setter
|
||||
});
|
||||
let obj = {
|
||||
a: value
|
||||
};
|
||||
a ??= obj;
|
||||
expectSame(called, false);
|
||||
expectSame(Object.getOwnPropertyDescriptor(a, "a").set, setter);
|
||||
|
||||
a = falsish;
|
||||
a ??= function () {};
|
||||
expectSame(typeof a, 'function');
|
||||
expectSame(a.name, 'a');
|
||||
|
||||
a = falsish;
|
||||
a ??= class {};
|
||||
expectSame(typeof a, 'function');
|
||||
expectSame(a.name, 'a');
|
||||
|
||||
const b = truish;
|
||||
b ??= value;
|
||||
expectSame(b, truish);
|
||||
|
||||
expectSame(this.b, undefined);
|
||||
this.b ??= value;
|
||||
expectSame(this.b, value);
|
||||
}
|
||||
|
||||
const nullishValues = [undefined, null];
|
||||
const logicalTrueValues = [1.1, Infinity, 1n, true, "foo", {}, Symbol()];
|
||||
const logicalFalseValues = [0, 0.0, NaN, 0n, false, "", ...nullishValues];
|
||||
const allValues = [...logicalTrueValues, ...logicalFalseValues];
|
||||
|
||||
function testValues(cb, falseValues) {
|
||||
for (const truish of logicalTrueValues) {
|
||||
for (const falsish of falseValues) {
|
||||
for (const value of allValues) {
|
||||
cb.call({}, truish, falsish, value);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function expectSame(x, value) {
|
||||
if (typeof x === 'number' && isNaN(x)) {
|
||||
assert(typeof value === 'number');
|
||||
assert(isNaN(value));
|
||||
} else {
|
||||
assert(x === value);
|
||||
}
|
||||
}
|
||||
|
||||
testValues(testLogicalOr, logicalFalseValues);
|
||||
testValues(testLogicalAnd, logicalFalseValues);
|
||||
testValues(testNullish, nullishValues);
|
||||
|
||||
function expectTypeError(str) {
|
||||
try {
|
||||
eval(str);
|
||||
assert(false);
|
||||
} catch (e) {
|
||||
assert(e instanceof TypeError);
|
||||
}
|
||||
}
|
||||
|
||||
expectTypeError("const c = 0; c ||= 8");
|
||||
expectTypeError("const c = 1; c &&= 8");
|
||||
expectTypeError("const c = undefined; c ??= 8");
|
||||
@@ -1821,57 +1821,6 @@
|
||||
<test id="built-ins/WeakRef/prototype/deref/this-does-not-have-internal-target-throws.js"><reason></reason></test>
|
||||
<!-- END - ESNext stage 4 proposal: WeakRefs -->
|
||||
|
||||
<!-- ESNext stage 4 proposal: Logical Assignment Operators
|
||||
features: [logical-assignment-operators]
|
||||
https://github.com/tc39/proposal-logical-assignment
|
||||
-->
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-bigint.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-lhs-before-rhs.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-namedevaluation-arrow-function.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-namedevaluation-class-expression.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-namedevaluation-function.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-no-set-put.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-no-set.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-non-extensible.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-non-writeable-put.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-non-writeable.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-unresolved-lhs.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-unresolved-rhs-put.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator-unresolved-rhs.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-assignment-operator.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-and-whitespace.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-bigint.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-lhs-before-rhs.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-namedevaluation-arrow-function.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-namedevaluation-class-expression.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-namedevaluation-function.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-no-set-put.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-no-set.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-non-extensible.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-non-writeable-put.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-non-writeable.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-unresolved-lhs.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-unresolved-rhs-put.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator-unresolved-rhs.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-assignment-operator.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-nullish-whitespace.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-bigint.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-lhs-before-rhs.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-namedevaluation-arrow-function.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-namedevaluation-class-expression.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-namedevaluation-function.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-no-set-put.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-no-set.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-non-extensible.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-non-writeable-put.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-non-writeable.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-unresolved-lhs.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-unresolved-rhs-put.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator-unresolved-rhs.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-assignment-operator.js"><reason></reason></test>
|
||||
<test id="language/expressions/logical-assignment/lgcl-or-whitespace.js"><reason></reason></test>
|
||||
<!-- END - ESNext stage 4 proposal: Logical Assignment Operators -->
|
||||
|
||||
<!-- ES2015: Proper Tail Call (PTC) Optimization
|
||||
features: [tail-call-optimization]
|
||||
-->
|
||||
|
||||
Reference in New Issue
Block a user