/* * Copyright (c) 2020, Andreas Kling * Copyright (c) 2022, Linus Groh * Copyright (c) 2025, Shannon Booth * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include namespace URL { // https://html.spec.whatwg.org/multipage/browsers.html#concept-origin class Origin { public: struct OpaqueData { enum class Type : u8 { Standard, File }; using Nonce = Array; Nonce nonce; Type type; }; explicit Origin(OpaqueData opaque_data) : m_state(opaque_data) { } static Origin create_opaque(OpaqueData::Type = OpaqueData::Type::Standard); Origin(Optional const& scheme, Host const& host, Optional port, Optional domain = {}) : m_state(Tuple { .scheme = scheme, .host = host, .port = move(port), .domain = move(domain), }) { } // https://html.spec.whatwg.org/multipage/origin.html#concept-origin-opaque bool is_opaque() const { return m_state.has(); } Optional const& scheme() const { return m_state.get().scheme; } Host const& host() const { return m_state.get().host; } Optional port() const { return m_state.get().port; } Optional domain() const { return m_state.get().domain; } OpaqueData const& opaque_data() const { return m_state.get(); } // https://html.spec.whatwg.org/multipage/origin.html#same-origin bool is_same_origin(Origin const& other) const { // 1. If A and B are the same opaque origin, then return true. if (is_opaque() && other.is_opaque()) return opaque_data().nonce == other.opaque_data().nonce; // 2. If A and B are both tuple origins and their schemes, hosts, and port are identical, then return true. if (!is_opaque() && !other.is_opaque() && scheme() == other.scheme() && host() == other.host() && port() == other.port()) { return true; } // 3. Return false. return false; } // https://html.spec.whatwg.org/multipage/origin.html#same-origin-domain bool is_same_origin_domain(Origin const& other) const { // 1. If A and B are the same opaque origin, then return true. if (is_opaque() && other.is_opaque()) return opaque_data().nonce == other.opaque_data().nonce; // 2. If A and B are both tuple origins, run these substeps: if (!is_opaque() && !other.is_opaque()) { // 1. If A and B's schemes are identical, and their domains are identical and non-null, then return true. if (domain().has_value() && domain() == other.domain() && scheme() == other.scheme()) return true; // 2. Otherwise, if A and B are same origin and their domains are identical and null, then return true. if (!domain().has_value() && !other.domain().has_value() && is_same_origin(other)) return true; } // 3. Return false. return false; } // https://html.spec.whatwg.org/multipage/browsers.html#same-site bool is_same_site(Origin const&) const; // https://html.spec.whatwg.org/multipage/origin.html#ascii-serialisation-of-an-origin String serialize() const; // https://html.spec.whatwg.org/multipage/origin.html#concept-origin-effective-domain Optional effective_domain() const { // 1. If origin is an opaque origin, then return null. if (is_opaque()) return {}; // 2. If origin's domain is non-null, then return origin's domain. auto const& tuple = m_state.get(); if (tuple.domain.has_value()) return Host { tuple.domain.value() }; // 3. Return origin's host. return tuple.host; } bool operator==(Origin const& other) const { return is_same_origin(other); } private: struct Tuple { Optional scheme; Host host; Optional port; Optional domain; }; Variant m_state; }; } namespace AK { template<> struct Traits : public DefaultTraits { static unsigned hash(URL::Origin const& origin); }; } // namespace AK