servo/components/script/dom/history.rs
Martin Robinson 6031a12fd1
Move ScriptToConstellationMsg to constellation_traits (#36364)
This is the last big change necessary to create the
`constellation_traits` crate. This moves the data structure for messages
that originate from the `ScriptThread` and are sent to the
`Contellation` to `constellation_traits`, effectively splitting
`script_traits` in half. Before, `script_traits` was responsible for
exposing the API of both the `ScriptThread` and the `Constellation` to
the rest of Servo.

- Data structures that are used by `ScriptToConstellationMsg` are moved
  to `constellation_traits`. The dependency graph looks a bit like this:
  `script_layout_interface` depends on `script_traits` depends on
  `constellation_traits` depends on `embedder_traits`.
- Data structures that are used in the embedding layer
  (`UntrustedNodeAddress`, `CompositorHitTestResult`, `TouchEventResult`
  and `AnimationState`) are moved to embedder_traits, to avoid a
  dependency cycle between `webrender_traits` and
  `constellation_traits`.
- Types dealing with MessagePorts and serialization are moved to
  `constellation_traits::message_port`.

Testing: This is covered by existing tests as it just moves types
around.
Signed-off-by: Martin Robinson <mrobinson@igalia.com>

Signed-off-by: Martin Robinson <mrobinson@igalia.com>
2025-04-05 22:13:29 +00:00

395 lines
14 KiB
Rust

/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
use std::cell::Cell;
use std::cmp::Ordering;
use base::id::HistoryStateId;
use constellation_traits::{
ScriptToConstellationMessage, StructuredSerializedData, TraversalDirection,
};
use dom_struct::dom_struct;
use js::jsapi::Heap;
use js::jsval::{JSVal, NullValue, UndefinedValue};
use js::rust::{HandleValue, MutableHandleValue};
use net_traits::{CoreResourceMsg, IpcSend};
use profile_traits::ipc;
use profile_traits::ipc::channel;
use servo_url::ServoUrl;
use crate::dom::bindings::codegen::Bindings::HistoryBinding::HistoryMethods;
use crate::dom::bindings::codegen::Bindings::LocationBinding::Location_Binding::LocationMethods;
use crate::dom::bindings::codegen::Bindings::WindowBinding::WindowMethods;
use crate::dom::bindings::error::{Error, ErrorResult, Fallible};
use crate::dom::bindings::inheritance::Castable;
use crate::dom::bindings::reflector::{DomGlobal, Reflector, reflect_dom_object};
use crate::dom::bindings::root::{Dom, DomRoot};
use crate::dom::bindings::str::{DOMString, USVString};
use crate::dom::bindings::structuredclone;
use crate::dom::event::Event;
use crate::dom::eventtarget::EventTarget;
use crate::dom::globalscope::GlobalScope;
use crate::dom::hashchangeevent::HashChangeEvent;
use crate::dom::popstateevent::PopStateEvent;
use crate::dom::window::Window;
use crate::script_runtime::{CanGc, JSContext};
enum PushOrReplace {
Push,
Replace,
}
/// <https://html.spec.whatwg.org/multipage/#the-history-interface>
#[dom_struct]
pub(crate) struct History {
reflector_: Reflector,
window: Dom<Window>,
#[ignore_malloc_size_of = "mozjs"]
state: Heap<JSVal>,
#[no_trace]
state_id: Cell<Option<HistoryStateId>>,
}
impl History {
pub(crate) fn new_inherited(window: &Window) -> History {
let state = Heap::default();
state.set(NullValue());
History {
reflector_: Reflector::new(),
window: Dom::from_ref(window),
state,
state_id: Cell::new(None),
}
}
pub(crate) fn new(window: &Window, can_gc: CanGc) -> DomRoot<History> {
reflect_dom_object(Box::new(History::new_inherited(window)), window, can_gc)
}
}
impl History {
fn traverse_history(&self, direction: TraversalDirection) -> ErrorResult {
if !self.window.Document().is_fully_active() {
return Err(Error::Security);
}
let msg = ScriptToConstellationMessage::TraverseHistory(direction);
let _ = self
.window
.as_global_scope()
.script_to_constellation_chan()
.send(msg);
Ok(())
}
/// <https://html.spec.whatwg.org/multipage/#history-traversal>
/// Steps 5-16
#[allow(unsafe_code)]
pub(crate) fn activate_state(
&self,
state_id: Option<HistoryStateId>,
url: ServoUrl,
can_gc: CanGc,
) {
// Steps 5
let document = self.window.Document();
let old_url = document.url().clone();
document.set_url(url.clone());
// Step 6
let hash_changed = old_url.fragment() != url.fragment();
// Step 8
if let Some(fragment) = url.fragment() {
document.check_and_scroll_fragment(fragment, can_gc);
}
// Step 11
let state_changed = state_id != self.state_id.get();
self.state_id.set(state_id);
let serialized_data = match state_id {
Some(state_id) => {
let (tx, rx) = ipc::channel(self.global().time_profiler_chan().clone()).unwrap();
let _ = self
.window
.as_global_scope()
.resource_threads()
.send(CoreResourceMsg::GetHistoryState(state_id, tx));
rx.recv().unwrap()
},
None => None,
};
match serialized_data {
Some(data) => {
let data = StructuredSerializedData {
serialized: data,
..Default::default()
};
rooted!(in(*GlobalScope::get_cx()) let mut state = UndefinedValue());
if structuredclone::read(self.window.as_global_scope(), data, state.handle_mut())
.is_err()
{
warn!("Error reading structuredclone data");
}
self.state.set(state.get());
},
None => {
self.state.set(NullValue());
},
}
// TODO: Queue events on DOM Manipulation task source if non-blocking flag is set.
// Step 16.1
if state_changed {
PopStateEvent::dispatch_jsval(
self.window.upcast::<EventTarget>(),
&self.window,
unsafe { HandleValue::from_raw(self.state.handle()) },
can_gc,
);
}
// Step 16.3
if hash_changed {
let event = HashChangeEvent::new(
&self.window,
atom!("hashchange"),
false,
false,
old_url.into_string(),
url.into_string(),
can_gc,
);
event
.upcast::<Event>()
.fire(self.window.upcast::<EventTarget>(), can_gc);
}
}
pub(crate) fn remove_states(&self, states: Vec<HistoryStateId>) {
let _ = self
.window
.as_global_scope()
.resource_threads()
.send(CoreResourceMsg::RemoveHistoryStates(states));
}
/// <https://html.spec.whatwg.org/multipage/#dom-history-pushstate>
/// <https://html.spec.whatwg.org/multipage/#dom-history-replacestate>
fn push_or_replace_state(
&self,
cx: JSContext,
data: HandleValue,
_title: DOMString,
url: Option<USVString>,
push_or_replace: PushOrReplace,
) -> ErrorResult {
// Step 1
let document = self.window.Document();
// Step 2
if !document.is_fully_active() {
return Err(Error::Security);
}
// TODO: Step 3 Optionally abort these steps
// https://github.com/servo/servo/issues/19159
// Step 4. Let serializedData be StructuredSerializeForStorage(data). Rethrow any exceptions.
let serialized_data = structuredclone::write(cx, data, None)?;
// Step 5. Let newURL be document's URL.
let new_url: ServoUrl = match url {
// Step 6. If url is not null or the empty string, then:
Some(urlstring) => {
let document_url = document.url();
// Step 6.1 Set newURL to the result of encoding-parsing a URL given url,
// relative to the relevant settings object of history.
let Ok(url) = ServoUrl::parse_with_base(Some(&document_url), &urlstring.0) else {
// Step 6.2 If newURL is failure, then throw a "SecurityError" DOMException.
return Err(Error::Security);
};
// Step 6.3 If document cannot have its URL rewritten to newURL,
// then throw a "SecurityError" DOMException.
if !Self::can_have_url_rewritten(&document_url, &url) {
return Err(Error::Security);
}
url
},
None => document.url(),
};
// Step 8
let state_id = match push_or_replace {
PushOrReplace::Push => {
let state_id = HistoryStateId::new();
self.state_id.set(Some(state_id));
let msg = ScriptToConstellationMessage::PushHistoryState(state_id, new_url.clone());
let _ = self
.window
.as_global_scope()
.script_to_constellation_chan()
.send(msg);
state_id
},
PushOrReplace::Replace => {
let state_id = match self.state_id.get() {
Some(state_id) => state_id,
None => {
let state_id = HistoryStateId::new();
self.state_id.set(Some(state_id));
state_id
},
};
let msg =
ScriptToConstellationMessage::ReplaceHistoryState(state_id, new_url.clone());
let _ = self
.window
.as_global_scope()
.script_to_constellation_chan()
.send(msg);
state_id
},
};
let _ = self.window.as_global_scope().resource_threads().send(
CoreResourceMsg::SetHistoryState(state_id, serialized_data.serialized.clone()),
);
// TODO: Step 9 Update current entry to represent a GET request
// https://github.com/servo/servo/issues/19156
// Step 10
document.set_url(new_url);
// Step 11
rooted!(in(*cx) let mut state = UndefinedValue());
if structuredclone::read(
self.window.as_global_scope(),
serialized_data,
state.handle_mut(),
)
.is_err()
{
warn!("Error reading structuredclone data");
}
// Step 12
self.state.set(state.get());
// TODO: Step 13 Update Document's latest entry to current entry
// https://github.com/servo/servo/issues/19158
Ok(())
}
/// <https://html.spec.whatwg.org/multipage/#can-have-its-url-rewritten>
/// Step 2-6
fn can_have_url_rewritten(document_url: &ServoUrl, target_url: &ServoUrl) -> bool {
// Step 2. If targetURL and documentURL differ in their scheme, username,
// password, host, or port components, then return false.
if target_url.scheme() != document_url.scheme() ||
target_url.username() != document_url.username() ||
target_url.password() != document_url.password() ||
target_url.host() != document_url.host() ||
target_url.port() != document_url.port()
{
return false;
}
// Step 3. If targetURL's scheme is an HTTP(S) scheme, then return true.
if target_url.scheme() == "http" || target_url.scheme() == "https" {
return true;
}
// Step 4. If targetURL's scheme is "file", then:
if target_url.scheme() == "file" {
// Step 4.1 If targetURL and documentURL differ in their path component, then return false.
// Step 4.2 Return true.
return target_url.path() == document_url.path();
}
// Step 5. If targetURL and documentURL differ in their path component
// or query components, then return false.
if target_url.path() != document_url.path() || target_url.query() != document_url.query() {
return false;
}
// Step 6. Return true.
true
}
}
impl HistoryMethods<crate::DomTypeHolder> for History {
/// <https://html.spec.whatwg.org/multipage/#dom-history-state>
fn GetState(&self, _cx: JSContext, mut retval: MutableHandleValue) -> Fallible<()> {
if !self.window.Document().is_fully_active() {
return Err(Error::Security);
}
retval.set(self.state.get());
Ok(())
}
/// <https://html.spec.whatwg.org/multipage/#dom-history-length>
fn GetLength(&self) -> Fallible<u32> {
if !self.window.Document().is_fully_active() {
return Err(Error::Security);
}
let (sender, recv) = channel(self.global().time_profiler_chan().clone())
.expect("Failed to create channel to send jsh length.");
let msg = ScriptToConstellationMessage::JointSessionHistoryLength(sender);
let _ = self
.window
.as_global_scope()
.script_to_constellation_chan()
.send(msg);
Ok(recv.recv().unwrap())
}
/// <https://html.spec.whatwg.org/multipage/#dom-history-go>
fn Go(&self, delta: i32, can_gc: CanGc) -> ErrorResult {
let direction = match delta.cmp(&0) {
Ordering::Greater => TraversalDirection::Forward(delta as usize),
Ordering::Less => TraversalDirection::Back(-delta as usize),
Ordering::Equal => return self.window.Location().Reload(can_gc),
};
self.traverse_history(direction)
}
/// <https://html.spec.whatwg.org/multipage/#dom-history-back>
fn Back(&self) -> ErrorResult {
self.traverse_history(TraversalDirection::Back(1))
}
/// <https://html.spec.whatwg.org/multipage/#dom-history-forward>
fn Forward(&self) -> ErrorResult {
self.traverse_history(TraversalDirection::Forward(1))
}
/// <https://html.spec.whatwg.org/multipage/#dom-history-pushstate>
fn PushState(
&self,
cx: JSContext,
data: HandleValue,
title: DOMString,
url: Option<USVString>,
) -> ErrorResult {
self.push_or_replace_state(cx, data, title, url, PushOrReplace::Push)
}
/// <https://html.spec.whatwg.org/multipage/#dom-history-replacestate>
fn ReplaceState(
&self,
cx: JSContext,
data: HandleValue,
title: DOMString,
url: Option<USVString>,
) -> ErrorResult {
self.push_or_replace_state(cx, data, title, url, PushOrReplace::Replace)
}
}