GCC Code Coverage Report


Directory: ./
File: modules/util/include/util.hpp
Date: 2026-01-09 01:27:18
Exec Total Coverage
Lines: 40 40 100.0%
Functions: 81 85 95.3%
Branches: 44 75 58.7%

Line Branch Exec Source
1 #pragma once
2
3 #include <algorithm>
4 #include <array>
5 #include <atomic>
6 #include <cctype>
7 #include <cstdint>
8 #include <cstdlib>
9 #include <filesystem>
10 #include <memory>
11 #include <sstream>
12 #include <string>
13 #include <string_view>
14 #include <system_error>
15 #include <typeinfo>
16 #ifdef __GNUG__
17 # include <cxxabi.h>
18 #endif
19
20 #include "nlohmann/json_fwd.hpp"
21
22 #ifdef _MSC_VER
23 # pragma warning(push)
24 # pragma warning(disable : 4459)
25 #endif
26
27 #include <gtest/gtest.h>
28
29 #include <libenvpp/detail/environment.hpp>
30 #include <libenvpp/detail/get.hpp>
31 #include <nlohmann/json.hpp>
32
33 /// @brief JSON namespace used for settings and config parsing.
34 using NlohmannJsonParseError = nlohmann::json::parse_error;
35 /// @brief JSON namespace used for settings and config typing.
36 using NlohmannJsonTypeError = nlohmann::json::type_error;
37 #ifdef _MSC_VER
38 # pragma warning(pop)
39 #endif
40
41 namespace ppc::util {
42
43 /// @brief Utility class for tracking destructor failure across tests.
44 /// @details Provides thread-safe methods to set, unset, and check the failure flag.
45 class DestructorFailureFlag {
46 public:
47 /// @brief Marks that a destructor failure has occurred.
48 static void Set() {
49 failure_flag.store(true);
50 30 }
51
52 /// @brief Clears the destructor failure flag.
53 static void Unset() {
54 failure_flag.store(false);
55 }
56
57 /// @brief Checks if a destructor failure was recorded.
58 /// @return True if failure occurred, false otherwise.
59 static bool Get() {
60 return failure_flag.load();
61 }
62
63 private:
64 inline static std::atomic<bool> failure_flag{false};
65 };
66
67 enum class GTestParamIndex : uint8_t { kTaskGetter, kNameTest, kTestParams };
68
69 std::string GetAbsoluteTaskPath(const std::string &id_path, const std::string &relative_path);
70 int GetNumThreads();
71 int GetNumProc();
72 double GetTaskMaxTime();
73 double GetPerfMaxTime();
74
75 template <typename T>
76
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 14076 times.
27842 std::string GetNamespace() {
77 27842 std::string name = typeid(T).name();
78 #ifdef __GNUC__
79
1/2
✓ Branch 1 taken 14096 times.
✗ Branch 2 not taken.
27842 int status = 0;
80
1/2
✓ Branch 1 taken 14096 times.
✗ Branch 2 not taken.
27842 std::unique_ptr<char, void (*)(void *)> demangled{abi::__cxa_demangle(name.c_str(), nullptr, nullptr, &status),
81 std::free};
82
4/8
✓ Branch 0 taken 14096 times.
✗ Branch 1 not taken.
✓ Branch 3 taken 14096 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 14076 times.
✓ Branch 7 taken 20 times.
✗ Branch 8 not taken.
✗ Branch 9 not taken.
55684 name = (status == 0) ? demangled.get() : name;
83 #endif
84 #ifdef _MSC_VER
85 const std::string prefixes[] = {"class ", "struct ", "enum ", "union "};
86 for (const auto &prefix : prefixes) {
87 if (name.starts_with(prefix)) {
88 name = name.substr(prefix.size());
89 break;
90 }
91 }
92 name.erase(0, name.find_first_not_of(' '));
93 #endif
94 auto pos = name.rfind("::");
95
3/4
✓ Branch 0 taken 14076 times.
✓ Branch 1 taken 20 times.
✓ Branch 3 taken 14076 times.
✗ Branch 4 not taken.
55684 return (pos != std::string::npos) ? name.substr(0, pos) : std::string{};
96 }
97
98 inline std::shared_ptr<nlohmann::json> InitJSONPtr() {
99
2/3
✓ Branch 1 taken 14056 times.
✓ Branch 2 taken 200 times.
✗ Branch 3 not taken.
14316 return std::make_shared<nlohmann::json>();
100 }
101
102 bool IsUnderMpirun();
103
104 namespace test {
105
106 2486 [[nodiscard]] inline std::string SanitizeToken(std::string_view token_sv) {
107 2486 std::string token{token_sv};
108 310146 auto is_allowed = [](char c) {
109
6/6
✓ Branch 0 taken 31040 times.
✓ Branch 1 taken 279106 times.
✓ Branch 2 taken 23582 times.
✓ Branch 3 taken 7458 times.
✓ Branch 4 taken 2486 times.
✓ Branch 5 taken 4972 times.
310146 return std::isalnum(static_cast<unsigned char>(c)) || c == '_' || c == '-' || c == '.';
110 };
111 std::ranges::replace(token, ' ', '_');
112
2/2
✓ Branch 0 taken 310146 times.
✓ Branch 1 taken 2486 times.
312632 for (char &ch : token) {
113
2/2
✓ Branch 0 taken 4972 times.
✓ Branch 1 taken 305174 times.
310146 if (!is_allowed(ch)) {
114 4972 ch = '_';
115 }
116 }
117 2486 return token;
118 }
119
120 2486 class ScopedPerTestEnv {
121 public:
122 2486 explicit ScopedPerTestEnv(const std::string &token)
123
2/4
✓ Branch 2 taken 2486 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 2486 times.
✗ Branch 6 not taken.
2486 : set_uid_("PPC_TEST_UID", token), set_tmp_("PPC_TEST_TMPDIR", CreateTmpDir(token)) {}
124
125 private:
126 2486 static std::string CreateTmpDir(const std::string &token) {
127 namespace fs = std::filesystem;
128 526 auto make_rank_suffix = []() -> std::string {
129 // Derive rank from common MPI env vars without including MPI headers
130 526 constexpr std::array<std::string_view, 5> kRankVars = {"OMPI_COMM_WORLD_RANK", "PMI_RANK", "PMIX_RANK",
131 "SLURM_PROCID", "MSMPI_RANK"};
132
1/2
✓ Branch 0 taken 526 times.
✗ Branch 1 not taken.
526 for (auto name : kRankVars) {
133
2/4
✓ Branch 1 taken 526 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 526 times.
✗ Branch 4 not taken.
526 if (auto r = env::get<int>(name); r.has_value() && r.value() >= 0) {
134
3/8
✓ Branch 2 taken 526 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 526 times.
✗ Branch 6 not taken.
✗ Branch 8 not taken.
✓ Branch 9 taken 526 times.
✗ Branch 11 not taken.
✗ Branch 12 not taken.
1052 return std::string("_rank_") + std::to_string(r.value());
135 }
136 }
137 return std::string{};
138 };
139
2/2
✓ Branch 1 taken 526 times.
✓ Branch 2 taken 1960 times.
2486 const std::string rank_suffix = IsUnderMpirun() ? make_rank_suffix() : std::string{};
140
5/10
✓ Branch 1 taken 2486 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 2486 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 2486 times.
✗ Branch 8 not taken.
✓ Branch 10 taken 2486 times.
✗ Branch 11 not taken.
✓ Branch 15 taken 2486 times.
✗ Branch 16 not taken.
7458 const fs::path tmp = fs::temp_directory_path() / (std::string("ppc_test_") + token + rank_suffix);
141 std::error_code ec;
142
1/2
✓ Branch 1 taken 2486 times.
✗ Branch 2 not taken.
2486 fs::create_directories(tmp, ec);
143 (void)ec;
144 2486 return tmp.string();
145 2486 }
146
147 env::detail::set_scoped_environment_variable set_uid_;
148 env::detail::set_scoped_environment_variable set_tmp_;
149 };
150
151 2486 [[nodiscard]] inline std::string MakeCurrentGTestToken(std::string_view fallback_name) {
152 2486 const auto *unit = ::testing::UnitTest::GetInstance();
153
1/2
✓ Branch 0 taken 2486 times.
✗ Branch 1 not taken.
2486 const auto *info = (unit != nullptr) ? unit->current_test_info() : nullptr;
154 2486 std::ostringstream os;
155
1/2
✓ Branch 0 taken 2486 times.
✗ Branch 1 not taken.
2486 if (info != nullptr) {
156
2/4
✓ Branch 1 taken 2486 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 2486 times.
✗ Branch 5 not taken.
4972 os << info->test_suite_name() << "." << info->name();
157 } else {
158 os << fallback_name;
159 }
160
1/2
✓ Branch 1 taken 2486 times.
✗ Branch 2 not taken.
4972 return SanitizeToken(os.str());
161 2486 }
162
163 2486 inline ScopedPerTestEnv MakePerTestEnvForCurrentGTest(std::string_view fallback_name) {
164
1/2
✓ Branch 2 taken 2486 times.
✗ Branch 3 not taken.
4972 return ScopedPerTestEnv(MakeCurrentGTestToken(fallback_name));
165 }
166
167 } // namespace test
168
169 } // namespace ppc::util
170