| function shouldBe(actual, expected) { |
| if (actual !== expected) |
| throw new Error(`expected ${expected} but got ${actual}`); |
| } |
| |
| function shouldNotThrow(script) { |
| eval(script); |
| } |
| |
| function shouldThrowSyntaxError(script) { |
| let error; |
| try { |
| eval(script); |
| } catch (e) { |
| error = e; |
| } |
| |
| if (!(error instanceof SyntaxError)) |
| throw new Error('Expected SyntaxError!'); |
| } |
| |
| function testBasicCases() { |
| shouldBe(undefined ?? 3, 3); |
| shouldBe(null ?? 3, 3); |
| shouldBe(true ?? 3, true); |
| shouldBe(false ?? 3, false); |
| shouldBe(0 ?? 3, 0); |
| shouldBe(1 ?? 3, 1); |
| shouldBe('' ?? 3, ''); |
| shouldBe('hi' ?? 3, 'hi'); |
| shouldBe(({} ?? 3) instanceof Object, true); |
| shouldBe(({ x: 'hi' } ?? 3).x, 'hi'); |
| shouldBe(([] ?? 3) instanceof Array, true); |
| shouldBe((['hi'] ?? 3)[0], 'hi'); |
| shouldBe((makeMasquerader() ?? 3) == null, true); |
| } |
| noInline(testBasicCases); |
| |
| for (let i = 0; i < 1e5; i++) |
| testBasicCases(); |
| |
| shouldBe(1 | null ?? 3, 1); |
| shouldBe(1 ^ null ?? 3, 1); |
| shouldBe(1 & null ?? 3, 0); |
| shouldBe(3 == null ?? 3, false); |
| shouldBe(3 != null ?? 3, true); |
| shouldBe(3 === null ?? 3, false); |
| shouldBe(3 !== null ?? 3, true); |
| shouldBe(1 < null ?? 3, false); |
| shouldBe(1 > null ?? 3, true); |
| shouldBe(1 <= null ?? 3, false); |
| shouldBe(1 >= null ?? 3, true); |
| shouldBe(1 << null ?? 3, 1); |
| shouldBe(1 >> null ?? 3, 1); |
| shouldBe(1 >>> null ?? 3, 1); |
| shouldBe(1 + null ?? 3, 1); |
| shouldBe(1 - null ?? 3, 1); |
| shouldBe(1 * null ?? 3, 0); |
| shouldBe(1 / null ?? 3, Infinity); |
| shouldBe(isNaN(1 % null ?? 3), true); |
| shouldBe(1 ** null ?? 3, 1); |
| |
| const obj = { |
| count: 0, |
| get x() { this.count++; return 'x'; } |
| }; |
| false ?? obj.x; |
| shouldBe(obj.count, 0); |
| null ?? obj.x; |
| shouldBe(obj.count, 1); |
| obj.x ?? obj.x; |
| shouldBe(obj.count, 2); |
| |
| shouldThrowSyntaxError('0 || 1 ?? 2'); |
| shouldThrowSyntaxError('0 && 1 ?? 2'); |
| shouldThrowSyntaxError('0 ?? 1 || 2'); |
| shouldThrowSyntaxError('0 ?? 1 && 2'); |
| shouldNotThrow('(0 || 1) ?? 2'); |
| shouldNotThrow('0 || (1 ?? 2)'); |
| shouldNotThrow('(0 && 1) ?? 2'); |
| shouldNotThrow('0 && (1 ?? 2)'); |
| shouldNotThrow('(0 ?? 1) || 2'); |
| shouldNotThrow('0 ?? (1 || 2)'); |
| shouldNotThrow('(0 ?? 1) && 2'); |
| shouldNotThrow('0 ?? (1 && 2)'); |
| |
| shouldNotThrow('0 || 1 && 2 | 3 ^ 4 & 5 == 6 != 7 === 8 !== 9 < 0 > 1 <= 2 >= 3 << 4 >> 5 >>> 6 + 7 - 8 * 9 / 0 % 1 ** 2'); |
| shouldThrowSyntaxError('0 || 1 && 2 | 3 ^ 4 & 5 == 6 != 7 === 8 !== 9 < 0 > 1 <= 2 >= 3 << 4 >> 5 >>> 6 + 7 - 8 * 9 / 0 % 1 ** 2 ?? 3'); |
| shouldThrowSyntaxError('3 ?? 2 ** 1 % 0 / 9 * 8 - 7 + 6 >>> 5 >> 4 << 3 >= 2 <= 1 > 0 < 9 !== 8 === 7 != 6 == 5 & 4 ^ 3 | 2 && 1 || 0'); |
| |
| shouldBe(null?.x ?? 3, 3); |
| shouldBe(({})?.x ?? 3, 3); |
| shouldBe(({ x: 0 })?.x ?? 3, 0); |
| shouldBe(null?.() ?? 3, 3); |
| shouldBe((() => 0)?.() ?? 3, 0); |
| shouldBe(({ x: 0 })?.[null?.a ?? 'x'] ?? 3, 0); |
| shouldBe((() => 0)?.(null?.a ?? 'x') ?? 3, 0); |