0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
nested_trait.h
Go to the documentation of this file.
1/**
2 * @file src/common/model/nested_trait.h
3 *
4 * Copyright (c) 2021-2025 Bartek Kryza <bkryza@gmail.com>
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 */
18#pragma once
19
20#include "common/types.h"
21#include "util/util.h"
22
23#include <iostream>
24#include <list>
25#include <optional>
26#include <set>
27#include <string>
28#include <vector>
29
31
32/**
33 * @brief Base class for elements nested in the diagram.
34 *
35 * This class provides a common trait for diagram elements which can contain
36 * other nested elements, e.g. packages.
37 *
38 * @embed{nested_trait_hierarchy_class.svg}
39 *
40 * @tparam T Type of element
41 * @tparam Path Type of nested path (e.g. namespace or directory path)
42 */
43template <typename T, typename Path> class nested_trait {
44public:
45 nested_trait() = default;
46
47 nested_trait(const nested_trait &) = delete;
48 nested_trait(nested_trait &&) noexcept = default;
49
50 nested_trait &operator=(const nested_trait &) = delete;
51 nested_trait &operator=(nested_trait &&) noexcept = default;
52
53 virtual ~nested_trait() = default;
54
55 bool is_root() const { return is_root_; }
56
57 void is_root(bool a) { is_root_ = a; }
58
59 /**
60 * Add element at the current nested level.
61 *
62 * @tparam V Type of element
63 * @param p Element
64 * @return True, if element was added.
65 */
66 template <typename V = T>
67 [[nodiscard]] bool add_element(std::unique_ptr<V> element)
68 {
69 const auto element_id = element->id();
70 const auto element_name = element->name();
71
72 if (elements_by_id_.count(element_id) > 0) {
73 // Element already in element tree
74 return false;
75 }
76
77 elements_.push_back(std::move(element));
78
79 auto element_it = std::prev(elements_.end());
80
81 elements_by_id_.emplace(element_id, element_it);
82 elements_by_name_.emplace(element_name, element_it);
83
84 assert(elements_.size() == elements_by_id_.size());
85 assert(elements_.size() == elements_by_name_.size());
86
87 return true;
88 }
89
90 /**
91 * Add element at a nested path.
92 *
93 * @tparam V Type of element
94 * @param path Nested path (e.g. list of namespaces)
95 * @param p Element
96 * @return True, if element was added.
97 */
98 template <typename V = T>
99 bool add_element(const Path &path, std::unique_ptr<V> p)
100 {
101 assert(p);
102
103 LOG_TRACE("Adding nested element {} at path '{}{}'", p->name(),
104 path.is_root() ? "::" : "", path.to_string());
105
106 if (path.is_empty()) {
107 return add_element(std::move(p));
108 }
109
110 auto parent = get_element(path);
111
112 if (parent) {
113 auto *nested_trait_ptr =
114 dynamic_cast<nested_trait<T, Path> *>(&parent.value());
115 if (nested_trait_ptr != nullptr) {
116 p->set_parent_element_id(parent.value().id());
117 return (*nested_trait_ptr)
118 .template add_element<V>(std::move(p));
119 }
120 }
121
122 LOG_INFO("No parent element found at: {}", path.to_string());
123
124 throw std::runtime_error(
125 "No parent element found for " + path.to_string());
126 }
127
128 /**
129 * Get element at path, if exists.
130 *
131 * @tparam V Element type.
132 * @param path Path to the element.
133 * @return Optional reference to the element.
134 */
135 template <typename V = T> auto get_element(const Path &path) const
136 {
137 if (path.is_empty() || !has_element(path[0])) {
138 LOG_DBG("Nested element {} not found in element", path.to_string());
139 return optional_ref<V>{};
140 }
141
142 if (path.size() == 1) {
143 return get_element<V>(path[0], path.is_root());
144 }
145
146 auto p = get_element<T>(path[0], path.is_root());
147
148 if (!p)
149 return optional_ref<V>{};
150
151 if (dynamic_cast<nested_trait<T, Path> *>(&p.value()))
152 return dynamic_cast<nested_trait<T, Path> &>(p.value())
153 .get_element<V>(Path{path.begin() + 1, path.end()});
154
155 return optional_ref<V>{};
156 }
157
158 /**
159 * Get element by name at the current nested level.
160 *
161 * @tparam V Type of element.
162 * @param name Name of the element (cannot contain namespace or path)
163 * @return Optional reference to the element.
164 */
165 template <typename V = T>
166 auto get_element(const std::string &name, bool is_root = false) const
167 {
168 assert(!util::contains(name, "::"));
169
170 // Try to find an element by name and assuming it is a specific type
171 // For some reason it is legal to have a C/C++ struct with the same
172 // name as some ObjC protocol/interface, so the name is not
173 // necessarily unique
174
175 auto [it, matches_end] = elements_by_name_.equal_range(name);
176
177 while (true) {
178 if (it == matches_end)
179 break;
180
181 auto element_it = it->second;
182 it++;
183
184 if (auto element_ptr = dynamic_cast<V *>(element_it->get());
185 element_ptr != nullptr) {
186 if (auto nt_ptr = dynamic_cast<nested_trait<T, Path> *>(
187 element_it->get());
188 nt_ptr != nullptr) {
189 if (nt_ptr->is_root_ == is_root)
190 return optional_ref<V>{std::ref<V>(*element_ptr)};
191 }
192 else
193 return optional_ref<V>{std::ref<V>(*element_ptr)};
194 }
195 }
196
197 return optional_ref<V>{};
198 }
199
200 /**
201 * Return result of functor f applied to all_of elements.
202 * @tparam F Functor type
203 * @param f Functor value
204 * @return True, if functor return true for elements, including nested
205 * ones.
206 */
207 template <typename F> bool all_of(F &&f) const
208 {
209 return std::all_of(
210 elements_.cbegin(), elements_.cend(), [f](const auto &e) {
211 const auto *package_ptr =
212 dynamic_cast<nested_trait<T, Path> *>(e.get());
213
214 if (package_ptr != nullptr)
215 return package_ptr->all_of(f);
216
217 return f(*e);
218 });
219 }
220
221 /**
222 * Check if nested element is empty.
223 *
224 * @return True if this nested element is empty.
225 */
226 bool is_empty(bool include_inner_packages = false) const
227 {
228 // If we're interested whether the tree contains any elements including
229 // packages (even if their empty) just check elements_
230 if (include_inner_packages)
231 return elements_.empty();
232
233 // If we're interested only in non-package elements, that we have to
234 // traverse the nested chain
235 return elements_.empty() ||
236 std::all_of(elements_.cbegin(), elements_.cend(), [](auto &e) {
237 const auto *package_ptr =
238 dynamic_cast<nested_trait<T, Path> *>(e.get());
239 return package_ptr != nullptr && package_ptr->is_empty();
240 });
241 }
242
243 auto cbegin() const { return elements_.cbegin(); }
244 auto cend() const { return elements_.cend(); }
245
246 auto begin() const { return elements_.begin(); }
247 auto end() const { return elements_.end(); }
248
249 /**
250 * Print the nested trait in the form of a tree.
251 *
252 * This method is used for debugging only.
253 *
254 * @param level Tree level
255 */
256 void print_tree(const int level) const
257 {
258 if (level == 0) {
259 std::cout << "--- Printing tree:\n";
260 }
261
262 for (auto it = elements_.cbegin(); it != elements_.cend(); it++) {
263 const auto &e = *it;
264 if (dynamic_cast<nested_trait<T, Path> *>(e.get())) {
265 std::cout << std::string(level, ' ');
266 std::cout << "[" << *e << "]\n";
267 dynamic_cast<nested_trait<T, Path> *>(e.get())->print_tree(
268 level + 2);
269 }
270 else {
271 std::cout << std::string(level, ' ');
272 std::cout << "- " << *e << "]\n";
273 }
274 }
275 }
276
277 template <typename V = T> std::unique_ptr<V> get_and_remove(eid_t id)
278 {
279 std::unique_ptr<V> result;
280
281 auto id_it = elements_by_id_.find(id);
282 if (id_it == elements_by_id_.end()) {
283 for (auto &p : elements_) {
284 if (dynamic_cast<nested_trait<T, Path> *>(p.get())) {
285 result = dynamic_cast<nested_trait<T, Path> *>(p.get())
286 ->get_and_remove<V>(id);
287 if (result)
288 break;
289 }
290 }
291 return result;
292 }
293
294 // Get the iterator into the elements_ storage.
295 auto element_it = id_it->second;
296
297 // Remove from the name index.
298 auto name = (*element_it)->name();
299 auto name_range = elements_by_name_.equal_range(name);
300 for (auto it = name_range.first; it != name_range.second; ++it) {
301 if (it->second == element_it) {
302 elements_by_name_.erase(it);
303 break;
304 }
305 }
306
307 elements_by_id_.erase(id);
308
309 result = util::unique_pointer_cast<V>(std::move(*element_it));
310
311 elements_.erase(element_it);
312
313 assert(result);
314
315 assert(elements_.size() == elements_by_id_.size());
316 assert(elements_.size() == elements_by_name_.size());
317
318 return result;
319 }
320
321 void remove(const std::set<eid_t> &element_ids)
322 {
323 for (const auto id : element_ids) {
324 get_and_remove(id);
325 }
326 }
327
328private:
329 /**
330 * Returns true of this nested level contains an element with specified
331 * name.
332 *
333 * @param name Name of the element.
334 * @return True if element exists.
335 */
336 bool has_element(const std::string &name) const
337 {
338 return elements_by_name_.count(name) != 0U;
339 }
340
341 bool is_root_{false};
342
343 std::list<std::unique_ptr<T>> elements_;
344 using element_iterator_t = typename std::list<std::unique_ptr<T>>::iterator;
345
346 std::map<eid_t, element_iterator_t> elements_by_id_;
347 std::multimap<std::string, element_iterator_t> elements_by_name_;
348};
349
350} // namespace clanguml::common::model