mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 03:02:45 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			185 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			185 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2023, Cameron Youell <cameronyouell@gmail.com>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include "../LibUnicode/GeneratorUtil.h"
 | |
| #include <AK/SourceGenerator.h>
 | |
| #include <AK/StringBuilder.h>
 | |
| #include <LibCore/ArgsParser.h>
 | |
| #include <LibCore/File.h>
 | |
| #include <LibMain/Main.h>
 | |
| 
 | |
| ErrorOr<void> generate_header_file(Core::InputBufferedFile&, Core::File&);
 | |
| ErrorOr<void> generate_implementation_file(Core::InputBufferedFile&, Core::File&);
 | |
| 
 | |
| ErrorOr<int> serenity_main(Main::Arguments arguments)
 | |
| {
 | |
|     StringView generated_header_path;
 | |
|     StringView generated_implementation_path;
 | |
|     StringView public_suffix_list_path;
 | |
| 
 | |
|     Core::ArgsParser args_parser;
 | |
|     args_parser.add_option(generated_header_path, "Path to the header file to generate", "generated-header-path", 'h', "generated-header-path");
 | |
|     args_parser.add_option(generated_implementation_path, "Path to the implementation file to generate", "generated-implementation-path", 'c', "generated-implementation-path");
 | |
|     args_parser.add_option(public_suffix_list_path, "Path to the public suffix list", "public-suffix-list-path", 'p', "public-suffix-list-path");
 | |
|     args_parser.parse(arguments);
 | |
| 
 | |
|     auto identifier_data = TRY(open_file(public_suffix_list_path, Core::File::OpenMode::Read));
 | |
| 
 | |
|     auto generated_header_file = TRY(Core::File::open(generated_header_path, Core::File::OpenMode::Write));
 | |
|     auto generated_implementation_file = TRY(Core::File::open(generated_implementation_path, Core::File::OpenMode::Write));
 | |
| 
 | |
|     TRY(generate_header_file(*identifier_data, *generated_header_file));
 | |
|     TRY(generate_implementation_file(*identifier_data, *generated_implementation_file));
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| ErrorOr<void> generate_header_file(Core::InputBufferedFile&, Core::File& file)
 | |
| {
 | |
|     StringBuilder builder;
 | |
|     SourceGenerator generator { builder };
 | |
|     generator.append(R"~~~(
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/Forward.h>
 | |
| #include <AK/Trie.h>
 | |
| #include <AK/Variant.h>
 | |
| 
 | |
| namespace WebView {
 | |
| 
 | |
| class PublicSuffixData {
 | |
| protected:
 | |
|     PublicSuffixData();
 | |
| 
 | |
| public:
 | |
|     PublicSuffixData(PublicSuffixData const&) = delete;
 | |
|     PublicSuffixData& operator=(PublicSuffixData const&) = delete;
 | |
| 
 | |
|     static PublicSuffixData* the()
 | |
|     {
 | |
|         static PublicSuffixData* s_the;
 | |
|         if (!s_the)
 | |
|             s_the = new PublicSuffixData;
 | |
|         return s_the;
 | |
|     }
 | |
| 
 | |
|     bool is_public_suffix(StringView host);
 | |
|     ErrorOr<Optional<String>> get_public_suffix(StringView string);
 | |
| 
 | |
| private:
 | |
|     Trie<char, Empty> m_dictionary;
 | |
| };
 | |
| 
 | |
| }
 | |
| 
 | |
| )~~~");
 | |
| 
 | |
|     TRY(file.write_until_depleted(generator.as_string_view().bytes()));
 | |
|     return {};
 | |
| }
 | |
| 
 | |
| ErrorOr<void> generate_implementation_file(Core::InputBufferedFile& input, Core::File& file)
 | |
| {
 | |
|     StringBuilder builder;
 | |
|     SourceGenerator generator { builder };
 | |
|     generator.append(R"~~~(
 | |
| #include <AK/String.h>
 | |
| #include <AK/Vector.h>
 | |
| #include <LibWebView/PublicSuffixData.h>
 | |
| 
 | |
| namespace WebView {
 | |
| 
 | |
| static constexpr auto s_public_suffixes = Array {)~~~");
 | |
| 
 | |
|     Array<u8, 1024> buffer {};
 | |
| 
 | |
|     while (TRY(input.can_read_line())) {
 | |
|         auto line = TRY(input.read_line(buffer));
 | |
| 
 | |
|         if (line.starts_with("//"sv) || line.is_empty())
 | |
|             continue;
 | |
| 
 | |
|         auto view = line.split_view("."sv);
 | |
|         view.reverse();
 | |
| 
 | |
|         StringBuilder builder;
 | |
|         builder.join("."sv, view);
 | |
|         auto val = builder.string_view();
 | |
| 
 | |
|         generator.set("line", val);
 | |
|         generator.append(R"~~~(
 | |
|     "@line@"sv,)~~~");
 | |
|     }
 | |
| 
 | |
|     generator.append(R"~~~(
 | |
| };
 | |
| 
 | |
| PublicSuffixData::PublicSuffixData()
 | |
|     : m_dictionary('/')
 | |
| {
 | |
|     // FIXME: Reduce the depth of this trie
 | |
|     for (auto str : s_public_suffixes) {
 | |
|         MUST(m_dictionary.insert(str.begin(), str.end(), Empty {}, [](auto const&, auto const&) -> Optional<Empty> { return {}; }));
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool PublicSuffixData::is_public_suffix(StringView host)
 | |
| {
 | |
|     auto it = host.begin();
 | |
|     auto& node = m_dictionary.traverse_until_last_accessible_node(it, host.end());
 | |
|     return it.is_end() && node.has_metadata();
 | |
| }
 | |
| 
 | |
| ErrorOr<Optional<String>> PublicSuffixData::get_public_suffix(StringView string)
 | |
| {
 | |
|     auto input = string.split_view("."sv);
 | |
|     input.reverse();
 | |
| 
 | |
|     StringBuilder overall_search_string;
 | |
|     StringBuilder search_string;
 | |
|     for (auto part : input) {
 | |
|         search_string.clear();
 | |
|         TRY(search_string.try_append(TRY(overall_search_string.to_string())));
 | |
|         TRY(search_string.try_append(part));
 | |
| 
 | |
|         if (is_public_suffix(search_string.string_view())) {
 | |
|             overall_search_string.append(TRY(String::from_utf8(part)));
 | |
|             overall_search_string.append("."sv);
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         search_string.clear();
 | |
|         TRY(search_string.try_append(TRY(overall_search_string.to_string())));
 | |
|         TRY(search_string.try_append("*"sv));
 | |
| 
 | |
|         if (is_public_suffix(search_string.string_view())) {
 | |
|             overall_search_string.append(TRY(String::from_utf8(part)));
 | |
|             overall_search_string.append("."sv);
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     auto view = overall_search_string.string_view().split_view("."sv);
 | |
|     view.reverse();
 | |
| 
 | |
|     StringBuilder return_string_builder;
 | |
|     return_string_builder.join('.', view);
 | |
|     auto returnString = TRY(return_string_builder.to_string());
 | |
|     if (!returnString.is_empty())
 | |
|         return returnString;
 | |
| 
 | |
|     return Optional<String> {};
 | |
| }
 | |
| 
 | |
| }
 | |
| 
 | |
| )~~~");
 | |
| 
 | |
|     TRY(file.write_until_depleted(generator.as_string_view().bytes()));
 | |
|     return {};
 | |
| }
 | 
