mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-25 05:32:32 +00:00 
			
		
		
		
	 8b50009e9b
			
		
	
	
		8b50009e9b
		
	
	
	
	
		
			
			This makes test-js style runners dump out output in the same format as libjs-test262's per-file output.
		
			
				
	
	
		
			299 lines
		
	
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			299 lines
		
	
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2020, Matthew Olsson <mattco@serenityos.org>
 | |
|  * Copyright (c) 2020-2021, Linus Groh <linusg@serenityos.org>
 | |
|  * Copyright (c) 2021, Ali Mohammad Pur <mpfard@serenityos.org>
 | |
|  * Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/Format.h>
 | |
| #include <AK/JsonObject.h>
 | |
| #include <AK/JsonValue.h>
 | |
| #include <AK/QuickSort.h>
 | |
| #include <AK/String.h>
 | |
| #include <AK/Vector.h>
 | |
| #include <LibCore/DirIterator.h>
 | |
| #include <LibTest/Results.h>
 | |
| #include <fcntl.h>
 | |
| #include <sys/time.h>
 | |
| 
 | |
| namespace Test {
 | |
| 
 | |
| class TestRunner {
 | |
| public:
 | |
|     static TestRunner* the()
 | |
|     {
 | |
|         return s_the;
 | |
|     }
 | |
| 
 | |
|     TestRunner(String test_root, bool print_times, bool print_progress, bool print_json, bool detailed_json = false)
 | |
|         : m_test_root(move(test_root))
 | |
|         , m_print_times(print_times)
 | |
|         , m_print_progress(print_progress)
 | |
|         , m_print_json(print_json)
 | |
|         , m_detailed_json(detailed_json)
 | |
|     {
 | |
|         VERIFY(!s_the);
 | |
|         s_the = this;
 | |
|     }
 | |
| 
 | |
|     virtual ~TestRunner() { s_the = nullptr; };
 | |
| 
 | |
|     virtual void run(String test_glob);
 | |
| 
 | |
|     const Test::Counts& counts() const { return m_counts; }
 | |
| 
 | |
|     bool is_printing_progress() const { return m_print_progress; }
 | |
| 
 | |
|     bool needs_detailed_suites() const { return m_detailed_json; }
 | |
|     Vector<Test::Suite> const& suites() const { return *m_suites; }
 | |
| 
 | |
|     Vector<Test::Suite>& ensure_suites()
 | |
|     {
 | |
|         if (!m_suites.has_value())
 | |
|             m_suites = Vector<Suite> {};
 | |
|         return *m_suites;
 | |
|     }
 | |
| 
 | |
| protected:
 | |
|     static TestRunner* s_the;
 | |
| 
 | |
|     void print_test_results() const;
 | |
|     void print_test_results_as_json() const;
 | |
| 
 | |
|     virtual Vector<String> get_test_paths() const = 0;
 | |
|     virtual void do_run_single_test(const String&, size_t current_test_index, size_t num_tests) = 0;
 | |
|     virtual const Vector<String>* get_failed_test_names() const { return nullptr; }
 | |
| 
 | |
|     String m_test_root;
 | |
|     bool m_print_times;
 | |
|     bool m_print_progress;
 | |
|     bool m_print_json;
 | |
|     bool m_detailed_json;
 | |
| 
 | |
|     double m_total_elapsed_time_in_ms { 0 };
 | |
|     Test::Counts m_counts;
 | |
|     Optional<Vector<Test::Suite>> m_suites;
 | |
| };
 | |
| 
 | |
| inline void cleanup()
 | |
| {
 | |
|     // Clear the taskbar progress.
 | |
|     if (TestRunner::the() && TestRunner::the()->is_printing_progress())
 | |
|         warn("\033]9;-1;\033\\");
 | |
| }
 | |
| 
 | |
| [[noreturn]] inline void cleanup_and_exit()
 | |
| {
 | |
|     cleanup();
 | |
|     exit(1);
 | |
| }
 | |
| 
 | |
| inline double get_time_in_ms()
 | |
| {
 | |
|     struct timeval tv1;
 | |
|     auto return_code = gettimeofday(&tv1, nullptr);
 | |
|     VERIFY(return_code >= 0);
 | |
|     return static_cast<double>(tv1.tv_sec) * 1000.0 + static_cast<double>(tv1.tv_usec) / 1000.0;
 | |
| }
 | |
| 
 | |
| template<typename Callback>
 | |
| inline void iterate_directory_recursively(const String& directory_path, Callback callback)
 | |
| {
 | |
|     Core::DirIterator directory_iterator(directory_path, Core::DirIterator::Flags::SkipDots);
 | |
| 
 | |
|     while (directory_iterator.has_next()) {
 | |
|         auto name = directory_iterator.next_path();
 | |
|         struct stat st = {};
 | |
|         if (fstatat(directory_iterator.fd(), name.characters(), &st, AT_SYMLINK_NOFOLLOW) < 0)
 | |
|             continue;
 | |
|         bool is_directory = S_ISDIR(st.st_mode);
 | |
|         auto full_path = String::formatted("{}/{}", directory_path, name);
 | |
|         if (is_directory && name != "/Fixtures"sv) {
 | |
|             iterate_directory_recursively(full_path, callback);
 | |
|         } else if (!is_directory) {
 | |
|             callback(full_path);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| inline void TestRunner::run(String test_glob)
 | |
| {
 | |
|     size_t progress_counter = 0;
 | |
|     auto test_paths = get_test_paths();
 | |
|     for (auto& path : test_paths) {
 | |
|         if (!path.matches(test_glob))
 | |
|             continue;
 | |
|         ++progress_counter;
 | |
|         do_run_single_test(path, progress_counter, test_paths.size());
 | |
|         if (m_print_progress)
 | |
|             warn("\033]9;{};{};\033\\", progress_counter, test_paths.size());
 | |
|     }
 | |
| 
 | |
|     if (m_print_progress)
 | |
|         warn("\033]9;-1;\033\\");
 | |
| 
 | |
|     if (!m_print_json)
 | |
|         print_test_results();
 | |
|     else
 | |
|         print_test_results_as_json();
 | |
| }
 | |
| 
 | |
| enum Modifier {
 | |
|     BG_RED,
 | |
|     BG_GREEN,
 | |
|     FG_RED,
 | |
|     FG_GREEN,
 | |
|     FG_ORANGE,
 | |
|     FG_GRAY,
 | |
|     FG_BLACK,
 | |
|     FG_BOLD,
 | |
|     ITALIC,
 | |
|     CLEAR,
 | |
| };
 | |
| 
 | |
| inline void print_modifiers(Vector<Modifier> modifiers)
 | |
| {
 | |
|     for (auto& modifier : modifiers) {
 | |
|         auto code = [&] {
 | |
|             switch (modifier) {
 | |
|             case BG_RED:
 | |
|                 return "\033[48;2;255;0;102m";
 | |
|             case BG_GREEN:
 | |
|                 return "\033[48;2;102;255;0m";
 | |
|             case FG_RED:
 | |
|                 return "\033[38;2;255;0;102m";
 | |
|             case FG_GREEN:
 | |
|                 return "\033[38;2;102;255;0m";
 | |
|             case FG_ORANGE:
 | |
|                 return "\033[38;2;255;102;0m";
 | |
|             case FG_GRAY:
 | |
|                 return "\033[38;2;135;139;148m";
 | |
|             case FG_BLACK:
 | |
|                 return "\033[30m";
 | |
|             case FG_BOLD:
 | |
|                 return "\033[1m";
 | |
|             case ITALIC:
 | |
|                 return "\033[3m";
 | |
|             case CLEAR:
 | |
|                 return "\033[0m";
 | |
|             }
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }();
 | |
|         out("{}", code);
 | |
|     }
 | |
| }
 | |
| 
 | |
| inline void TestRunner::print_test_results() const
 | |
| {
 | |
|     out("\nTest Suites: ");
 | |
|     if (m_counts.suites_failed) {
 | |
|         print_modifiers({ FG_RED });
 | |
|         out("{} failed, ", m_counts.suites_failed);
 | |
|         print_modifiers({ CLEAR });
 | |
|     }
 | |
|     if (m_counts.suites_passed) {
 | |
|         print_modifiers({ FG_GREEN });
 | |
|         out("{} passed, ", m_counts.suites_passed);
 | |
|         print_modifiers({ CLEAR });
 | |
|     }
 | |
|     outln("{} total", m_counts.suites_failed + m_counts.suites_passed);
 | |
| 
 | |
|     out("Tests:       ");
 | |
|     if (m_counts.tests_failed) {
 | |
|         print_modifiers({ FG_RED });
 | |
|         out("{} failed, ", m_counts.tests_failed);
 | |
|         print_modifiers({ CLEAR });
 | |
|     }
 | |
|     if (m_counts.tests_skipped) {
 | |
|         print_modifiers({ FG_ORANGE });
 | |
|         out("{} skipped, ", m_counts.tests_skipped);
 | |
|         print_modifiers({ CLEAR });
 | |
|     }
 | |
|     if (m_counts.tests_passed) {
 | |
|         print_modifiers({ FG_GREEN });
 | |
|         out("{} passed, ", m_counts.tests_passed);
 | |
|         print_modifiers({ CLEAR });
 | |
|     }
 | |
|     outln("{} total", m_counts.tests_failed + m_counts.tests_skipped + m_counts.tests_passed);
 | |
| 
 | |
|     outln("Files:       {} total", m_counts.files_total);
 | |
| 
 | |
|     out("Time:        ");
 | |
|     if (m_total_elapsed_time_in_ms < 1000.0) {
 | |
|         outln("{}ms", static_cast<int>(m_total_elapsed_time_in_ms));
 | |
|     } else {
 | |
|         outln("{:>.3}s", m_total_elapsed_time_in_ms / 1000.0);
 | |
|     }
 | |
|     if (auto* failed_tests = get_failed_test_names(); failed_tests && !failed_tests->is_empty()) {
 | |
|         outln("Failed tests: {}", *failed_tests);
 | |
|     }
 | |
|     outln();
 | |
| }
 | |
| 
 | |
| inline void TestRunner::print_test_results_as_json() const
 | |
| {
 | |
|     JsonObject root;
 | |
|     if (needs_detailed_suites()) {
 | |
|         auto& suites = this->suites();
 | |
|         u64 duration_us = 0;
 | |
|         JsonObject tests;
 | |
| 
 | |
|         for (auto& suite : suites) {
 | |
|             for (auto& case_ : suite.tests) {
 | |
|                 duration_us += case_.duration_us;
 | |
|                 StringView result_name;
 | |
|                 switch (case_.result) {
 | |
|                 case Result::Pass:
 | |
|                     result_name = "PASSED";
 | |
|                     break;
 | |
|                 case Result::Fail:
 | |
|                     result_name = "FAILED";
 | |
|                     break;
 | |
|                 case Result::Skip:
 | |
|                     result_name = "SKIPPED";
 | |
|                     break;
 | |
|                 case Result::Crashed:
 | |
|                     result_name = "PROCESS_ERROR";
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
|                 auto name = suite.name;
 | |
|                 if (name == "__$$TOP_LEVEL$$__"sv)
 | |
|                     name = String::empty();
 | |
| 
 | |
|                 auto path = LexicalPath::relative_path(suite.path, m_test_root);
 | |
| 
 | |
|                 tests.set(String::formatted("{}/{}::{}", path, name, case_.name), result_name);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         root.set("duration", static_cast<double>(duration_us) / 1000000.);
 | |
|         root.set("results", move(tests));
 | |
|     } else {
 | |
|         JsonObject suites;
 | |
|         suites.set("failed", m_counts.suites_failed);
 | |
|         suites.set("passed", m_counts.suites_passed);
 | |
|         suites.set("total", m_counts.suites_failed + m_counts.suites_passed);
 | |
| 
 | |
|         JsonObject tests;
 | |
|         tests.set("failed", m_counts.tests_failed);
 | |
|         tests.set("passed", m_counts.tests_passed);
 | |
|         tests.set("skipped", m_counts.tests_skipped);
 | |
|         tests.set("total", m_counts.tests_failed + m_counts.tests_passed + m_counts.tests_skipped);
 | |
| 
 | |
|         JsonObject results;
 | |
|         results.set("suites", suites);
 | |
|         results.set("tests", tests);
 | |
| 
 | |
|         root.set("results", results);
 | |
|         root.set("files_total", m_counts.files_total);
 | |
|         root.set("duration", m_total_elapsed_time_in_ms / 1000.0);
 | |
|     }
 | |
|     outln("{}", root.to_string());
 | |
| }
 | |
| 
 | |
| }
 |