// Copyright 2019 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // Flags: --harmony-optional-chaining function shouldThrowSyntaxError(script) { let error; try { eval(script); } catch (e) { error = e; } if (!(error instanceof SyntaxError)) { throw new Error('Expected SyntaxError!'); } } assertEquals(undefined?.valueOf(), undefined); assertEquals(null?.valueOf(), undefined); assertEquals(true?.valueOf(), true); assertEquals(false?.valueOf(), false); assertEquals(0?.valueOf(), 0); assertEquals(1?.valueOf(), 1); assertEquals(''?.valueOf(), ''); assertEquals('hi'?.valueOf(), 'hi'); assertEquals(({})?.constructor, Object); assertEquals(({ x: 'hi' })?.x, 'hi'); assertEquals([]?.length, 0); assertEquals(['hi']?.length, 1); assertEquals(undefined?.['valueOf'](), undefined); assertEquals(null?.['valueOf'](), undefined); assertEquals(true?.['valueOf'](), true); assertEquals(false?.['valueOf'](), false); assertEquals(0?.['valueOf'](), 0); assertEquals(1?.['valueOf'](), 1); assertEquals(''?.['valueOf'](), ''); assertEquals('hi'?.['valueOf'](), 'hi'); assertEquals(({})?.['constructor'], Object); assertEquals(({ x: 'hi' })?.['x'], 'hi'); assertEquals([]?.['length'], 0); assertEquals(['hi']?.[0], 'hi'); assertEquals(undefined?.(), undefined); assertEquals(null?.(), undefined); assertThrows(() => true?.(), TypeError); assertThrows(() => false?.(), TypeError); assertThrows(() => 0?.(), TypeError); assertThrows(() => 1?.(), TypeError); assertThrows(() => ''?.(), TypeError); assertThrows(() => 'hi'?.(), TypeError); assertThrows(() => ({})?.(), TypeError); assertThrows(() => ({ x: 'hi' })?.(), TypeError); assertThrows(() => []?.(), TypeError); assertThrows(() => ['hi']?.(), TypeError); assertThrows(() => ({})?.a['b'], TypeError); assertEquals(({})?.a?.['b'], undefined); assertEquals(null?.a['b']().c, undefined); assertThrows(() => ({})?.['a'].b); assertEquals(({})?.['a']?.b, undefined); assertEquals(null?.['a'].b()['c'], undefined); assertThrows(() => (() => {})?.()(), TypeError); assertEquals((() => {})?.()?.(), undefined); assertEquals(null?.()().a['b'], undefined); assertEquals(delete undefined?.foo, true); assertEquals(delete null?.foo, true); assertEquals(delete undefined?.['foo'], true); assertEquals(delete null?.['foo'], true); assertEquals(delete undefined?.(), true); assertEquals(delete null?.(), true); assertEquals(undefined?.(...a), undefined); assertEquals(null?.(1, ...a), undefined); assertEquals(({}).a?.(...a), undefined); assertEquals(({ a: null }).a?.(...a), undefined); assertEquals(undefined?.(...a)?.(1, ...a), undefined); assertThrows(() => 5?.(...[]), TypeError); const o1 = { x: 0, y: 0, z() {} }; assertEquals(delete o1?.x, true); assertEquals(o1.x, undefined); assertEquals(delete o1?.x, true); assertEquals(delete o1?.['y'], true); assertEquals(o1.y, undefined); assertEquals(delete o1?.['y'], true); assertEquals(delete o1.z?.(), true); assertThrows(() => { delete ({})?.foo.bar; }); shouldThrowSyntaxError('class C {} class D extends C { foo() { return super?.bar; } }'); shouldThrowSyntaxError('class C {} class D extends C { foo() { return super?.["bar"]; } }'); shouldThrowSyntaxError('class C {} class D extends C { constructor() { super?.(); } }'); shouldThrowSyntaxError('const o = { C: class {} }; new o?.C();'); shouldThrowSyntaxError('const o = { C: class {} }; new o?.["C"]();'); shouldThrowSyntaxError('class C {} new C?.();'); shouldThrowSyntaxError('function foo() { new?.target; }'); shouldThrowSyntaxError('function tag() {} tag?.``;'); shouldThrowSyntaxError('const o = { tag() {} }; o?.tag``;'); shouldThrowSyntaxError('class A { #foo = "hi"; constructor() { this?.#foo; } }') const o2 = { count: 0, get x() { this.count += 1; return () => {}; }, }; o2.x?.y; assertEquals(o2.count, 1); o2.x?.['y']; assertEquals(o2.count, 2); o2.x?.(); assertEquals(o2.count, 3); assertEquals(true?.5:5, 0.5);