LibWeb: Capture <script> element's node document on execution

Step 1 of the spec is to capture the <script> element's node document
into a local variable.

When I originally implemented this, I thought this was not necessary.
However, I realised that the script that runs can adopt the current
script element into a different document, meaning step 5.4 and 6 then
operate on the incorrect document.

Covered by this WPT: 7b0ebaccc6/html/semantics/scripting-1/the-script-element/moving-between-documents-during-evaluation.html
This commit is contained in:
Luke Wilde 2021-12-15 03:59:36 +00:00 committed by Brian Gianforcaro
parent cacac7927b
commit 54454952e0
Notes: sideshowbarker 2024-07-17 22:26:28 +09:00
2 changed files with 45 additions and 8 deletions

View file

@ -45,10 +45,11 @@ void HTMLScriptElement::set_non_blocking(Badge<HTMLParser>, bool non_blocking)
// https://html.spec.whatwg.org/multipage/scripting.html#execute-the-script-block
void HTMLScriptElement::execute_script()
{
// 1. Let document be scriptElement's node document. (NOTE: This is not necessary)
// 1. Let document be scriptElement's node document.
NonnullRefPtr<DOM::Document> node_document = document();
// 2. If scriptElement's preparation-time document is not equal to document, then return.
if (m_preparation_time_document.ptr() != &document()) {
if (m_preparation_time_document.ptr() != node_document.ptr()) {
dbgln("HTMLScriptElement: Refusing to run script because the preparation time document is not the same as the node document.");
return;
}
@ -63,7 +64,7 @@ void HTMLScriptElement::execute_script()
// 4. If scriptElement is from an external file, or the script's type for scriptElement is "module", then increment document's ignore-destructive-writes counter.
bool incremented_destructive_writes_counter = false;
if (m_from_an_external_file || m_script_type == ScriptType::Module) {
document().increment_ignore_destructive_writes_counter();
node_document->increment_ignore_destructive_writes_counter();
incremented_destructive_writes_counter = true;
}
@ -74,9 +75,9 @@ void HTMLScriptElement::execute_script()
auto old_current_script = document().current_script();
// 2. If scriptElement's root is not a shadow root, then set document's currentScript attribute to scriptElement. Otherwise, set it to null.
if (!is<DOM::ShadowRoot>(root()))
document().set_current_script({}, this);
node_document->set_current_script({}, this);
else
document().set_current_script({}, nullptr);
node_document->set_current_script({}, nullptr);
if (m_from_an_external_file)
dbgln_if(HTML_SCRIPT_DEBUG, "HTMLScriptElement: Running script {}", attribute(HTML::AttributeNames::src));
@ -86,8 +87,8 @@ void HTMLScriptElement::execute_script()
// 3. Run the classic script given by the script's script for scriptElement.
verify_cast<ClassicScript>(*m_script).run();
// Set document's currentScript attribute to oldCurrentScript.
document().set_current_script({}, old_current_script);
// 4. Set document's currentScript attribute to oldCurrentScript.
node_document->set_current_script({}, old_current_script);
} else {
// -> "module"
// 1. Assert: document's currentScript attribute is null.
@ -99,7 +100,7 @@ void HTMLScriptElement::execute_script()
// 6. Decrement the ignore-destructive-writes counter of document, if it was incremented in the earlier step.
if (incremented_destructive_writes_counter)
document().decrement_ignore_destructive_writes_counter();
node_document->decrement_ignore_destructive_writes_counter();
// 7. If scriptElement is from an external file, then fire an event named load at scriptElement.
if (m_from_an_external_file)

View file

@ -0,0 +1,36 @@
describe("currentScript", () => {
loadLocalPage("/res/html/misc/blank.html");
beforeInitialPageLoad(page => {
expect(page.document.currentScript).toBeNull();
});
afterInitialPageLoad(page => {
test("reset to null even if currentScript is adopted into another document", () => {
const script = page.document.createElement("script");
script.id = "test";
script.innerText = `
const newDocument = globalThis.pageObject.document.implementation.createHTMLDocument();
const thisScript = globalThis.pageObject.document.getElementById("test");
// currentScript should stay the same even across adoption.
expect(globalThis.pageObject.document.currentScript).toBe(thisScript);
newDocument.adoptNode(thisScript);
expect(globalThis.pageObject.document.currentScript).toBe(thisScript);
`;
// currentScript should be null before and after running the script on insertion.
expect(page.document.currentScript).toBeNull();
expect(script.ownerDocument).toBe(page.document);
globalThis.pageObject = page;
page.document.body.appendChild(script);
globalThis.pageObject = undefined;
expect(page.document.currentScript).toBeNull();
expect(script.ownerDocument).not.toBe(page.document);
});
});
waitForPageToLoad();
});