LibWeb: Implement CookieStore::set(options)

This commit is contained in:
Idan Horowitz 2025-08-06 12:04:18 +03:00 committed by Tim Flynn
commit f724f542ed
Notes: github-actions[bot] 2025-08-08 17:11:52 +00:00
3 changed files with 68 additions and 0 deletions

View file

@ -522,4 +522,48 @@ GC::Ref<WebIDL::Promise> CookieStore::set(String name, String value)
return promise;
}
// https://cookiestore.spec.whatwg.org/#dom-cookiestore-set-options
GC::Ref<WebIDL::Promise> CookieStore::set(CookieInit const& options)
{
auto& realm = this->realm();
// 1. Let settings be thiss relevant settings object.
auto const& settings = HTML::relevant_settings_object(*this);
// 2. Let origin be settingss origin.
auto const& origin = settings.origin();
// 3. If origin is an opaque origin, then return a promise rejected with a "SecurityError" DOMException.
if (origin.is_opaque())
return WebIDL::create_rejected_promise(realm, WebIDL::SecurityError::create(realm, "Document origin is opaque"_string));
// 4. Let url be settingss creation URL.
auto url = settings.creation_url;
// 5. Let p be a new promise.
auto promise = WebIDL::create_promise(realm);
// 6. Run the following steps in parallel:
Platform::EventLoopPlugin::the().deferred_invoke(GC::create_function(realm.heap(), [&realm, client = m_client, promise, url = move(url), options = options]() {
// 1. Let r be the result of running set a cookie with url, options["name"], options["value"], options["expires"],
// options["domain"], options["path"], options["sameSite"], and options["partitioned"].
auto result = set_a_cookie(client, url, options.name, options.value, options.expires, options.domain, options.path, options.same_site, options.partitioned);
// AD-HOC: Queue a global task to perform the next steps
// Spec issue: https://github.com/whatwg/cookiestore/issues/239
queue_global_task(HTML::Task::Source::Unspecified, realm.global_object(), GC::create_function(realm.heap(), [&realm, promise, result]() {
HTML::TemporaryExecutionContext execution_context { realm };
// 2. If r is failure, then reject p with a TypeError and abort these steps.
if (!result)
return WebIDL::reject_promise(realm, promise, JS::TypeError::create(realm, "Name, value, domain or path are malformed"sv));
// 3. Resolve p with undefined.
WebIDL::resolve_promise(realm, promise);
}));
}));
// 7. Return p.
return promise;
}
}

View file

@ -10,6 +10,7 @@
#include <AK/String.h>
#include <LibWeb/Bindings/CookieStorePrototype.h>
#include <LibWeb/DOM/EventTarget.h>
#include <LibWeb/HighResolutionTime/DOMHighResTimeStamp.h>
namespace Web::CookieStore {
@ -25,6 +26,17 @@ struct CookieStoreGetOptions {
Optional<String> url;
};
// https://cookiestore.spec.whatwg.org/#dictdef-cookieinit
struct CookieInit {
String name;
String value;
Optional<HighResolutionTime::DOMHighResTimeStamp> expires;
Optional<String> domain;
String path;
Bindings::CookieSameSite same_site;
bool partitioned { false };
};
// https://cookiestore.spec.whatwg.org/#cookiestore
class CookieStore final : public DOM::EventTarget {
WEB_PLATFORM_OBJECT(CookieStore, DOM::EventTarget);
@ -38,6 +50,7 @@ public:
GC::Ref<WebIDL::Promise> get_all(CookieStoreGetOptions const&);
GC::Ref<WebIDL::Promise> set(String name, String value);
GC::Ref<WebIDL::Promise> set(CookieInit const&);
private:
CookieStore(JS::Realm&, PageClient&);

View file

@ -22,6 +22,16 @@ enum CookieSameSite {
"none"
};
dictionary CookieInit {
required USVString name;
required USVString value;
DOMHighResTimeStamp? expires = null;
USVString? domain = null;
USVString path = "/";
CookieSameSite sameSite = "strict";
boolean partitioned = false;
};
// https://cookiestore.spec.whatwg.org/#cookiestore
[Exposed=(ServiceWorker,Window), SecureContext]
interface CookieStore : EventTarget {
@ -32,6 +42,7 @@ interface CookieStore : EventTarget {
Promise<CookieList> getAll(optional CookieStoreGetOptions options = {});
Promise<undefined> set(USVString name, USVString value);
Promise<undefined> set(CookieInit options);
};
// https://cookiestore.spec.whatwg.org/#Window