/* * Copyright (c) 2022, Ali Mohammad Pur * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include #include static bool g_color = false; static bool g_only_contents = false; enum class ColorRole { PITag, PITarget, PIData, AttributeName, Eq, AttributeValue, Tag, Text, Comment, Reset, Doctype, Keyword, }; static void color(ColorRole role) { if (!g_color) return; switch (role) { case ColorRole::PITag: case ColorRole::Doctype: out("\x1b[{};{}m", 1, "38;5;223"); break; case ColorRole::PITarget: out("\x1b[{};{}m", 1, "38;5;23"); break; case ColorRole::PIData: out("\x1b[{};{}m", 1, "38;5;43"); break; case ColorRole::AttributeName: out("\x1b[38;5;27m"); break; case ColorRole::Eq: break; case ColorRole::AttributeValue: out("\x1b[38;5;46m"); break; case ColorRole::Tag: out("\x1b[{};{}m", 1, "38;5;220"); break; case ColorRole::Text: break; case ColorRole::Comment: out("\x1b[{};{}m", 3, "38;5;250"); break; case ColorRole::Reset: out("\x1b[0m"); break; case ColorRole::Keyword: out("\x1b[38;5;40m"); break; } } static void dump(XML::Node const& node) { node.content.visit( [](XML::Node::Text const& text) { out("{}", text.builder.string_view()); }, [](XML::Node::Comment const& comment) { color(ColorRole::Comment); out("", comment.text); color(ColorRole::Reset); }, [](XML::Node::Element const& element) { color(ColorRole::Tag); out("<{}", element.name); color(ColorRole::Reset); if (!element.attributes.is_empty()) { for (auto& attribute : element.attributes) { auto quote = attribute.value.contains('"') ? '\'' : '"'; color(ColorRole::AttributeName); out(" {}", attribute.key); color(ColorRole::Eq); out("="); color(ColorRole::AttributeValue); out("{}{}{}", quote, attribute.value, quote); color(ColorRole::Reset); } } if (element.children.is_empty()) { color(ColorRole::Tag); out("/>"); color(ColorRole::Reset); } else { color(ColorRole::Tag); out(">"); color(ColorRole::Reset); for (auto& node : element.children) dump(node); color(ColorRole::Tag); out("", element.name); color(ColorRole::Reset); } }); } static void dump(XML::Document& document) { if (!g_only_contents) { { color(ColorRole::PITag); out(""); } for (auto& pi : document.processing_instructions()) { color(ColorRole::PITag); out(""); } if (auto maybe_doctype = document.doctype(); maybe_doctype.has_value()) { auto& doctype = *maybe_doctype; color(ColorRole::Doctype); out(""); }, [&](XML::AttributeListDeclaration const& list) { color(ColorRole::Doctype); out(" "); }, [&](XML::EntityDeclaration const& entity) { color(ColorRole::Doctype); out(" public_literal); } else { color(ColorRole::Keyword); out("SYSTEM "); } color(ColorRole::PITarget); out("\"{}\" ", definition.id.system_id.system_literal); if (definition.notation.has_value()) { color(ColorRole::Keyword); out(" NDATA "); color(ColorRole::PITarget); out("{}", *definition.notation); } }); color(ColorRole::Tag); outln(">"); }, [](XML::PEDeclaration const& declaration) { color(ColorRole::Tag); out("{} ", declaration.name); declaration.definition.visit( [](String const& value) { color(ColorRole::AttributeValue); out("\"{}\"", value); }, [](XML::ExternalID const& id) { if (id.public_id.has_value()) { color(ColorRole::Keyword); out("PUBLIC "); color(ColorRole::PITarget); out("\"{}\" ", id.public_id->public_literal); } else { color(ColorRole::Keyword); out("SYSTEM "); } color(ColorRole::PITarget); out("\"{}\"", id.system_id.system_literal); }); color(ColorRole::Tag); outln(">"); }); }, [&](XML::NotationDeclaration const&) { }); } color(ColorRole::Reset); out("]"); } color(ColorRole::Doctype); outln(">"); } } dump(document.root()); } static String s_path; static auto parse(StringView contents) { return XML::Parser { contents, { .preserve_comments = true, .resolve_external_resource = [&](XML::SystemID const& system_id, Optional const&) -> ErrorOr { auto base = URL::create_with_file_scheme(s_path); auto url = URLParser::parse(system_id.system_literal, &base); if (!url.is_valid()) return Error::from_string_literal("Invalid URL"); if (url.scheme() != "file") return Error::from_string_literal("NYI: Nonlocal entity"); auto file = TRY(Core::File::open(url.path(), Core::OpenMode::ReadOnly)); return String::copy(file->read_all()); }, }, }; } enum class TestResult { Passed, Failed, RunnerFailed, }; static HashMap s_test_results {}; static void do_run_tests(XML::Document& document) { auto& root = document.root().content.get(); VERIFY(root.name == "TESTSUITE"); Queue suites; auto dump_cases = [&](auto& root) { for (auto& node : root.children) { auto element = node.content.template get_pointer(); if (!element) continue; if (element->name != "TESTCASES" && element->name != "TEST") continue; suites.enqueue(&node); } }; dump_cases(root); auto base_path = LexicalPath::dirname(s_path); while (!suites.is_empty()) { auto& node = *suites.dequeue(); auto& suite = node.content.get(); if (suite.name == "TESTCASES") { dump_cases(suite); continue; } if (suite.name == "TEST") { Vector bases; for (auto* parent = node.parent; parent; parent = parent->parent) { auto& attributes = parent->content.get().attributes; auto it = attributes.find("xml:base"); if (it == attributes.end()) continue; bases.append(it->value); } auto type = suite.attributes.find("TYPE")->value; StringBuilder path_builder; path_builder.append(base_path); path_builder.append("/"); for (auto& entry : bases.in_reverse()) { path_builder.append(entry); path_builder.append("/"); } auto test_base_path = path_builder.to_string(); path_builder.append(suite.attributes.find("URI")->value); auto url = URL::create_with_file_scheme(path_builder.string_view()); if (!url.is_valid()) { warnln("Invalid URL {}", path_builder.string_view()); s_test_results.set(path_builder.string_view(), TestResult::RunnerFailed); continue; } auto file_result = Core::File::open(url.path(), Core::OpenMode::ReadOnly); if (file_result.is_error()) { warnln("Read error for {}: {}", url.path(), file_result.error()); s_test_results.set(url.path(), TestResult::RunnerFailed); continue; } warnln("Running test {}", url.path()); auto contents = file_result.value()->read_all(); auto parser = parse(contents); auto doc_or_error = parser.parse(); if (doc_or_error.is_error()) { if (type == "invalid" || type == "error" || type == "not-wf") s_test_results.set(url.path(), TestResult::Passed); else s_test_results.set(url.path(), TestResult::Failed); continue; } auto out = suite.attributes.find("OUTPUT"); if (out != suite.attributes.end()) { auto out_path = LexicalPath::join(test_base_path, out->value).string(); auto file_result = Core::File::open(out_path, Core::OpenMode::ReadOnly); if (file_result.is_error()) { warnln("Read error for {}: {}", out_path, file_result.error()); s_test_results.set(url.path(), TestResult::RunnerFailed); continue; } auto contents = file_result.value()->read_all(); auto parser = parse(contents); auto out_doc_or_error = parser.parse(); if (out_doc_or_error.is_error()) { warnln("Parse error for {}: {}", out_path, out_doc_or_error.error()); s_test_results.set(url.path(), TestResult::RunnerFailed); continue; } auto out_doc = out_doc_or_error.release_value(); if (out_doc.root() != doc_or_error.value().root()) { s_test_results.set(url.path(), TestResult::Failed); continue; } } if (type == "invalid" || type == "error" || type == "not-wf") s_test_results.set(url.path(), TestResult::Failed); else s_test_results.set(url.path(), TestResult::Passed); } } } ErrorOr serenity_main(Main::Arguments arguments) { StringView filename; bool run_tests { false }; Core::ArgsParser parser; parser.set_general_help("Parse and dump XML files"); parser.add_option(g_color, "Syntax highlight the output", "color", 'c'); parser.add_option(g_only_contents, "Only display markup and text", "only-contents", 'o'); parser.add_option(run_tests, "Run tests", "run-tests", 't'); parser.add_positional_argument(filename, "File to read from", "file"); parser.parse(arguments); s_path = Core::File::real_path_for(filename); auto file = TRY(Core::File::open(s_path, Core::OpenMode::ReadOnly)); auto contents = file->read_all(); auto xml_parser = parse(contents); auto result = xml_parser.parse(); if (result.is_error()) { if (xml_parser.parse_error_causes().is_empty()) { warnln("{}", result.error()); } else { warnln("{}; caused by:", result.error()); for (auto const& cause : xml_parser.parse_error_causes()) warnln(" {}", cause); } return 1; } auto doc = result.release_value(); if (run_tests) { do_run_tests(doc); size_t passed = 0; size_t failed = 0; size_t runner_error = 0; size_t total = 0; for (auto& entry : s_test_results) { total++; switch (entry.value) { case TestResult::Passed: passed++; break; case TestResult::Failed: failed++; break; case TestResult::RunnerFailed: runner_error++; break; } } outln("{} passed, {} failed, {} runner failed of {} tests run.", passed, failed, runner_error, total); return 0; } dump(doc); if (!g_only_contents) outln(); return 0; }