summaryrefslogtreecommitdiffstats
path: root/docshell/test/unit/test_browsing_context_structured_clone.js
blob: d06f7aecf601ca1495f670ca3a4f321d3ef57c72 (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
"use strict";

add_task(async function test_BrowsingContext_structured_clone() {
  let browser = Services.appShell.createWindowlessBrowser(false);

  let frame = browser.document.createElement("iframe");

  await new Promise(r => {
    frame.onload = () => r();
    browser.document.body.appendChild(frame);
  });

  let { browsingContext } = frame;

  let sch = new StructuredCloneHolder({ browsingContext });

  let deserialize = () => sch.deserialize({}, true);

  // Check that decoding a live browsing context produces the correct
  // object.
  equal(
    deserialize().browsingContext,
    browsingContext,
    "Got correct browsing context from StructuredClone deserialize"
  );

  // Check that decoding a second time still succeeds.
  equal(
    deserialize().browsingContext,
    browsingContext,
    "Got correct browsing context from second StructuredClone deserialize"
  );

  // Destroy the browsing context and make sure that the decode fails
  // with a DataCloneError.
  //
  // Making sure the BrowsingContext is actually destroyed by the time
  // we do the second decode is a bit tricky. We obviously have clear
  // our local references to it, and give the GC a chance to reap them.
  // And we also, of course, have to destroy the frame that it belongs
  // to, or its frame loader and window global would hold it alive.
  //
  // Beyond that, we don't *have* to reload or destroy the parent
  // document, but we do anyway just to be safe.
  //

  frame.remove();
  frame = null;
  browsingContext = null;

  browser.document.location.reload();
  browser.close();

  // We will schedule a precise GC and do both GC and CC a few times, to make
  // sure we have completely destroyed the WindowGlobal actors (which keep
  // references to their BrowsingContexts) in order
  // to allow their (now snow-white) references to be collected.
  await schedulePreciseGCAndForceCC(3);

  // OK. We can be fairly confident that the BrowsingContext object
  // stored in our structured clone data has been destroyed. Make sure
  // that attempting to decode it again leads to the appropriate error.
  Assert.throws(
    deserialize,
    e => e.name === "DataCloneError",
    "Should get a DataCloneError when trying to decode a dead BrowsingContext"
  );
});