summaryrefslogtreecommitdiffstats
path: root/dom/serviceworkers/test/browser_navigationPreload_read_after_respondWith.js
blob: 6c9fd65adb7aba692b7eb0d2c644426c70edfe2e (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
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
const TOP_DOMAIN = "http://mochi.test:8888/";
const SW_DOMAIN = "https://example.org/";

const TOP_TEST_ROOT = getRootDirectory(gTestPath).replace(
  "chrome://mochitests/content/",
  TOP_DOMAIN
);
const SW_TEST_ROOT = getRootDirectory(gTestPath).replace(
  "chrome://mochitests/content/",
  SW_DOMAIN
);

const TOP_EMPTY_PAGE = `${TOP_TEST_ROOT}empty_with_utils.html`;
const SW_REGISTER_PAGE = `${SW_TEST_ROOT}empty_with_utils.html`;
const SW_IFRAME_PAGE = `${SW_TEST_ROOT}navigationPreload_page.html`;
// An empty script suffices for our SW needs; it's by definition no-fetch.
const SW_REL_SW_SCRIPT = "sw_with_navigationPreload.js";

/**
 * Test the FetchEvent.preloadResponse can be read after FetchEvent.respondWith()
 *
 * Step 1. register a ServiceWorker which only handles FetchEvent when request
 *         url includes navigationPreload_page.html. Otherwise, it alwasy
 *         fallbacks the fetch to the network.
 *         If the request url includes navigationPreload_page.html, it call
 *         FetchEvent.respondWith() with a new Resposne, and then call
 *         FetchEvent.waitUtil() to wait FetchEvent.preloadResponse and post the
 *         preloadResponse's text to clients.
 * Step 2. Open a controlled page and register message event handler to receive
 *         the postMessage from ServiceWorker.
 * Step 3. Create a iframe which url is navigationPreload_page.html, such that
 *         ServiceWorker can fake the response and then send preloadResponse's
 *         result.
 * Step 4. Unregister the ServiceWorker and cleanup the environment.
 */
add_task(async function () {
  await SpecialPowers.pushPrefEnv({
    set: [
      ["dom.serviceWorkers.enabled", true],
      ["dom.serviceWorkers.testing.enabled", true],
    ],
  });

  // Step 1.
  info("Opening a new tab: " + SW_REGISTER_PAGE);
  let topTab = await BrowserTestUtils.openNewForegroundTab({
    gBrowser,
    opening: SW_REGISTER_PAGE,
  });

  // ## Install SW
  await SpecialPowers.spawn(
    topTab.linkedBrowser,
    [{ sw: SW_REL_SW_SCRIPT }],
    async function ({ sw }) {
      // Waive the xray to use the content utils.js script functions.
      dump(`register serviceworker...\n`);
      await content.wrappedJSObject.registerAndWaitForActive(sw);
    }
  );

  // Step 2.
  info("Loading a controlled page: " + SW_REGISTER_PAGE);
  let browserLoadedPromise = BrowserTestUtils.browserLoaded(
    topTab.linkedBrowser
  );
  BrowserTestUtils.loadURIString(topTab.linkedBrowser, SW_REGISTER_PAGE);
  await browserLoadedPromise;

  info("Create a target iframe: " + SW_IFRAME_PAGE);
  let result = await SpecialPowers.spawn(
    topTab.linkedBrowser,
    [{ url: SW_IFRAME_PAGE }],
    async function ({ url }) {
      async function waitForNavigationPreload() {
        return new Promise(resolve => {
          content.wrappedJSObject.navigator.serviceWorker.addEventListener(
            `message`,
            event => {
              resolve(event.data);
            }
          );
        });
      }

      let promise = waitForNavigationPreload();

      // Step 3.
      const iframe = content.wrappedJSObject.document.createElement("iframe");
      iframe.src = url;
      content.wrappedJSObject.document.body.appendChild(iframe);
      await new Promise(r => {
        iframe.onload = r;
      });

      let result = await promise;
      return result;
    }
  );

  is(result, "NavigationPreload\n", "Should get NavigationPreload result");

  // Step 4.
  info("Loading the SW unregister page: " + SW_REGISTER_PAGE);
  browserLoadedPromise = BrowserTestUtils.browserLoaded(topTab.linkedBrowser);
  BrowserTestUtils.loadURIString(topTab.linkedBrowser, SW_REGISTER_PAGE);
  await browserLoadedPromise;

  await SpecialPowers.spawn(topTab.linkedBrowser, [], async function () {
    await content.wrappedJSObject.unregisterAll();
  });

  // Close the testing tab.
  BrowserTestUtils.removeTab(topTab);
});