| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2024-10-04 13:19:50 +02:00
										 |  |  |  * Copyright (c) 2022, Andreas Kling <andreas@ladybird.org> | 
					
						
							| 
									
										
										
										
											2023-02-28 00:06:10 +00:00
										 |  |  |  * Copyright (c) 2023, Luke Wilde <lukew@serenityos.org> | 
					
						
							| 
									
										
										
										
											2024-12-23 19:10:01 +13:00
										 |  |  |  * Copyright (c) 2024, Shannon Booth <shannon@serenityos.org> | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-26 16:30:02 +12:00
										 |  |  | #include <AK/String.h>
 | 
					
						
							| 
									
										
										
										
											2022-09-25 16:38:21 -06:00
										 |  |  | #include <LibWeb/Bindings/Intrinsics.h>
 | 
					
						
							| 
									
										
										
										
											2024-04-27 12:09:58 +12:00
										 |  |  | #include <LibWeb/Bindings/StoragePrototype.h>
 | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | #include <LibWeb/HTML/Storage.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Web::HTML { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  | GC_DEFINE_ALLOCATOR(Storage); | 
					
						
							| 
									
										
										
										
											2023-11-19 19:47:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-23 19:10:01 +13:00
										 |  |  | GC::Ref<Storage> Storage::create(JS::Realm& realm, u64 quota_bytes) | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-12-23 19:10:01 +13:00
										 |  |  |     return realm.create<Storage>(realm, quota_bytes); | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-23 19:10:01 +13:00
										 |  |  | Storage::Storage(JS::Realm& realm, u64 quota_bytes) | 
					
						
							| 
									
										
										
										
											2024-01-09 16:05:03 -07:00
										 |  |  |     : Bindings::PlatformObject(realm) | 
					
						
							| 
									
										
										
										
											2024-12-23 19:10:01 +13:00
										 |  |  |     , m_quota_bytes(quota_bytes) | 
					
						
							| 
									
										
										
										
											2022-09-03 19:44:37 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-01-09 16:05:03 -07:00
										 |  |  |     m_legacy_platform_object_flags = LegacyPlatformObjectFlags { | 
					
						
							| 
									
										
										
										
											2024-10-18 00:04:35 +02:00
										 |  |  |         .supports_indexed_properties = true, | 
					
						
							| 
									
										
										
										
											2024-01-09 16:05:03 -07:00
										 |  |  |         .supports_named_properties = true, | 
					
						
							| 
									
										
										
										
											2024-10-18 00:04:35 +02:00
										 |  |  |         .has_indexed_property_setter = true, | 
					
						
							| 
									
										
										
										
											2024-01-09 16:05:03 -07:00
										 |  |  |         .has_named_property_setter = true, | 
					
						
							|  |  |  |         .has_named_property_deleter = true, | 
					
						
							| 
									
										
										
										
											2024-10-18 00:04:35 +02:00
										 |  |  |         .indexed_property_setter_has_identifier = true, | 
					
						
							| 
									
										
										
										
											2024-01-09 16:05:03 -07:00
										 |  |  |         .named_property_setter_has_identifier = true, | 
					
						
							|  |  |  |         .named_property_deleter_has_identifier = true, | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2022-09-03 19:44:37 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-14 13:21:51 -06:00
										 |  |  | Storage::~Storage() = default; | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-07 08:41:28 +02:00
										 |  |  | void Storage::initialize(JS::Realm& realm) | 
					
						
							| 
									
										
										
										
											2023-01-10 06:28:20 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-08-07 08:41:28 +02:00
										 |  |  |     Base::initialize(realm); | 
					
						
							| 
									
										
										
										
											2024-03-16 13:13:08 +01:00
										 |  |  |     WEB_SET_PROTOTYPE_FOR_INTERFACE(Storage); | 
					
						
							| 
									
										
										
										
											2023-01-10 06:28:20 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | // https://html.spec.whatwg.org/multipage/webstorage.html#dom-storage-length
 | 
					
						
							|  |  |  | size_t Storage::length() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // The length getter steps are to return this's map's size.
 | 
					
						
							|  |  |  |     return m_map.size(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // https://html.spec.whatwg.org/multipage/webstorage.html#dom-storage-key
 | 
					
						
							| 
									
										
										
										
											2023-08-26 16:30:02 +12:00
										 |  |  | Optional<String> Storage::key(size_t index) | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     // 1. If index is greater than or equal to this's map's size, then return null.
 | 
					
						
							|  |  |  |     if (index >= m_map.size()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 2. Let keys be the result of running get the keys on this's map.
 | 
					
						
							|  |  |  |     auto keys = m_map.keys(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 3. Return keys[index].
 | 
					
						
							|  |  |  |     return keys[index]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // https://html.spec.whatwg.org/multipage/webstorage.html#dom-storage-getitem
 | 
					
						
							| 
									
										
										
										
											2023-08-26 16:30:02 +12:00
										 |  |  | Optional<String> Storage::get_item(StringView key) const | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     // 1. If this's map[key] does not exist, then return null.
 | 
					
						
							|  |  |  |     auto it = m_map.find(key); | 
					
						
							|  |  |  |     if (it == m_map.end()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 2. Return this's map[key].
 | 
					
						
							|  |  |  |     return it->value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // https://html.spec.whatwg.org/multipage/webstorage.html#dom-storage-setitem
 | 
					
						
							| 
									
										
										
										
											2023-08-26 16:30:02 +12:00
										 |  |  | WebIDL::ExceptionOr<void> Storage::set_item(String const& key, String const& value) | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-12-23 19:10:01 +13:00
										 |  |  |     auto& realm = this->realm(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  |     // 1. Let oldValue be null.
 | 
					
						
							| 
									
										
										
										
											2023-08-26 16:30:02 +12:00
										 |  |  |     String old_value; | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // 2. Let reorder be true.
 | 
					
						
							|  |  |  |     bool reorder = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 3. If this's map[key] exists:
 | 
					
						
							| 
									
										
										
										
											2024-12-23 19:10:01 +13:00
										 |  |  |     auto new_size = m_stored_bytes; | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  |     if (auto it = m_map.find(key); it != m_map.end()) { | 
					
						
							|  |  |  |         // 1. Set oldValue to this's map[key].
 | 
					
						
							|  |  |  |         old_value = it->value; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // 2. If oldValue is value, then return.
 | 
					
						
							|  |  |  |         if (old_value == value) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // 3. Set reorder to false.
 | 
					
						
							|  |  |  |         reorder = false; | 
					
						
							| 
									
										
										
										
											2024-12-23 19:10:01 +13:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         new_size += key.bytes().size(); | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-23 19:10:01 +13:00
										 |  |  |     // 4. If value cannot be stored, then throw a "QuotaExceededError" DOMException exception.
 | 
					
						
							|  |  |  |     new_size += value.bytes().size() - old_value.bytes().size(); | 
					
						
							|  |  |  |     if (new_size > m_quota_bytes) | 
					
						
							|  |  |  |         return WebIDL::QuotaExceededError::create(realm, MUST(String::formatted("Unable to store more than {} bytes in storage"sv, m_quota_bytes))); | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // 5. Set this's map[key] to value.
 | 
					
						
							|  |  |  |     m_map.set(key, value); | 
					
						
							| 
									
										
										
										
											2024-12-23 19:10:01 +13:00
										 |  |  |     m_stored_bytes = new_size; | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // 6. If reorder is true, then reorder this.
 | 
					
						
							|  |  |  |     if (reorder) | 
					
						
							|  |  |  |         this->reorder(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 7. Broadcast this with key, oldValue, and value.
 | 
					
						
							|  |  |  |     broadcast(key, old_value, value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // https://html.spec.whatwg.org/multipage/webstorage.html#dom-storage-removeitem
 | 
					
						
							| 
									
										
										
										
											2023-08-26 16:30:02 +12:00
										 |  |  | void Storage::remove_item(StringView key) | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     // 1. If this's map[key] does not exist, then return null.
 | 
					
						
							|  |  |  |     // FIXME: Return null?
 | 
					
						
							|  |  |  |     auto it = m_map.find(key); | 
					
						
							|  |  |  |     if (it == m_map.end()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 2. Set oldValue to this's map[key].
 | 
					
						
							|  |  |  |     auto old_value = it->value; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 3. Remove this's map[key].
 | 
					
						
							|  |  |  |     m_map.remove(it); | 
					
						
							| 
									
										
										
										
											2024-12-23 19:10:01 +13:00
										 |  |  |     m_stored_bytes = m_stored_bytes - key.bytes().size() - old_value.bytes().size(); | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // 4. Reorder this.
 | 
					
						
							|  |  |  |     reorder(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 5. Broadcast this with key, oldValue, and null.
 | 
					
						
							|  |  |  |     broadcast(key, old_value, {}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // https://html.spec.whatwg.org/multipage/webstorage.html#dom-storage-clear
 | 
					
						
							|  |  |  | void Storage::clear() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // 1. Clear this's map.
 | 
					
						
							|  |  |  |     m_map.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 2. Broadcast this with null, null, and null.
 | 
					
						
							|  |  |  |     broadcast({}, {}, {}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // https://html.spec.whatwg.org/multipage/webstorage.html#concept-storage-reorder
 | 
					
						
							|  |  |  | void Storage::reorder() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // To reorder a Storage object storage, reorder storage's map's entries in an implementation-defined manner.
 | 
					
						
							|  |  |  |     // NOTE: This basically means that we're not required to maintain any particular iteration order.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // https://html.spec.whatwg.org/multipage/webstorage.html#concept-storage-broadcast
 | 
					
						
							| 
									
										
										
										
											2023-08-26 16:30:02 +12:00
										 |  |  | void Storage::broadcast(StringView key, StringView old_value, StringView new_value) | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     (void)key; | 
					
						
							|  |  |  |     (void)old_value; | 
					
						
							|  |  |  |     (void)new_value; | 
					
						
							|  |  |  |     // FIXME: Implement.
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-24 20:59:00 +01:00
										 |  |  | Vector<FlyString> Storage::supported_property_names() const | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     // The supported property names on a Storage object storage are the result of running get the keys on storage's map.
 | 
					
						
							| 
									
										
										
										
											2023-12-24 20:59:00 +01:00
										 |  |  |     Vector<FlyString> names; | 
					
						
							| 
									
										
										
										
											2023-08-26 16:30:02 +12:00
										 |  |  |     names.ensure_capacity(m_map.size()); | 
					
						
							|  |  |  |     for (auto const& key : m_map.keys()) | 
					
						
							| 
									
										
										
										
											2023-11-21 07:39:58 +13:00
										 |  |  |         names.unchecked_append(key); | 
					
						
							| 
									
										
										
										
											2023-08-26 16:30:02 +12:00
										 |  |  |     return names; | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-18 00:04:35 +02:00
										 |  |  | Optional<JS::Value> Storage::item_value(size_t index) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Handle index as a string since that's our key type
 | 
					
						
							|  |  |  |     auto key = String::number(index); | 
					
						
							| 
									
										
										
										
											2024-10-23 13:25:30 +02:00
										 |  |  |     auto value = get_item(key); | 
					
						
							|  |  |  |     if (!value.has_value()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     return JS::PrimitiveString::create(vm(), value.release_value()); | 
					
						
							| 
									
										
										
										
											2024-10-18 00:04:35 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-25 17:36:10 +12:00
										 |  |  | JS::Value Storage::named_item_value(FlyString const& name) const | 
					
						
							| 
									
										
										
										
											2023-02-28 00:06:10 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     auto value = get_item(name); | 
					
						
							| 
									
										
										
										
											2023-08-26 16:30:02 +12:00
										 |  |  |     if (!value.has_value()) | 
					
						
							| 
									
										
										
										
											2024-10-20 20:40:17 +01:00
										 |  |  |         // AD-HOC: Spec leaves open to a description at: https://html.spec.whatwg.org/multipage/webstorage.html#the-storage-interface
 | 
					
						
							|  |  |  |         // However correct behavior expected here: https://github.com/whatwg/html/issues/8684
 | 
					
						
							|  |  |  |         return JS::js_undefined(); | 
					
						
							| 
									
										
										
										
											2023-08-26 16:30:02 +12:00
										 |  |  |     return JS::PrimitiveString::create(vm(), value.release_value()); | 
					
						
							| 
									
										
										
										
											2023-02-28 00:06:10 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-09 16:05:03 -07:00
										 |  |  | WebIDL::ExceptionOr<Bindings::PlatformObject::DidDeletionFail> Storage::delete_value(String const& name) | 
					
						
							| 
									
										
										
										
											2023-02-28 00:06:10 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     remove_item(name); | 
					
						
							|  |  |  |     return DidDeletionFail::NotRelevant; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-18 00:04:35 +02:00
										 |  |  | WebIDL::ExceptionOr<void> Storage::set_value_of_indexed_property(u32 index, JS::Value unconverted_value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Handle index as a string since that's our key type
 | 
					
						
							|  |  |  |     auto key = String::number(index); | 
					
						
							|  |  |  |     return set_value_of_named_property(key, unconverted_value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-21 12:15:56 +13:00
										 |  |  | WebIDL::ExceptionOr<void> Storage::set_value_of_named_property(String const& key, JS::Value unconverted_value) | 
					
						
							| 
									
										
										
										
											2023-02-28 00:06:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-01-09 16:05:03 -07:00
										 |  |  |     // NOTE: Since PlatformObject does not know the type of value, we must convert it ourselves.
 | 
					
						
							| 
									
										
										
										
											2023-02-28 00:06:10 +00:00
										 |  |  |     //       The type of `value` is `DOMString`.
 | 
					
						
							| 
									
										
										
										
											2023-08-26 16:30:02 +12:00
										 |  |  |     auto value = TRY(unconverted_value.to_string(vm())); | 
					
						
							| 
									
										
										
										
											2023-11-21 12:15:56 +13:00
										 |  |  |     return set_item(key, value); | 
					
						
							| 
									
										
										
										
											2023-02-28 00:06:10 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-08 19:50:14 +01:00
										 |  |  | void Storage::dump() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     dbgln("Storage ({} key(s))", m_map.size()); | 
					
						
							|  |  |  |     size_t i = 0; | 
					
						
							|  |  |  |     for (auto const& it : m_map) { | 
					
						
							|  |  |  |         dbgln("[{}] \"{}\": \"{}\"", i, it.key, it.value); | 
					
						
							|  |  |  |         ++i; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-08 19:38:29 +01:00
										 |  |  | } |