| <!DOCTYPE html> |
| <meta charset="utf-8"> |
| <title>Incumbent settings object for promise jobs</title> |
| <script src="/resources/testharness.js"></script> |
| <script src="/resources/testharnessreport.js"></script> |
| |
| <!-- This is the entry page. --> |
| |
| <iframe src="resources/promise-job-incumbent-incumbent.html"></iframe> |
| <iframe src="resources/promise-job-incumbent-resolver.html"></iframe> |
| |
| <script> |
| setup({ explicit_done: true }); |
| |
| // postMessage should pick the incumbent page as its .source value to set on the MessageEvent, even |
| // inside promise jobs. |
| const expectedURL = (new URL("resources/promise-job-incumbent-incumbent.html", location.href)).href; |
| |
| let testId = 0; |
| |
| window.onload = () => { |
| const relevantWindow = frames[0].document.querySelector("#r").contentWindow; |
| const runInResolver = frames[1].runWhatYouGiveMe; |
| |
| function setupTest(t) { |
| ++testId; |
| const thisTestId = testId; |
| |
| relevantWindow.addEventListener("messagereceived", t.step_func(e => { |
| const [receivedTestId, receivedSourceURL] = e.detail; |
| |
| if (receivedTestId !== thisTestId) { |
| return; |
| } |
| |
| assert_equals(receivedSourceURL, expectedURL); |
| t.done(); |
| })); |
| |
| return thisTestId; |
| } |
| |
| async_test(t => { |
| const thisTestId = setupTest(t); |
| |
| frames[0].runWindowPostMessageVeryIndirectly(thisTestId, "*"); |
| }, "Sanity check: this all works as expected with no promises involved"); |
| |
| async_test(t => { |
| const thisTestId = setupTest(t); |
| |
| // No t.step_func because that could change the realms |
| Promise.resolve().then(() => { |
| frames[0].runWindowPostMessageVeryIndirectly(thisTestId, "*"); |
| }); |
| }, "Fulfillment handler on fulfilled promise"); |
| |
| async_test(t => { |
| const thisTestId = setupTest(t); |
| |
| const p = Promise.resolve(); |
| frames[0].runWindowPostMessageVeryIndirectlyWithNoUserCode(p, "then", thisTestId, "*"); |
| }, "Fulfillment handler on fulfilled promise, using backup incumbent settings object stack"); |
| |
| async_test(t => { |
| const thisTestId = setupTest(t); |
| |
| // No t.step_func because that could change the realms |
| Promise.reject().catch(() => { |
| frames[0].runWindowPostMessageVeryIndirectly(thisTestId, "*"); |
| }); |
| }, "Rejection handler on rejected promise"); |
| |
| async_test(t => { |
| const thisTestId = setupTest(t); |
| |
| const p = Promise.reject(); |
| frames[0].runWindowPostMessageVeryIndirectlyWithNoUserCode(p, "catch", thisTestId, "*"); |
| }, "Rejection handler on rejected promise, using backup incumbent settings object stack"); |
| |
| // The following tests test that we derive the incumbent settings object at promise-job time from |
| // the incumbent realm at the time the handler was added, not at the time the resolve()/reject() |
| // was done. See https://github.com/whatwg/html/issues/5213 for the spec side of this issue. |
| |
| async_test(t => { |
| const thisTestId = setupTest(t); |
| |
| let resolve; |
| const p = new Promise(r => { resolve = r; }); |
| |
| // No t.step_func because that could change the realms |
| p.then(() => { |
| frames[0].runWindowPostMessageVeryIndirectly(thisTestId, "*"); |
| }); |
| |
| t.step_timeout(() => { |
| runInResolver(resolve); |
| }, 0); |
| }, "Fulfillment handler on pending-then-fulfilled promise"); |
| |
| async_test(t => { |
| const thisTestId = setupTest(t); |
| |
| let resolve; |
| const p = new Promise(r => { resolve = r; }); |
| |
| frames[0].runWindowPostMessageVeryIndirectlyWithNoUserCode(p, "then", thisTestId, "*"); |
| |
| t.step_timeout(() => { |
| runInResolver(resolve); |
| }, 0); |
| }, "Fulfillment handler on pending-then-fulfilled promise, using backup incumbent settings object stack"); |
| |
| async_test(t => { |
| const thisTestId = setupTest(t); |
| |
| let reject; |
| const p = new Promise((_, r) => { reject = r; }); |
| |
| // No t.step_func because that could change the realms |
| p.catch(() => { |
| frames[0].runWindowPostMessageVeryIndirectly(thisTestId, "*"); |
| }); |
| |
| t.step_timeout(() => { |
| runInResolver(reject); |
| }, 0); |
| }, "Rejection handler on pending-then-rejected promise"); |
| |
| async_test(t => { |
| const thisTestId = setupTest(t); |
| |
| let reject; |
| const p = new Promise((_, r) => { reject = r; }); |
| |
| frames[0].runWindowPostMessageVeryIndirectlyWithNoUserCode(p, "catch", thisTestId, "*"); |
| |
| t.step_timeout(() => { |
| runInResolver(reject); |
| }, 0); |
| }, "Rejection handler on pending-then-rejected promise, using backup incumbent settings object stack"); |
| |
| async_test(t => { |
| const thisTestId = setupTest(t); |
| |
| const thenable = { |
| // No t.step_func because that could change the realms |
| then(f) { |
| frames[0].runWindowPostMessageVeryIndirectly(thisTestId, "*"); |
| } |
| }; |
| |
| Promise.resolve(thenable); |
| }, "Thenable resolution"); |
| |
| async_test(t => { |
| const thisTestId = setupTest(t); |
| |
| frames[0].resolveThenableThatRunsWindowPostMessageVeryIndirectlyWithNoUserCode(testId, "*", []); |
| }, "Thenable resolution, using backup incumbent settings object stack"); |
| |
| done(); |
| }; |
| </script> |