summaryrefslogtreecommitdiffstats
path: root/testing/web-platform/tests/webidl/ecmascript-binding/global-object-implicit-this-value.any.js
blob: 4c159c67519c0bdd33b6d2263c94591bfb7e1470 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
// META: global=window,worker

// https://webidl.spec.whatwg.org/#dfn-attribute-getter (step 1.1.2.1)
// https://webidl.spec.whatwg.org/#dfn-attribute-setter (step 4.5.1)
// https://webidl.spec.whatwg.org/#dfn-create-operation-function (step 2.1.2.1)

const notGlobalObject = Object.create(Object.getPrototypeOf(globalThis));

test(() => {
    assert_throws_js(TypeError, () => { Object.create(globalThis).self; });
    assert_throws_js(TypeError, () => { getGlobalPropertyDescriptor("location").get.call(notGlobalObject); });
    assert_throws_js(TypeError, () => { Reflect.get(globalThis, "navigator", notGlobalObject); });
    assert_throws_js(TypeError, () => { new Proxy(globalThis, {}).onerror; });
}, "Global object's getter throws when called on incompatible object");

test(() => {
    assert_throws_js(TypeError, () => { Object.create(globalThis).origin = origin; });
    assert_throws_js(TypeError, () => { getGlobalPropertyDescriptor("onerror").set.call(notGlobalObject, onerror); });
    assert_throws_js(TypeError, () => { Reflect.set(globalThis, "onoffline", onoffline, notGlobalObject); });
    assert_throws_js(TypeError, () => { new Proxy(globalThis, {}).ononline = ononline; });
}, "Global object's setter throws when called on incompatible object");

test(() => {
    assert_throws_js(TypeError, () => { Object.create(globalThis).setInterval(() => {}, 100); });
    assert_throws_js(TypeError, () => { clearTimeout.call(notGlobalObject, () => {}); });
    assert_throws_js(TypeError, () => { Reflect.apply(btoa, notGlobalObject, [""]); });
    assert_throws_js(TypeError, () => { new Proxy(globalThis, {}).removeEventListener("foo", () => {}); });
}, "Global object's operation throws when called on incompatible object");

if (typeof document !== "undefined") {
    test(() => {
        assert_throws_js(TypeError, () => { Object.getOwnPropertyDescriptor(window, "document").get.call(document.all); });
    }, "Global object's getter throws when called on incompatible object (document.all)");

    test(() => {
        assert_throws_js(TypeError, () => { Object.getOwnPropertyDescriptor(window, "name").set.call(document.all); });
    }, "Global object's setter throws when called on incompatible object (document.all)");

    test(() => {
        assert_throws_js(TypeError, () => { focus.call(document.all); });
    }, "Global object's operation throws when called on incompatible object (document.all)");
}

// An engine might have different code path for calling a function from outer scope to implement step 1.b.iii of https://tc39.es/ecma262/#sec-evaluatecall
const locationGetter = getGlobalPropertyDescriptor("location").get;
test(() => {
    assert_equals(getGlobalPropertyDescriptor("self").get.call(null), self);
    assert_equals((() => locationGetter())(), location);
    assert_equals(Reflect.get(globalThis, "origin", null), origin);
    assert_equals(Reflect.get(globalThis, "onoffline", undefined), onoffline);
}, "Global object's getter works when called on null / undefined");

test(() => {
    const fn = () => {};

    // origin is [Replaceable]
    getGlobalPropertyDescriptor("origin").set.call(null, "foo");
    assert_equals(origin, "foo");
    getGlobalPropertyDescriptor("onerror").set.call(undefined, fn);
    assert_equals(onerror, fn);
    assert_true(Reflect.set(globalThis, "onoffline", fn, null));
    assert_equals(onoffline, fn);

    const ononlineSetter = getGlobalPropertyDescriptor("ononline").set;
    (() => { ononlineSetter(fn); })();
    assert_equals(ononline, fn);
}, "Global object's setter works when called on null / undefined");

// An engine might have different code path for calling a function from outer scope to implement step 1.b.iii of https://tc39.es/ecma262/#sec-evaluatecall
const __addEventListener = addEventListener;
test(() => {
    assert_equals(atob.call(null, ""), "");
    assert_equals(typeof (0, setInterval)(() => {}, 100), "number");

    (() => { __addEventListener("foo", event => { event.preventDefault(); }); })();
    const __dispatchEvent = dispatchEvent;
    (() => { assert_false(__dispatchEvent(new Event("foo", { cancelable: true }))); })();
}, "Global object's operation works when called on null / undefined");

function getGlobalPropertyDescriptor(key) {
    for (let obj = globalThis; obj; obj = Object.getPrototypeOf(obj)) {
        const desc = Object.getOwnPropertyDescriptor(obj, key);
        if (desc) return desc;
    }
}