2020-01-18 09:38:21 +01:00
|
|
|
/*
|
2022-02-25 19:35:51 +01:00
|
|
|
* Copyright (c) 2018-2022, Andreas Kling <kling@serenityos.org>
|
2021-08-05 00:29:06 +08:00
|
|
|
* Copyright (c) 2021, the SerenityOS developers.
|
2020-01-18 09:38:21 +01:00
|
|
|
*
|
2021-04-22 01:24:48 -07:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-01-18 09:38:21 +01:00
|
|
|
*/
|
|
|
|
|
|
2021-04-24 23:53:23 -06:00
|
|
|
#include <LibTest/TestCase.h>
|
2019-08-10 17:27:56 +02:00
|
|
|
|
|
|
|
|
#include <AK/URL.h>
|
2022-12-27 16:17:30 -03:00
|
|
|
#include <AK/URLParser.h>
|
2019-08-10 17:27:56 +02:00
|
|
|
|
|
|
|
|
TEST_CASE(construct)
|
|
|
|
|
{
|
|
|
|
|
EXPECT_EQ(URL().is_valid(), false);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(basic)
|
|
|
|
|
{
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("http://www.serenityos.org"sv);
|
2019-08-10 17:27:56 +02:00
|
|
|
EXPECT_EQ(url.is_valid(), true);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.scheme(), "http");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "www.serenityos.org");
|
2021-09-13 23:12:16 +03:00
|
|
|
EXPECT_EQ(url.port_or_default(), 80);
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.fragment().is_null());
|
2020-04-11 23:38:28 +01:00
|
|
|
}
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("https://www.serenityos.org/index.html"sv);
|
2020-04-11 23:38:28 +01:00
|
|
|
EXPECT_EQ(url.is_valid(), true);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.scheme(), "https");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "www.serenityos.org");
|
2021-09-13 23:12:16 +03:00
|
|
|
EXPECT_EQ(url.port_or_default(), 443);
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/index.html");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.fragment().is_null());
|
2019-08-10 17:27:56 +02:00
|
|
|
}
|
2023-07-25 19:43:00 +12:00
|
|
|
{
|
|
|
|
|
URL url("https://www.serenityos.org1/index.html"sv);
|
|
|
|
|
EXPECT_EQ(url.is_valid(), true);
|
|
|
|
|
EXPECT_EQ(url.scheme(), "https");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "www.serenityos.org1");
|
2023-07-25 19:43:00 +12:00
|
|
|
EXPECT_EQ(url.port_or_default(), 443);
|
|
|
|
|
EXPECT_EQ(url.serialize_path(), "/index.html");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2023-07-25 19:43:00 +12:00
|
|
|
EXPECT(url.fragment().is_null());
|
|
|
|
|
}
|
2019-08-10 17:27:56 +02:00
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("https://localhost:1234/~anon/test/page.html"sv);
|
2019-08-10 17:27:56 +02:00
|
|
|
EXPECT_EQ(url.is_valid(), true);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.scheme(), "https");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "localhost");
|
2021-09-13 23:12:16 +03:00
|
|
|
EXPECT_EQ(url.port_or_default(), 1234);
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/~anon/test/page.html");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.fragment().is_null());
|
2020-04-11 23:38:28 +01:00
|
|
|
}
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("http://www.serenityos.org/index.html?#"sv);
|
2020-04-11 23:38:28 +01:00
|
|
|
EXPECT_EQ(url.is_valid(), true);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.scheme(), "http");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "www.serenityos.org");
|
2021-09-13 23:12:16 +03:00
|
|
|
EXPECT_EQ(url.port_or_default(), 80);
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/index.html");
|
2020-04-11 23:38:28 +01:00
|
|
|
EXPECT_EQ(url.query(), "");
|
|
|
|
|
EXPECT_EQ(url.fragment(), "");
|
|
|
|
|
}
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("http://www.serenityos.org/index.html?foo=1&bar=2"sv);
|
2020-04-11 23:38:28 +01:00
|
|
|
EXPECT_EQ(url.is_valid(), true);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.scheme(), "http");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "www.serenityos.org");
|
2021-09-13 23:12:16 +03:00
|
|
|
EXPECT_EQ(url.port_or_default(), 80);
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/index.html");
|
2020-04-11 23:38:28 +01:00
|
|
|
EXPECT_EQ(url.query(), "foo=1&bar=2");
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.fragment().is_null());
|
2020-04-11 23:38:28 +01:00
|
|
|
}
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("http://www.serenityos.org/index.html#fragment"sv);
|
2020-04-11 23:38:28 +01:00
|
|
|
EXPECT_EQ(url.is_valid(), true);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.scheme(), "http");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "www.serenityos.org");
|
2021-09-13 23:12:16 +03:00
|
|
|
EXPECT_EQ(url.port_or_default(), 80);
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/index.html");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2020-04-11 23:38:28 +01:00
|
|
|
EXPECT_EQ(url.fragment(), "fragment");
|
|
|
|
|
}
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("http://www.serenityos.org/index.html?foo=1&bar=2&baz=/?#frag/ment?test#"sv);
|
2020-04-11 23:38:28 +01:00
|
|
|
EXPECT_EQ(url.is_valid(), true);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.scheme(), "http");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "www.serenityos.org");
|
2021-09-13 23:12:16 +03:00
|
|
|
EXPECT_EQ(url.port_or_default(), 80);
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/index.html");
|
2020-04-11 23:38:28 +01:00
|
|
|
EXPECT_EQ(url.query(), "foo=1&bar=2&baz=/?");
|
|
|
|
|
EXPECT_EQ(url.fragment(), "frag/ment?test#");
|
2019-08-10 17:27:56 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(some_bad_urls)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
EXPECT_EQ(URL("http//serenityos.org"sv).is_valid(), false);
|
|
|
|
|
EXPECT_EQ(URL("serenityos.org"sv).is_valid(), false);
|
|
|
|
|
EXPECT_EQ(URL("://serenityos.org"sv).is_valid(), false);
|
|
|
|
|
EXPECT_EQ(URL("://:80"sv).is_valid(), false);
|
|
|
|
|
EXPECT_EQ(URL("http://serenityos.org:80:80/"sv).is_valid(), false);
|
|
|
|
|
EXPECT_EQ(URL("http://serenityos.org:80:80"sv).is_valid(), false);
|
|
|
|
|
EXPECT_EQ(URL("http://serenityos.org:abc"sv).is_valid(), false);
|
|
|
|
|
EXPECT_EQ(URL("http://serenityos.org:abc:80"sv).is_valid(), false);
|
|
|
|
|
EXPECT_EQ(URL("http://serenityos.org:abc:80/"sv).is_valid(), false);
|
2019-08-10 17:27:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(serialization)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
EXPECT_EQ(URL("http://www.serenityos.org/"sv).serialize(), "http://www.serenityos.org/");
|
|
|
|
|
EXPECT_EQ(URL("http://www.serenityos.org:0/"sv).serialize(), "http://www.serenityos.org:0/");
|
|
|
|
|
EXPECT_EQ(URL("http://www.serenityos.org:80/"sv).serialize(), "http://www.serenityos.org/");
|
|
|
|
|
EXPECT_EQ(URL("http://www.serenityos.org:81/"sv).serialize(), "http://www.serenityos.org:81/");
|
|
|
|
|
EXPECT_EQ(URL("https://www.serenityos.org:443/foo/bar.html?query#fragment"sv).serialize(), "https://www.serenityos.org/foo/bar.html?query#fragment");
|
2019-08-10 17:27:56 +02:00
|
|
|
}
|
|
|
|
|
|
2020-05-09 16:47:05 +02:00
|
|
|
TEST_CASE(file_url_with_hostname)
|
2021-05-29 21:52:13 +02:00
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("file://courage/my/file"sv);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "file");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "courage");
|
2021-09-13 23:12:16 +03:00
|
|
|
EXPECT_EQ(url.port_or_default(), 0);
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/my/file");
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.serialize(), "file://courage/my/file");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.fragment().is_null());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(file_url_with_localhost)
|
2020-05-09 16:47:05 +02:00
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("file://localhost/my/file"sv);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "file");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "");
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/my/file");
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.serialize(), "file:///my/file");
|
2020-05-09 16:47:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(file_url_without_hostname)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("file:///my/file"sv);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "file");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "");
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/my/file");
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.serialize(), "file:///my/file");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(file_url_with_encoded_characters)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("file:///my/file/test%23file.txt"sv);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "file");
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/my/file/test#file.txt");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.fragment().is_null());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(file_url_with_fragment)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("file:///my/file#fragment"sv);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "file");
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/my/file");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.fragment(), "fragment");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(file_url_with_root_path)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("file:///"sv);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "file");
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/");
|
2021-05-29 21:52:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(file_url_serialization)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
EXPECT_EQ(URL("file://courage/my/file"sv).serialize(), "file://courage/my/file");
|
|
|
|
|
EXPECT_EQ(URL("file://localhost/my/file"sv).serialize(), "file:///my/file");
|
|
|
|
|
EXPECT_EQ(URL("file:///my/file"sv).serialize(), "file:///my/file");
|
|
|
|
|
EXPECT_EQ(URL("file:///my/directory/"sv).serialize(), "file:///my/directory/");
|
|
|
|
|
EXPECT_EQ(URL("file:///my/file%23test"sv).serialize(), "file:///my/file%23test");
|
|
|
|
|
EXPECT_EQ(URL("file:///my/file#fragment"sv).serialize(), "file:///my/file#fragment");
|
2020-05-09 16:47:05 +02:00
|
|
|
}
|
|
|
|
|
|
2023-06-18 02:43:11 +03:00
|
|
|
TEST_CASE(file_url_relative)
|
|
|
|
|
{
|
|
|
|
|
EXPECT_EQ(URL("https://vkoskiv.com/index.html"sv).complete_url("/static/foo.js"sv).serialize(), "https://vkoskiv.com/static/foo.js");
|
|
|
|
|
EXPECT_EQ(URL("file:///home/vkoskiv/test/index.html"sv).complete_url("/static/foo.js"sv).serialize(), "file:///home/vkoskiv/test/static/foo.js");
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-10 11:11:48 +02:00
|
|
|
TEST_CASE(about_url)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("about:blank"sv);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "about");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT(url.host().has<Empty>());
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "blank");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.fragment().is_null());
|
|
|
|
|
EXPECT_EQ(url.serialize(), "about:blank");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(mailto_url)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("mailto:mail@example.com"sv);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "mailto");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT(url.host().has<Empty>());
|
2021-09-13 23:12:16 +03:00
|
|
|
EXPECT_EQ(url.port_or_default(), 0);
|
2023-04-13 23:29:51 +01:00
|
|
|
EXPECT_EQ(url.path_segment_count(), 1u);
|
|
|
|
|
EXPECT_EQ(url.path_segment_at_index(0), "mail@example.com");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.fragment().is_null());
|
|
|
|
|
EXPECT_EQ(url.serialize(), "mailto:mail@example.com");
|
2020-05-10 11:11:48 +02:00
|
|
|
}
|
|
|
|
|
|
2023-07-06 19:13:42 +02:00
|
|
|
TEST_CASE(mailto_url_with_subject)
|
|
|
|
|
{
|
|
|
|
|
URL url("mailto:mail@example.com?subject=test"sv);
|
|
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "mailto");
|
|
|
|
|
EXPECT(url.host().has<Empty>());
|
|
|
|
|
EXPECT_EQ(url.port_or_default(), 0);
|
|
|
|
|
EXPECT_EQ(url.path_segment_count(), 1u);
|
|
|
|
|
EXPECT_EQ(url.path_segment_at_index(0), "mail@example.com");
|
|
|
|
|
EXPECT_EQ(url.query(), "subject=test");
|
|
|
|
|
EXPECT(url.fragment().is_null());
|
|
|
|
|
EXPECT_EQ(url.serialize(), "mailto:mail@example.com?subject=test");
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-04 06:20:20 +00:00
|
|
|
TEST_CASE(data_url)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("data:text/html,test"sv);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "data");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT(url.host().has<Empty>());
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.serialize(), "data:text/html,test");
|
AK: Decode data URLs to separate class (and parse like every other URL)
Parsing 'data:' URLs took it's own route. It never set standard URL
fields like path, query or fragment (except for scheme) and instead
gave us separate methods called `data_payload()`, `data_mime_type()`,
and `data_payload_is_base64()`.
Because parsing 'data:' didn't use standard fields, running the
following JS code:
new URL('#a', 'data:text/plain,hello').toString()
not only cleared the path as URLParser doesn't check for data from
data_payload() function (making the result be 'data:#a'), but it also
crashes the program because we forbid having an empty MIME type when we
serialize to string.
With this change, 'data:' URLs will be parsed like every other URLs.
To decode the 'data:' URL contents, one needs to call process_data_url()
on a URL, which will return a struct containing MIME type with already
decoded data! :^)
2023-07-06 19:11:58 +02:00
|
|
|
|
|
|
|
|
auto data_url = TRY_OR_FAIL(url.process_data_url());
|
|
|
|
|
EXPECT_EQ(data_url.mime_type, "text/html");
|
|
|
|
|
EXPECT_EQ(StringView(data_url.body.bytes()), "test"sv);
|
2020-11-04 06:20:20 +00:00
|
|
|
}
|
|
|
|
|
|
2021-08-05 00:29:06 +08:00
|
|
|
TEST_CASE(data_url_default_mime_type)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("data:,test"sv);
|
2021-08-05 00:29:06 +08:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "data");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT(url.host().has<Empty>());
|
AK: Decode data URLs to separate class (and parse like every other URL)
Parsing 'data:' URLs took it's own route. It never set standard URL
fields like path, query or fragment (except for scheme) and instead
gave us separate methods called `data_payload()`, `data_mime_type()`,
and `data_payload_is_base64()`.
Because parsing 'data:' didn't use standard fields, running the
following JS code:
new URL('#a', 'data:text/plain,hello').toString()
not only cleared the path as URLParser doesn't check for data from
data_payload() function (making the result be 'data:#a'), but it also
crashes the program because we forbid having an empty MIME type when we
serialize to string.
With this change, 'data:' URLs will be parsed like every other URLs.
To decode the 'data:' URL contents, one needs to call process_data_url()
on a URL, which will return a struct containing MIME type with already
decoded data! :^)
2023-07-06 19:11:58 +02:00
|
|
|
EXPECT_EQ(url.serialize(), "data:,test");
|
|
|
|
|
|
|
|
|
|
auto data_url = TRY_OR_FAIL(url.process_data_url());
|
|
|
|
|
EXPECT_EQ(data_url.mime_type, "text/plain;charset=US-ASCII");
|
|
|
|
|
EXPECT_EQ(StringView(data_url.body.bytes()), "test"sv);
|
2021-08-05 00:29:06 +08:00
|
|
|
}
|
|
|
|
|
|
2020-11-30 00:28:27 +00:00
|
|
|
TEST_CASE(data_url_encoded)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("data:text/html,Hello%20friends%2C%0X%X0"sv);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "data");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT(url.host().has<Empty>());
|
AK: Decode data URLs to separate class (and parse like every other URL)
Parsing 'data:' URLs took it's own route. It never set standard URL
fields like path, query or fragment (except for scheme) and instead
gave us separate methods called `data_payload()`, `data_mime_type()`,
and `data_payload_is_base64()`.
Because parsing 'data:' didn't use standard fields, running the
following JS code:
new URL('#a', 'data:text/plain,hello').toString()
not only cleared the path as URLParser doesn't check for data from
data_payload() function (making the result be 'data:#a'), but it also
crashes the program because we forbid having an empty MIME type when we
serialize to string.
With this change, 'data:' URLs will be parsed like every other URLs.
To decode the 'data:' URL contents, one needs to call process_data_url()
on a URL, which will return a struct containing MIME type with already
decoded data! :^)
2023-07-06 19:11:58 +02:00
|
|
|
EXPECT_EQ(url.serialize(), "data:text/html,Hello%20friends%2C%0X%X0");
|
|
|
|
|
|
|
|
|
|
auto data_url = TRY_OR_FAIL(url.process_data_url());
|
|
|
|
|
EXPECT_EQ(data_url.mime_type, "text/html");
|
|
|
|
|
EXPECT_EQ(StringView(data_url.body.bytes()), "Hello friends,%0X%X0"sv);
|
2020-11-30 00:28:27 +00:00
|
|
|
}
|
|
|
|
|
|
2020-11-04 06:20:20 +00:00
|
|
|
TEST_CASE(data_url_base64_encoded)
|
|
|
|
|
{
|
AK: Decode data URLs to separate class (and parse like every other URL)
Parsing 'data:' URLs took it's own route. It never set standard URL
fields like path, query or fragment (except for scheme) and instead
gave us separate methods called `data_payload()`, `data_mime_type()`,
and `data_payload_is_base64()`.
Because parsing 'data:' didn't use standard fields, running the
following JS code:
new URL('#a', 'data:text/plain,hello').toString()
not only cleared the path as URLParser doesn't check for data from
data_payload() function (making the result be 'data:#a'), but it also
crashes the program because we forbid having an empty MIME type when we
serialize to string.
With this change, 'data:' URLs will be parsed like every other URLs.
To decode the 'data:' URL contents, one needs to call process_data_url()
on a URL, which will return a struct containing MIME type with already
decoded data! :^)
2023-07-06 19:11:58 +02:00
|
|
|
URL url("data:text/html;base64,dGVzdA=="sv);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "data");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT(url.host().has<Empty>());
|
AK: Decode data URLs to separate class (and parse like every other URL)
Parsing 'data:' URLs took it's own route. It never set standard URL
fields like path, query or fragment (except for scheme) and instead
gave us separate methods called `data_payload()`, `data_mime_type()`,
and `data_payload_is_base64()`.
Because parsing 'data:' didn't use standard fields, running the
following JS code:
new URL('#a', 'data:text/plain,hello').toString()
not only cleared the path as URLParser doesn't check for data from
data_payload() function (making the result be 'data:#a'), but it also
crashes the program because we forbid having an empty MIME type when we
serialize to string.
With this change, 'data:' URLs will be parsed like every other URLs.
To decode the 'data:' URL contents, one needs to call process_data_url()
on a URL, which will return a struct containing MIME type with already
decoded data! :^)
2023-07-06 19:11:58 +02:00
|
|
|
EXPECT_EQ(url.serialize(), "data:text/html;base64,dGVzdA==");
|
|
|
|
|
|
|
|
|
|
auto data_url = TRY_OR_FAIL(url.process_data_url());
|
|
|
|
|
EXPECT_EQ(data_url.mime_type, "text/html");
|
|
|
|
|
EXPECT_EQ(StringView(data_url.body.bytes()), "test"sv);
|
2020-11-04 06:20:20 +00:00
|
|
|
}
|
|
|
|
|
|
2021-08-05 00:29:06 +08:00
|
|
|
TEST_CASE(data_url_base64_encoded_default_mime_type)
|
|
|
|
|
{
|
AK: Decode data URLs to separate class (and parse like every other URL)
Parsing 'data:' URLs took it's own route. It never set standard URL
fields like path, query or fragment (except for scheme) and instead
gave us separate methods called `data_payload()`, `data_mime_type()`,
and `data_payload_is_base64()`.
Because parsing 'data:' didn't use standard fields, running the
following JS code:
new URL('#a', 'data:text/plain,hello').toString()
not only cleared the path as URLParser doesn't check for data from
data_payload() function (making the result be 'data:#a'), but it also
crashes the program because we forbid having an empty MIME type when we
serialize to string.
With this change, 'data:' URLs will be parsed like every other URLs.
To decode the 'data:' URL contents, one needs to call process_data_url()
on a URL, which will return a struct containing MIME type with already
decoded data! :^)
2023-07-06 19:11:58 +02:00
|
|
|
URL url("data:;base64,dGVzdA=="sv);
|
2021-08-05 00:29:06 +08:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "data");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT(url.host().has<Empty>());
|
AK: Decode data URLs to separate class (and parse like every other URL)
Parsing 'data:' URLs took it's own route. It never set standard URL
fields like path, query or fragment (except for scheme) and instead
gave us separate methods called `data_payload()`, `data_mime_type()`,
and `data_payload_is_base64()`.
Because parsing 'data:' didn't use standard fields, running the
following JS code:
new URL('#a', 'data:text/plain,hello').toString()
not only cleared the path as URLParser doesn't check for data from
data_payload() function (making the result be 'data:#a'), but it also
crashes the program because we forbid having an empty MIME type when we
serialize to string.
With this change, 'data:' URLs will be parsed like every other URLs.
To decode the 'data:' URL contents, one needs to call process_data_url()
on a URL, which will return a struct containing MIME type with already
decoded data! :^)
2023-07-06 19:11:58 +02:00
|
|
|
EXPECT_EQ(url.serialize(), "data:;base64,dGVzdA==");
|
|
|
|
|
|
|
|
|
|
auto data_url = TRY_OR_FAIL(url.process_data_url());
|
|
|
|
|
EXPECT_EQ(data_url.mime_type, "text/plain;charset=US-ASCII");
|
|
|
|
|
EXPECT_EQ(StringView(data_url.body.bytes()), "test"sv);
|
2021-08-05 00:29:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(data_url_base64_encoded_with_whitespace)
|
|
|
|
|
{
|
AK: Decode data URLs to separate class (and parse like every other URL)
Parsing 'data:' URLs took it's own route. It never set standard URL
fields like path, query or fragment (except for scheme) and instead
gave us separate methods called `data_payload()`, `data_mime_type()`,
and `data_payload_is_base64()`.
Because parsing 'data:' didn't use standard fields, running the
following JS code:
new URL('#a', 'data:text/plain,hello').toString()
not only cleared the path as URLParser doesn't check for data from
data_payload() function (making the result be 'data:#a'), but it also
crashes the program because we forbid having an empty MIME type when we
serialize to string.
With this change, 'data:' URLs will be parsed like every other URLs.
To decode the 'data:' URL contents, one needs to call process_data_url()
on a URL, which will return a struct containing MIME type with already
decoded data! :^)
2023-07-06 19:11:58 +02:00
|
|
|
URL url("data: text/html ; bAsE64 , dGVz dA== "sv);
|
2021-08-05 00:29:06 +08:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "data");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT(url.host().has<Empty>());
|
AK: Decode data URLs to separate class (and parse like every other URL)
Parsing 'data:' URLs took it's own route. It never set standard URL
fields like path, query or fragment (except for scheme) and instead
gave us separate methods called `data_payload()`, `data_mime_type()`,
and `data_payload_is_base64()`.
Because parsing 'data:' didn't use standard fields, running the
following JS code:
new URL('#a', 'data:text/plain,hello').toString()
not only cleared the path as URLParser doesn't check for data from
data_payload() function (making the result be 'data:#a'), but it also
crashes the program because we forbid having an empty MIME type when we
serialize to string.
With this change, 'data:' URLs will be parsed like every other URLs.
To decode the 'data:' URL contents, one needs to call process_data_url()
on a URL, which will return a struct containing MIME type with already
decoded data! :^)
2023-07-06 19:11:58 +02:00
|
|
|
EXPECT_EQ(url.serialize(), "data: text/html ; bAsE64 , dGVz dA==");
|
|
|
|
|
|
|
|
|
|
auto data_url = TRY_OR_FAIL(url.process_data_url());
|
|
|
|
|
EXPECT_EQ(data_url.mime_type, "text/html");
|
|
|
|
|
EXPECT_EQ(StringView(data_url.body.bytes()), "test");
|
2021-08-05 00:29:06 +08:00
|
|
|
}
|
|
|
|
|
|
2022-02-25 19:35:51 +01:00
|
|
|
TEST_CASE(data_url_base64_encoded_with_inline_whitespace)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("data:text/javascript;base64,%20ZD%20Qg%0D%0APS%20An%20Zm91cic%0D%0A%207%20"sv);
|
2022-02-25 19:35:51 +01:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "data");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT(url.host().has<Empty>());
|
AK: Decode data URLs to separate class (and parse like every other URL)
Parsing 'data:' URLs took it's own route. It never set standard URL
fields like path, query or fragment (except for scheme) and instead
gave us separate methods called `data_payload()`, `data_mime_type()`,
and `data_payload_is_base64()`.
Because parsing 'data:' didn't use standard fields, running the
following JS code:
new URL('#a', 'data:text/plain,hello').toString()
not only cleared the path as URLParser doesn't check for data from
data_payload() function (making the result be 'data:#a'), but it also
crashes the program because we forbid having an empty MIME type when we
serialize to string.
With this change, 'data:' URLs will be parsed like every other URLs.
To decode the 'data:' URL contents, one needs to call process_data_url()
on a URL, which will return a struct containing MIME type with already
decoded data! :^)
2023-07-06 19:11:58 +02:00
|
|
|
|
|
|
|
|
auto data_url = TRY_OR_FAIL(url.process_data_url());
|
|
|
|
|
EXPECT_EQ(data_url.mime_type, "text/javascript");
|
|
|
|
|
EXPECT_EQ(StringView(data_url.body.bytes()), "d4 = 'four';"sv);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(data_url_completed_with_fragment)
|
|
|
|
|
{
|
|
|
|
|
auto url = URL("data:text/plain,test"sv).complete_url("#a"sv);
|
|
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "data");
|
|
|
|
|
EXPECT_EQ(url.fragment(), "a");
|
|
|
|
|
EXPECT(url.host().has<Empty>());
|
|
|
|
|
|
|
|
|
|
auto data_url = TRY_OR_FAIL(url.process_data_url());
|
|
|
|
|
EXPECT_EQ(data_url.mime_type, "text/plain");
|
|
|
|
|
EXPECT_EQ(StringView(data_url.body.bytes()), "test"sv);
|
2022-02-25 19:35:51 +01:00
|
|
|
}
|
|
|
|
|
|
2020-05-17 21:41:36 +10:00
|
|
|
TEST_CASE(trailing_slash_with_complete_url)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
EXPECT_EQ(URL("http://a/b/"sv).complete_url("c/"sv).serialize(), "http://a/b/c/");
|
|
|
|
|
EXPECT_EQ(URL("http://a/b/"sv).complete_url("c"sv).serialize(), "http://a/b/c");
|
|
|
|
|
EXPECT_EQ(URL("http://a/b"sv).complete_url("c/"sv).serialize(), "http://a/c/");
|
|
|
|
|
EXPECT_EQ(URL("http://a/b"sv).complete_url("c"sv).serialize(), "http://a/c");
|
2020-05-17 21:41:36 +10:00
|
|
|
}
|
|
|
|
|
|
2020-12-12 13:24:19 -03:00
|
|
|
TEST_CASE(trailing_port)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url("http://example.com:8086"sv);
|
2021-09-13 23:12:16 +03:00
|
|
|
EXPECT_EQ(url.port_or_default(), 8086);
|
2020-12-12 13:24:19 -03:00
|
|
|
}
|
|
|
|
|
|
2021-05-29 21:52:13 +02:00
|
|
|
TEST_CASE(port_overflow)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
EXPECT_EQ(URL("http://example.com:123456789/"sv).is_valid(), false);
|
2021-05-29 21:52:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(equality)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
EXPECT(URL("http://serenityos.org"sv).equals("http://serenityos.org#test"sv, URL::ExcludeFragment::Yes));
|
|
|
|
|
EXPECT_EQ(URL("http://example.com/index.html"sv), URL("http://ex%61mple.com/index.html"sv));
|
|
|
|
|
EXPECT_EQ(URL("file:///my/file"sv), URL("file://localhost/my/file"sv));
|
|
|
|
|
EXPECT_NE(URL("http://serenityos.org/index.html"sv), URL("http://serenityos.org/test.html"sv));
|
2021-05-29 21:52:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(create_with_file_scheme)
|
2021-03-17 07:42:46 +00:00
|
|
|
{
|
2021-05-29 21:52:13 +02:00
|
|
|
auto url = URL::create_with_file_scheme("/home/anon/README.md");
|
|
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "file");
|
2021-09-13 23:12:16 +03:00
|
|
|
EXPECT_EQ(url.port_or_default(), 0);
|
2023-04-13 23:29:51 +01:00
|
|
|
EXPECT_EQ(url.path_segment_count(), 3u);
|
|
|
|
|
EXPECT_EQ(url.path_segment_at_index(0), "home");
|
|
|
|
|
EXPECT_EQ(url.path_segment_at_index(1), "anon");
|
|
|
|
|
EXPECT_EQ(url.path_segment_at_index(2), "README.md");
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/home/anon/README.md");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.fragment().is_null());
|
|
|
|
|
|
|
|
|
|
url = URL::create_with_file_scheme("/home/anon/");
|
|
|
|
|
EXPECT(url.is_valid());
|
2023-04-13 23:29:51 +01:00
|
|
|
EXPECT_EQ(url.path_segment_count(), 3u);
|
|
|
|
|
EXPECT_EQ(url.path_segment_at_index(0), "home");
|
|
|
|
|
EXPECT_EQ(url.path_segment_at_index(1), "anon");
|
|
|
|
|
EXPECT_EQ(url.path_segment_at_index(2), "");
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/home/anon/");
|
2021-05-29 21:52:13 +02:00
|
|
|
|
2022-07-11 20:59:08 +00:00
|
|
|
url = URL("file:///home/anon/"sv);
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/home/anon/");
|
2021-05-29 21:52:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(complete_url)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL base_url("http://serenityos.org/index.html#fragment"sv);
|
|
|
|
|
URL url = base_url.complete_url("test.html"sv);
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT(url.is_valid());
|
|
|
|
|
EXPECT_EQ(url.scheme(), "http");
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "serenityos.org");
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/test.html");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2021-05-29 21:52:13 +02:00
|
|
|
EXPECT_EQ(url.cannot_be_a_base_url(), false);
|
|
|
|
|
|
2022-07-11 20:59:08 +00:00
|
|
|
EXPECT(base_url.complete_url("../index.html#fragment"sv).equals(base_url));
|
2021-03-17 07:42:46 +00:00
|
|
|
}
|
2021-06-01 13:11:14 +02:00
|
|
|
|
|
|
|
|
TEST_CASE(leading_whitespace)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url { " https://foo.com/"sv };
|
2021-06-01 13:11:14 +02:00
|
|
|
EXPECT(url.is_valid());
|
2022-12-06 01:12:49 +00:00
|
|
|
EXPECT_EQ(url.to_deprecated_string(), "https://foo.com/");
|
2021-06-01 13:11:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(trailing_whitespace)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url { "https://foo.com/ "sv };
|
2021-06-01 13:11:14 +02:00
|
|
|
EXPECT(url.is_valid());
|
2022-12-06 01:12:49 +00:00
|
|
|
EXPECT_EQ(url.to_deprecated_string(), "https://foo.com/");
|
2021-06-01 13:11:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_CASE(leading_and_trailing_whitespace)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url { " https://foo.com/ "sv };
|
2021-06-01 13:11:14 +02:00
|
|
|
EXPECT(url.is_valid());
|
2022-12-06 01:12:49 +00:00
|
|
|
EXPECT_EQ(url.to_deprecated_string(), "https://foo.com/");
|
2021-06-01 13:11:14 +02:00
|
|
|
}
|
2021-06-03 12:36:02 +02:00
|
|
|
|
|
|
|
|
TEST_CASE(unicode)
|
|
|
|
|
{
|
2022-07-11 20:59:08 +00:00
|
|
|
URL url { "http://example.com/_ünicöde_téxt_©"sv };
|
2021-06-03 12:36:02 +02:00
|
|
|
EXPECT(url.is_valid());
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/_ünicöde_téxt_©");
|
2023-08-12 19:28:19 +12:00
|
|
|
EXPECT(!url.query().has_value());
|
2021-06-03 12:36:02 +02:00
|
|
|
EXPECT(url.fragment().is_null());
|
|
|
|
|
}
|
2022-09-20 15:38:53 +02:00
|
|
|
|
|
|
|
|
TEST_CASE(complete_file_url_with_base)
|
|
|
|
|
{
|
|
|
|
|
URL url { "file:///home/index.html" };
|
|
|
|
|
EXPECT(url.is_valid());
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(url.serialize_path(), "/home/index.html");
|
2023-04-13 23:29:51 +01:00
|
|
|
EXPECT_EQ(url.path_segment_count(), 2u);
|
|
|
|
|
EXPECT_EQ(url.path_segment_at_index(0), "home");
|
|
|
|
|
EXPECT_EQ(url.path_segment_at_index(1), "index.html");
|
2022-09-20 15:38:53 +02:00
|
|
|
|
2023-02-13 17:42:27 +00:00
|
|
|
auto sub_url = url.complete_url("js/app.js"sv);
|
2022-09-20 15:38:53 +02:00
|
|
|
EXPECT(sub_url.is_valid());
|
2023-04-14 20:12:03 +01:00
|
|
|
EXPECT_EQ(sub_url.serialize_path(), "/home/js/app.js");
|
2022-09-20 15:38:53 +02:00
|
|
|
}
|
2022-12-27 16:17:30 -03:00
|
|
|
|
|
|
|
|
TEST_CASE(empty_url_with_base_url)
|
|
|
|
|
{
|
|
|
|
|
URL base_url { "https://foo.com/"sv };
|
2023-07-15 14:29:20 +12:00
|
|
|
URL parsed_url = URLParser::basic_parse(""sv, base_url);
|
2022-12-27 16:17:30 -03:00
|
|
|
EXPECT_EQ(parsed_url.is_valid(), true);
|
|
|
|
|
EXPECT(base_url.equals(parsed_url));
|
|
|
|
|
}
|
2023-04-09 14:21:00 +01:00
|
|
|
|
|
|
|
|
TEST_CASE(google_street_view)
|
|
|
|
|
{
|
|
|
|
|
constexpr auto streetview_url = "https://www.google.co.uk/maps/@53.3354159,-1.9573545,3a,75y,121.1h,75.67t/data=!3m7!1e1!3m5!1sSY8xCv17jAX4S7SRdV38hg!2e0!6shttps:%2F%2Fstreetviewpixels-pa.googleapis.com%2Fv1%2Fthumbnail%3Fpanoid%3DSY8xCv17jAX4S7SRdV38hg%26cb_client%3Dmaps_sv.tactile.gps%26w%3D203%26h%3D100%26yaw%3D188.13148%26pitch%3D0%26thumbfov%3D100!7i13312!8i6656";
|
|
|
|
|
URL url(streetview_url);
|
|
|
|
|
EXPECT_EQ(url.serialize(), streetview_url);
|
|
|
|
|
}
|
2023-07-17 06:52:29 +02:00
|
|
|
|
|
|
|
|
TEST_CASE(ipv6_address)
|
|
|
|
|
{
|
|
|
|
|
{
|
|
|
|
|
constexpr auto ipv6_url = "http://[::1]/index.html"sv;
|
|
|
|
|
URL url(ipv6_url);
|
|
|
|
|
EXPECT(url.is_valid());
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "[::1]"sv);
|
2023-07-17 06:52:29 +02:00
|
|
|
EXPECT_EQ(url, ipv6_url);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
constexpr auto ipv6_url = "http://[0:f:0:0:f:f:0:0]/index.html"sv;
|
|
|
|
|
URL url(ipv6_url);
|
|
|
|
|
EXPECT(url.is_valid());
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "[0:f::f:f:0:0]"sv);
|
2023-07-17 06:52:29 +02:00
|
|
|
EXPECT_EQ(url, ipv6_url);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
constexpr auto ipv6_url = "https://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]/index.html"sv;
|
|
|
|
|
URL url(ipv6_url);
|
|
|
|
|
EXPECT(url.is_valid());
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "[2001:db8:85a3::8a2e:370:7334]"sv);
|
2023-07-17 06:52:29 +02:00
|
|
|
EXPECT_EQ(url, ipv6_url);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
constexpr auto bad_ipv6_url = "https://[oops]/index.html"sv;
|
|
|
|
|
URL url(bad_ipv6_url);
|
|
|
|
|
EXPECT_EQ(url.is_valid(), false);
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-07-23 21:09:29 +12:00
|
|
|
|
|
|
|
|
TEST_CASE(ipv4_address)
|
|
|
|
|
{
|
|
|
|
|
{
|
|
|
|
|
constexpr auto ipv4_url = "http://127.0.0.1/index.html"sv;
|
|
|
|
|
URL url(ipv4_url);
|
|
|
|
|
EXPECT(url.is_valid());
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "127.0.0.1"sv);
|
2023-07-23 21:09:29 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
constexpr auto ipv4_url = "http://0x.0x.0"sv;
|
|
|
|
|
URL url(ipv4_url);
|
|
|
|
|
EXPECT(url.is_valid());
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "0.0.0.0"sv);
|
2023-07-23 21:09:29 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
constexpr auto bad_ipv4_url = "https://127..0.0.1"sv;
|
|
|
|
|
URL url(bad_ipv4_url);
|
|
|
|
|
EXPECT(!url.is_valid());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
constexpr auto ipv4_url = "http://256"sv;
|
|
|
|
|
URL url(ipv4_url);
|
|
|
|
|
EXPECT(url.is_valid());
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "0.0.1.0"sv);
|
2023-07-23 21:09:29 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
constexpr auto ipv4_url = "http://888888888"sv;
|
|
|
|
|
URL url(ipv4_url);
|
|
|
|
|
EXPECT(url.is_valid());
|
2023-07-27 21:40:41 +12:00
|
|
|
EXPECT_EQ(MUST(url.serialized_host()), "52.251.94.56"sv);
|
2023-07-23 21:09:29 +12:00
|
|
|
}
|
|
|
|
|
}
|