/* * Copyright (c) 2020, Andreas Kling * Copyright (c) 2022, Linus Groh * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include namespace Web::HTML { class Origin { public: Origin() = default; Origin(String const& scheme, String const& host, u16 port) : m_scheme(scheme) , m_host(host) , m_port(port) { } // https://html.spec.whatwg.org/multipage/origin.html#concept-origin-opaque bool is_opaque() const { return m_scheme.is_null() && m_host.is_null() && m_port == 0; } String const& scheme() const { return m_scheme; } String const& host() const { return m_host; } u16 port() const { return m_port; } // 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 true; // 2. If A and B are both tuple origins and their schemes, hosts, and port are identical, then return true. // 3. Return false. return scheme() == other.scheme() && host() == other.host() && port() == other.port(); } // 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 true; // 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. // FIXME: Check domains once supported. if (scheme() == other.scheme()) return true; // 2. Otherwise, if A and B are same origin and their domains are identical and null, then return true. // FIXME: Check domains once supported. if (is_same_origin(other)) return true; } // 3. Return false. return false; } // https://html.spec.whatwg.org/multipage/origin.html#ascii-serialisation-of-an-origin String serialize() const { // 1. If origin is an opaque origin, then return "null" if (is_opaque()) return "null"; // 2. Otherwise, let result be origin's scheme. StringBuilder result; result.append(scheme()); // 3. Append "://" to result. result.append("://"sv); // 4. Append origin's host, serialized, to result. result.append(host()); // 5. If origin's port is non-null, append a U+003A COLON character (:), and origin's port, serialized, to result. if (port() != 0) { result.append(':'); result.append(String::number(port())); } // 6. Return result return result.to_string(); } // 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 Optional {}; // FIXME: 2. If origin's domain is non-null, then return origin's domain. // 3. Return origin's host. return m_host; } bool operator==(Origin const& other) const { return is_same_origin(other); } private: String m_scheme; String m_host; u16 m_port { 0 }; }; } namespace AK { template<> struct Traits : public GenericTraits { static unsigned hash(Web::HTML::Origin const& origin) { return pair_int_hash(origin.scheme().hash(), pair_int_hash(int_hash(origin.port()), origin.host().hash())); } }; } // namespace AK