1use std::default::Default;
6
7use dom_struct::dom_struct;
8use js::rust::HandleObject;
9use net_traits::CoreResourceMsg;
10use net_traits::blob_url_store::parse_blob_url;
11use net_traits::filemanager_thread::FileManagerThreadMsg;
12use profile_traits::generic_channel;
13use script_bindings::cell::DomRefCell;
14use script_bindings::cformat;
15use script_bindings::reflector::{Reflector, reflect_dom_object_with_proto};
16use servo_base::generic_channel::GenericSend;
17use servo_url::{ImmutableOrigin, ServoUrl};
18use url::Url;
19use uuid::Uuid;
20
21use crate::dom::bindings::codegen::Bindings::URLBinding::URLMethods;
22use crate::dom::bindings::error::{Error, ErrorResult, Fallible};
23use crate::dom::bindings::reflector::DomGlobal;
24use crate::dom::bindings::root::{DomRoot, MutNullableDom};
25use crate::dom::bindings::str::{DOMString, USVString};
26use crate::dom::blob::Blob;
27use crate::dom::globalscope::GlobalScope;
28use crate::dom::url::urlhelper::UrlHelper;
29use crate::dom::url::urlsearchparams::URLSearchParams;
30use crate::script_runtime::CanGc;
31
32#[dom_struct]
34#[expect(clippy::upper_case_acronyms)]
35pub(crate) struct URL {
36 reflector_: Reflector,
37
38 #[no_trace]
40 url: DomRefCell<ServoUrl>,
41
42 search_params: MutNullableDom<URLSearchParams>,
44}
45
46impl URL {
47 fn new_inherited(url: ServoUrl) -> URL {
48 URL {
49 reflector_: Reflector::new(),
50 url: DomRefCell::new(url),
51 search_params: Default::default(),
52 }
53 }
54
55 fn new(
56 global: &GlobalScope,
57 proto: Option<HandleObject>,
58 url: ServoUrl,
59 can_gc: CanGc,
60 ) -> DomRoot<URL> {
61 reflect_dom_object_with_proto(Box::new(URL::new_inherited(url)), global, proto, can_gc)
62 }
63
64 pub(crate) fn query_pairs(&self) -> Vec<(String, String)> {
65 self.url
66 .borrow()
67 .as_url()
68 .query_pairs()
69 .into_owned()
70 .collect()
71 }
72
73 pub(crate) fn origin(&self) -> ImmutableOrigin {
74 self.url.borrow().origin()
75 }
76
77 pub(crate) fn set_query_pairs(&self, pairs: &[(String, String)]) {
78 let mut url = self.url.borrow_mut();
79
80 if pairs.is_empty() {
81 url.as_mut_url().set_query(None);
82 } else {
83 url.as_mut_url()
84 .query_pairs_mut()
85 .clear()
86 .extend_pairs(pairs);
87 }
88 }
89
90 fn unicode_serialization_blob_url(origin: &ImmutableOrigin, id: &Uuid) -> String {
92 let mut result = "blob:".to_string();
95
96 result.push_str(&origin.ascii_serialization());
103
104 result.push('/');
106
107 result.push_str(&id.to_string());
109
110 result
112 }
113}
114
115impl URLMethods<crate::DomTypeHolder> for URL {
116 fn Constructor(
118 global: &GlobalScope,
119 proto: Option<HandleObject>,
120 can_gc: CanGc,
121 url: USVString,
122 base: Option<USVString>,
123 ) -> Fallible<DomRoot<URL>> {
124 let parsed_base = match base {
126 None => None,
127 Some(base) => {
128 match ServoUrl::parse(&base.0) {
129 Ok(base) => Some(base),
130 Err(error) => {
131 return Err(Error::Type(cformat!("could not parse base: {}", error)));
133 },
134 }
135 },
136 };
137 let parsed_url = match ServoUrl::parse_with_base(parsed_base.as_ref(), &url.0) {
138 Ok(url) => url,
139 Err(error) => {
140 return Err(Error::Type(cformat!("could not parse URL: {}", error)));
142 },
143 };
144
145 Ok(URL::new(global, proto, parsed_url, can_gc))
156 }
157
158 fn CanParse(_global: &GlobalScope, url: USVString, base: Option<USVString>) -> bool {
160 let parsed_url = run_api_url_parser(url, base);
162
163 parsed_url.is_ok()
166 }
167
168 fn Parse(
170 global: &GlobalScope,
171 url: USVString,
172 base: Option<USVString>,
173 can_gc: CanGc,
174 ) -> Option<DomRoot<URL>> {
175 let parsed_url = run_api_url_parser(url, base).ok()?;
179
180 Some(URL::new(
186 global,
187 None,
188 ServoUrl::from_url(parsed_url),
189 can_gc,
190 ))
191 }
192
193 fn CreateObjectURL(global: &GlobalScope, blob: &Blob) -> DOMString {
195 let origin = global.origin().immutable();
198
199 let id = blob.get_blob_url_id();
200
201 DOMString::from(URL::unicode_serialization_blob_url(origin, &id))
202 }
203
204 fn RevokeObjectURL(global: &GlobalScope, url: DOMString) {
206 let origin = global.origin().immutable();
210
211 if let Ok(url) = ServoUrl::parse(&url.str()) &&
212 url.fragment().is_none() &&
213 *origin == url.origin() &&
214 let Ok((id, _)) = parse_blob_url(&url)
215 {
216 let resource_threads = global.resource_threads();
217 let (tx, rx) = generic_channel::channel(global.time_profiler_chan().clone()).unwrap();
218 let msg = FileManagerThreadMsg::RevokeBlobURL(id, origin.clone(), tx);
219 let _ = resource_threads.send(CoreResourceMsg::ToFileManager(msg));
220
221 let _ = rx.recv().unwrap();
222 }
223 }
224
225 fn Hash(&self) -> USVString {
227 UrlHelper::Hash(&self.url.borrow())
228 }
229
230 fn SetHash(&self, value: USVString) {
232 UrlHelper::SetHash(&mut self.url.borrow_mut(), value);
233 }
234
235 fn Host(&self) -> USVString {
237 UrlHelper::Host(&self.url.borrow())
238 }
239
240 fn SetHost(&self, value: USVString) {
242 UrlHelper::SetHost(&mut self.url.borrow_mut(), value);
243 }
244
245 fn Hostname(&self) -> USVString {
247 UrlHelper::Hostname(&self.url.borrow())
248 }
249
250 fn SetHostname(&self, value: USVString) {
252 UrlHelper::SetHostname(&mut self.url.borrow_mut(), value);
253 }
254
255 fn Href(&self) -> USVString {
257 UrlHelper::Href(&self.url.borrow())
258 }
259
260 fn SetHref(&self, value: USVString) -> ErrorResult {
262 match ServoUrl::parse(&value.0) {
263 Ok(url) => {
264 *self.url.borrow_mut() = url;
265 self.search_params.set(None); Ok(())
267 },
268 Err(error) => Err(Error::Type(cformat!("could not parse URL: {}", error))),
269 }
270 }
271
272 fn Password(&self) -> USVString {
274 UrlHelper::Password(&self.url.borrow())
275 }
276
277 fn SetPassword(&self, value: USVString) {
279 UrlHelper::SetPassword(&mut self.url.borrow_mut(), value);
280 }
281
282 fn Pathname(&self) -> USVString {
284 UrlHelper::Pathname(&self.url.borrow())
285 }
286
287 fn SetPathname(&self, value: USVString) {
289 UrlHelper::SetPathname(&mut self.url.borrow_mut(), value);
290 }
291
292 fn Port(&self) -> USVString {
294 UrlHelper::Port(&self.url.borrow())
295 }
296
297 fn SetPort(&self, value: USVString) {
299 UrlHelper::SetPort(&mut self.url.borrow_mut(), value);
300 }
301
302 fn Protocol(&self) -> USVString {
304 UrlHelper::Protocol(&self.url.borrow())
305 }
306
307 fn SetProtocol(&self, value: USVString) {
309 UrlHelper::SetProtocol(&mut self.url.borrow_mut(), value);
310 }
311
312 fn Origin(&self) -> USVString {
314 UrlHelper::Origin(&self.url.borrow())
315 }
316
317 fn Search(&self) -> USVString {
319 UrlHelper::Search(&self.url.borrow())
320 }
321
322 fn SetSearch(&self, value: USVString) {
324 UrlHelper::SetSearch(&mut self.url.borrow_mut(), value);
325 if let Some(search_params) = self.search_params.get() {
326 search_params.set_list(self.query_pairs());
327 }
328 }
329
330 fn SearchParams(&self, can_gc: CanGc) -> DomRoot<URLSearchParams> {
332 self.search_params
333 .or_init(|| URLSearchParams::new(&self.global(), Some(self), can_gc))
334 }
335
336 fn Username(&self) -> USVString {
338 UrlHelper::Username(&self.url.borrow())
339 }
340
341 fn SetUsername(&self, value: USVString) {
343 UrlHelper::SetUsername(&mut self.url.borrow_mut(), value);
344 }
345
346 fn ToJSON(&self) -> USVString {
348 self.Href()
349 }
350}
351
352fn run_api_url_parser(url: USVString, base: Option<USVString>) -> Result<Url, url::ParseError> {
354 let parsed_base = base.map(|base| Url::parse(&base.0)).transpose()?;
359
360 Url::options().base_url(parsed_base.as_ref()).parse(&url.0)
362}