mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-26 02:12:07 +00:00 
			
		
		
		
	 8751be09f9
			
		
	
	
		8751be09f9
		
	
	
	
	
		
			
			In order to follow spec text to achieve this, we need to change the underlying representation of a host in AK::URL to deserialized format. Before this, we were parsing the host and then immediately serializing it again. Making that change resulted in a whole bunch of fallout. After this change, callers can access the serialized data through this concept-host-serializer. The functional end result of this change is that IPv6 hosts are now correctly serialized to be surrounded with '[' and ']'.
		
			
				
	
	
		
			131 lines
		
	
	
	
		
			4.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			131 lines
		
	
	
	
		
			4.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
 | |
|  * Copyright (c) 2022, Linus Groh <linusg@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/DeprecatedString.h>
 | |
| #include <AK/URL.h>
 | |
| #include <AK/URLParser.h>
 | |
| 
 | |
| namespace Web::HTML {
 | |
| 
 | |
| class Origin {
 | |
| public:
 | |
|     Origin() = default;
 | |
|     Origin(DeprecatedString const& scheme, AK::URL::Host 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.has<Empty>() && m_port == 0; }
 | |
| 
 | |
|     DeprecatedString const& scheme() const { return m_scheme; }
 | |
|     AK::URL::Host 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
 | |
|     DeprecatedString 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(URLParser::serialize_host(host()).release_value_but_fixme_should_propagate_errors().to_deprecated_string());
 | |
| 
 | |
|         // 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(DeprecatedString::number(port()));
 | |
|         }
 | |
|         // 6. Return result
 | |
|         return result.to_deprecated_string();
 | |
|     }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/origin.html#concept-origin-effective-domain
 | |
|     Optional<AK::URL::Host> effective_domain() const
 | |
|     {
 | |
|         // 1. If origin is an opaque origin, then return null.
 | |
|         if (is_opaque())
 | |
|             return {};
 | |
| 
 | |
|         // 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:
 | |
|     DeprecatedString m_scheme;
 | |
|     AK::URL::Host m_host;
 | |
|     u16 m_port { 0 };
 | |
| };
 | |
| 
 | |
| }
 | |
| 
 | |
| namespace AK {
 | |
| template<>
 | |
| struct Traits<Web::HTML::Origin> : public GenericTraits<Web::HTML::Origin> {
 | |
|     static unsigned hash(Web::HTML::Origin const& origin)
 | |
|     {
 | |
|         auto hash_without_host = pair_int_hash(origin.scheme().hash(), origin.port());
 | |
|         if (origin.host().has<Empty>())
 | |
|             return hash_without_host;
 | |
|         return pair_int_hash(hash_without_host, URLParser::serialize_host(origin.host()).release_value_but_fixme_should_propagate_errors().hash());
 | |
|     }
 | |
| };
 | |
| } // namespace AK
 |