0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
config.cc
Go to the documentation of this file.
1/**
2 * @file src/config/config.cc
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
19#include "config.h"
20#include "diagram_templates.h"
21#include "glob/glob.hpp"
22
23#include <filesystem>
24
26
27std::string to_string(const hint_t t)
28{
29 switch (t) {
30 case hint_t::up:
31 return "up";
32 case hint_t::down:
33 return "down";
34 case hint_t::left:
35 return "left";
36 case hint_t::right:
37 return "right";
39 return "together";
40 case hint_t::row:
41 return "row";
42 case hint_t::column:
43 return "column";
44 default:
45 assert(false);
46 return "";
47 }
48}
49
50std::string to_string(element_filter_t::filtered_type ft)
51{
52 switch (ft) {
54 return "any";
56 return "class";
58 return "function";
60 return "method";
62 return "member";
64 return "enum";
66 return "concept";
68 return "package";
70 return "function_template";
72 return "objc_method";
74 return "objc_member";
76 return "objc_protocol";
78 return "objc_category";
80 return "objc_interface";
81 default:
82 assert(false);
83 return "";
84 }
85}
86
87std::string to_string(const method_arguments ma)
88{
89 switch (ma) {
91 return "full";
93 return "abbreviated";
95 return "none";
96 default:
97 assert(false);
98 return "";
99 }
100}
101
102std::string to_string(method_type mt)
103{
104 switch (mt) {
106 return "constructor";
108 return "destructor";
110 return "assignment";
112 return "operator";
114 return "defaulted";
116 return "deleted";
118 return "static";
119 }
120
121 assert(false);
122 return "";
123}
124
125std::string to_string(callee_type mt)
126{
127 switch (mt) {
129 return "constructor";
131 return "assignment";
133 return "operator";
135 return "defaulted";
137 return "static";
139 return "method";
141 return "function";
143 return "function_template";
145 return "lambda";
147 return "cuda_kernel";
149 return "cuda_device";
150 }
151
152 assert(false);
153 return "";
154}
155
156std::string to_string(const comment_parser_t cp)
157{
158 switch (cp) {
160 return "clang";
162 return "plain";
163 default:
164 assert(false);
165 return "";
166 }
167}
168
169std::string to_string(location_t cp)
170{
171 switch (cp) {
173 return "fileline";
175 return "function";
177 return "marker";
178 default:
179 assert(false);
180 return "";
181 }
182}
183
184std::string to_string(package_type_t pt)
185{
186 switch (pt) {
188 return "namespace";
190 return "directory";
192 return "module";
193 default:
194 assert(false);
195 return "";
196 }
197}
198
199std::string to_string(member_order_t mo)
200{
201 switch (mo) {
203 return "lexical";
205 return "as_is";
206 default:
207 assert(false);
208 return "";
209 }
210}
211std::string to_string(context_direction_t cd)
212{
213 switch (cd) {
215 return "inward";
217 return "outward";
219 return "any";
220 default:
221 assert(false);
222 return "";
223 }
224}
225
226std::string to_string(filter_mode_t fm)
227{
228 switch (fm) {
230 return "basic";
232 return "advanced";
233 default:
234 assert(false);
235 return "";
236 }
237}
238
239std::optional<std::string> plantuml::get_style(
240 const common::model::relationship_t relationship_type) const
241{
242 if (style.count(to_string(relationship_type)) == 0)
243 return {};
244
245 return style.at(to_string(relationship_type));
246}
247
248std::optional<std::string> plantuml::get_style(
249 const std::string &element_type) const
250{
251 if (style.count(element_type) == 0)
252 return {};
253
254 return style.at(element_type);
255}
256
258{
259 before.insert(before.end(), r.before.begin(), r.before.end());
260 after.insert(after.end(), r.after.begin(), r.after.end());
261 if (cmd.empty())
262 cmd = r.cmd;
263}
264
266{
267 before.insert(before.end(), r.before.begin(), r.before.end());
268 after.insert(after.end(), r.after.begin(), r.after.end());
269 if (cmd.empty())
270 cmd = r.cmd;
271}
272
274{
275 auto tmp = r.notes;
276 notes.merge(tmp);
277}
278
280{
281 return weakly_canonical(absolute(base_directory() / relative_to()));
282}
283
285 const inheritable_diagram_options &parent)
286{
287 glob.override(parent.glob);
288 using_namespace.override(parent.using_namespace);
294 include.override(parent.include);
295 exclude.override(parent.exclude);
296 puml.override(parent.puml);
297 mermaid.override(parent.mermaid);
298 graphml.override(parent.graphml);
312 generate_links.override(parent.generate_links);
314 git.override(parent.git);
330}
331
333 std::string full_name) const
334{
336 aliases.insert(type_aliases().begin(), type_aliases().end());
337
338 bool matched{true};
339 while (matched) {
340 auto matched_in_iteration{false};
341 for (const auto &[pattern, replacement] : aliases) {
342 matched_in_iteration =
343 util::replace_all(full_name, pattern, replacement) ||
344 matched_in_iteration;
345 }
346 matched = matched_in_iteration;
347 }
348
349 return full_name;
350}
351
353{
356}
357
358std::vector<std::string> diagram::glob_translation_units(
359 const std::vector<std::string> &compilation_database_files,
360 bool is_fixed) const
361{
362 // Make sure that the paths are in preferred format for a given platform
363 // before intersecting the matches with compliation database
364 std::vector<std::string> compilation_database_paths;
365 compilation_database_paths.reserve(compilation_database_files.size());
366 for (const auto &cdf : compilation_database_files) {
367 std::filesystem::path p{cdf};
368 p.make_preferred();
369 compilation_database_paths.emplace_back(p.string());
370 }
371
372 // If glob is not defined use all translation units from the
373 // compilation database
374 if (!glob.has_value || (glob().include.empty() && glob().exclude.empty())) {
375 return compilation_database_paths;
376 }
377
378 // Otherwise, get all translation units matching the glob from diagram
379 // configuration
380 std::vector<std::string> glob_matches{};
381
382 LOG_DBG("Looking for translation units in {}", root_directory().string());
383
384 for (const auto &g : glob().include) {
385 if (g.is_regex()) {
386 LOG_DBG("Matching inclusive glob regex {}", g.to_string());
387
388 std::regex regex_pattern(
389 g.to_string(), std::regex_constants::optimize);
390
391 std::copy_if(compilation_database_paths.begin(),
392 compilation_database_paths.end(),
393 std::back_inserter(glob_matches),
394 [&regex_pattern](const auto &tu) {
395 std::smatch m;
396
397 return exists(std::filesystem::path{tu}) &&
398 std::regex_search(tu, m, regex_pattern);
399 });
400 }
401 else {
402 std::filesystem::path absolute_glob_path{g.to_string()};
403
404#ifdef _MSC_VER
405 if (!absolute_glob_path.has_root_name())
406#else
407 if (!absolute_glob_path.is_absolute())
408#endif
409 absolute_glob_path = root_directory() / absolute_glob_path;
410
411 LOG_DBG("Searching glob path {}", absolute_glob_path.string());
412
413 auto matches = glob::glob(absolute_glob_path.string(), true, false);
414
415 for (const auto &match : matches) {
416 const auto path =
417 std::filesystem::canonical(root_directory() / match);
418
419 glob_matches.emplace_back(path.string());
420 }
421 }
422 }
423
424 if (glob().include.empty())
425 glob_matches = compilation_database_paths;
426
427 for (const auto &g : glob().exclude) {
428 if (g.is_regex()) {
429 LOG_DBG("Matching exclusive glob regex {}", g.to_string());
430
431 std::regex regex_pattern(
432 g.to_string(), std::regex_constants::optimize);
433
434 for (const auto &cdf : compilation_database_paths) {
435 std::smatch m;
436 if (std::regex_search(cdf, m, regex_pattern)) {
437 glob_matches.erase(
438 remove(begin(glob_matches), end(glob_matches), cdf),
439 glob_matches.end());
440 }
441 }
442 }
443 else {
444 std::filesystem::path absolute_glob_path{g.to_string()};
445
446#ifdef _MSC_VER
447 if (!absolute_glob_path.has_root_name())
448#else
449 if (!absolute_glob_path.is_absolute())
450#endif
451 absolute_glob_path = root_directory() / absolute_glob_path;
452
453 LOG_DBG("Searching exclusive glob path {}",
454 absolute_glob_path.string());
455
456 auto matches = glob::glob(absolute_glob_path.string(), true, false);
457
458 for (const auto &match : matches) {
459 const auto path =
460 std::filesystem::canonical(root_directory() / match);
461
462 glob_matches.erase(remove(begin(glob_matches),
463 end(glob_matches), path.string()),
464 glob_matches.end());
465 }
466 }
467 }
468
469 // Calculate intersection between glob matches and compilation database
470 std::vector<std::string> result;
471 for (const auto &gm : glob_matches) {
472 std::filesystem::path gm_path{gm};
473 gm_path.make_preferred();
474 if (is_fixed || util::contains(compilation_database_paths, gm_path) ||
475 util::contains(compilation_database_files, gm)) {
476 result.emplace_back(gm_path.string());
477 }
478 }
479
480 return result;
481}
482
483std::filesystem::path diagram::make_path_relative(
484 const std::filesystem::path &p) const
485{
486 return relative(p, root_directory()).lexically_normal().string();
487}
488
489std::vector<std::string> diagram::make_module_relative(
490 const std::optional<std::string> &maybe_module) const
491{
492 if (!maybe_module)
493 return {};
494
495 auto module_path = common::model::path(
496 maybe_module.value(), common::model::path_type::kModule)
497 .tokens();
498
499 if (using_module.has_value) {
500 auto using_module_path = common::model::path(
501 using_module(), common::model::path_type::kModule)
502 .tokens();
503
504 if (util::starts_with(module_path, using_module_path)) {
505 util::remove_prefix(module_path, using_module_path);
506 }
507 }
508
509 return module_path;
510}
511
512std::optional<std::string> diagram::get_together_group(
513 const std::string &full_name) const
514{
515 const auto relative_name = using_namespace().relative(full_name);
516
517 for (const auto &[hint_target, hints] : layout()) {
518 for (const auto &hint : hints) {
519 if (hint.hint == hint_t::together) {
520 const auto &together_others =
521 std::get<std::vector<std::string>>(hint.entity);
522
523 if ((full_name == hint_target) ||
524 util::contains(together_others, full_name))
525 return hint_target;
526
527 if ((relative_name == hint_target) ||
528 util::contains(together_others, relative_name))
529 return hint_target;
530 }
531 }
532 }
533
534 return std::nullopt;
535}
536
537void diagram::initialize_type_aliases()
538{
539 if (type_aliases().count("std::basic_string<char>") == 0U) {
540 type_aliases().insert({"std::basic_string<char>", "std::string"});
541 }
542 if (type_aliases().count("std::basic_string<char,std::char_traits<"
543 "char>,std::allocator<char>>") == 0U) {
544 type_aliases().insert({"std::basic_string<char,std::char_traits<"
545 "char>,std::allocator<char>>",
546 "std::string"});
547 }
548 if (type_aliases().count("std::basic_string<wchar_t>") == 0U) {
549 type_aliases().insert({"std::basic_string<wchar_t>", "std::wstring"});
550 }
551 if (type_aliases().count("std::basic_string<char16_t>") == 0U) {
552 type_aliases().insert(
553 {"std::basic_string<char16_t>", "std::u16string"});
554 }
555 if (type_aliases().count("std::basic_string<char32_t>") == 0U) {
556 type_aliases().insert(
557 {"std::basic_string<char32_t>", "std::u32string"});
558 }
559 if (type_aliases().count("std::integral_constant<bool,true>") == 0U) {
560 type_aliases().insert(
561 {"std::integral_constant<bool,true>", "std::true_type"});
562 }
563 if (type_aliases().count("std::integral_constant<bool,false>") == 0U) {
564 type_aliases().insert(
565 {"std::integral_constant<bool,false>", "std::false_type"});
566 }
567#if LLVM_VERSION_MAJOR >= 16
568 if (type_aliases().count("std::basic_string") == 0U) {
569 type_aliases().insert({"std::basic_string", "std::string"});
570 }
571#endif
572}
573
574common::model::diagram_t class_diagram::type() const
575{
577}
578
579common::model::diagram_t sequence_diagram::type() const
580{
582}
583
584common::model::diagram_t package_diagram::type() const
585{
587}
588
589common::model::diagram_t include_diagram::type() const
590{
592}
593
594void class_diagram::initialize_relationship_hints()
595{
597
598 if (relationship_hints().count("std::vector") == 0U) {
599 relationship_hints().insert({"std::vector", {}});
600 }
601 if (relationship_hints().count("std::unique_ptr") == 0U) {
602 relationship_hints().insert({"std::unique_ptr", {}});
603 }
604 if (relationship_hints().count("std::shared_ptr") == 0U) {
605 relationship_hints().insert(
606 {"std::shared_ptr", {relationship_t::kAssociation}});
607 }
608 if (relationship_hints().count("std::weak_ptr") == 0U) {
609 relationship_hints().insert(
610 {"std::weak_ptr", {relationship_t::kAssociation}});
611 }
612 if (relationship_hints().count("std::tuple") == 0U) {
613 relationship_hints().insert({"std::tuple", {}});
614 }
615 if (relationship_hints().count("std::map") == 0U) {
616 relationship_hint_t hint{relationship_t::kNone};
617 hint.argument_hints.insert({1, relationship_t::kAggregation});
618 relationship_hints().insert({"std::tuple", std::move(hint)});
619 }
620}
621
622template <> void append_value<plantuml>(plantuml &l, const plantuml &r)
623{
624 l.append(r);
625}
626} // namespace clanguml::config