0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
sequence_diagram_generator.cc
Go to the documentation of this file.
1/**
2 * @file src/sequence_diagram/generators/plantuml/sequence_diagram_generator.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
20
21#include "util/error.h"
22
24
30
31using namespace clanguml::util;
32
33//
34// generator
35//
36
40{
41}
42
43void generator::generate_call(const message &m, std::ostream &ostr) const
44{
45 const auto &from = model().get_participant<model::participant>(m.from());
46 const auto &to = model().get_participant<model::participant>(m.to());
47
48 if (!from || !to) {
49 LOG_DBG("Skipping empty call from '{}' to '{}'", m.from(), m.to());
50 return;
51 }
52
53 generate_participant(ostr, m.from());
54 generate_participant(ostr, m.to());
55
56 std::string message;
57
60
61 if (to.value().type_name() == "method") {
62 const auto &f = dynamic_cast<const model::method &>(to.value());
63 const std::string_view style = f.is_static() ? "__" : "";
64
65 if (m.type() == message_t::kCoAwait)
66 message = fmt::format("{}<< co_await >>\\n{}{}", style,
67 f.message_name(render_mode), style);
68 else
69 message = fmt::format(
70 "{}{}{}", style, f.message_name(render_mode), style);
71 }
72 else if (to.value().type_name() == "objc_method") {
73 const auto &f = dynamic_cast<const model::objc_method &>(to.value());
74 const std::string_view style = f.is_static() ? "__" : "";
75 message =
76 fmt::format("{}{}{}", style, f.message_name(render_mode), style);
77 }
78 else if (config().combine_free_functions_into_file_participants()) {
79 if (to.value().type_name() == "function") {
80 const auto &f = dynamic_cast<const model::function &>(to.value());
81 message = f.message_name(render_mode);
82
83 if (f.is_cuda_kernel())
84 message = fmt::format("<< CUDA Kernel >>\\n{}", message);
85 else if (f.is_cuda_device())
86 message = fmt::format("<< CUDA Device >>\\n{}", message);
87 else if (f.is_coroutine())
88 message = fmt::format("<< Coroutine >>\\n{}", message);
89 }
90 else if (to.value().type_name() == "function_template") {
91 const auto &f = dynamic_cast<const model::function &>(to.value());
92 message = f.message_name(render_mode);
93
94 if (f.is_cuda_kernel())
95 message = fmt::format("<< CUDA Kernel >>\\n{}", message);
96 else if (f.is_cuda_device())
97 message = fmt::format("<< CUDA Device >>\\n{}", message);
98 else if (f.is_coroutine())
99 message = fmt::format("<< Coroutine >>\\n{}", message);
100 }
101 }
102
103 message = config().simplify_template_type(message);
104
105 const std::string from_alias = generate_alias(from.value());
106 const std::string to_alias = generate_alias(to.value());
107
108 print_debug(m, ostr);
109
111
112 ostr << from_alias << " "
113 << common::generators::plantuml::to_plantuml(message_t::kCall) << " ";
114
115 ostr << to_alias;
116
117 if (config().generate_links) {
119 }
120
121 ostr << " : ";
122
124 ostr << "**[**";
125
127
129 ostr << "**]**";
130
131 ostr << '\n';
132
133 LOG_DBG("Generated call '{}' from {} [{}] to {} [{}]", message,
134 from.value().full_name(false), m.from(), to.value().full_name(false),
135 m.to());
136}
137
138void generator::generate_return(const message &m, std::ostream &ostr) const
139{
140 // Add return activity only for messages between different actors
141 // and only if the return type is different than void
142 if (m.from() == m.to())
143 return;
144
145 std::string message_stereotype;
146 if (m.type() == message_t::kCoReturn) {
147 message_stereotype = "<< co_return >>";
148 }
149 else if (m.type() == message_t::kCoYield) {
150 message_stereotype = "<< co_yield >>";
151 }
152
153 std::string message_label;
154
155 const auto &from = model().get_participant<model::function>(m.from());
156 const auto &to = model().get_participant<model::participant>(m.to());
157 if (to.has_value() && from.has_value() && !from.value().is_void()) {
158 const std::string from_alias = generate_alias(from.value());
159
160 const std::string to_alias = generate_alias(to.value());
161
162 ostr << from_alias << " "
163 << common::generators::plantuml::to_plantuml(message_t::kReturn)
164 << " " << to_alias;
165
166 if (config().generate_return_types())
167 message_label = render_message_name(m.return_type());
168 else if (config().generate_return_values())
169 message_label = render_message_name(m.message_name());
170 }
171 else if (from.has_value() && !from.value().is_void() &&
172 (from.value().type_name() == "method" ||
173 from.value().type_name() == "objc_method" ||
174 config().combine_free_functions_into_file_participants())) {
175 const std::string from_alias = generate_alias(from.value());
176
177 ostr << "[<--" << " " << from_alias;
178 if (config().generate_return_types())
179 message_label = render_message_name(from.value().return_type());
180 else if (config().generate_return_values())
181 message_label = render_message_name(m.message_name());
182 }
183
184 if (!message_stereotype.empty()) {
185 if (message_label.empty())
186 message_label = fmt::format("//{}//", message_stereotype);
187 else
188 message_label = fmt::format(
189 "//{}//\\n//{}//", message_stereotype, message_label);
190 }
191 else {
192 if (!message_label.empty())
193 message_label = fmt::format("//{}//", message_label);
194 }
195
196 if (!message_label.empty())
197 ostr << " : " << message_label;
198
199 ostr << '\n';
200}
201
203 eid_t activity_id, std::ostream &ostr, std::vector<eid_t> &visited) const
204{
205 const auto &a = model().get_activity(activity_id);
206
207 const auto [it, inserted] = generated_activities_.emplace(activity_id);
208
209 if (config().fold_repeated_activities() && !inserted &&
210 !a.messages().empty()) {
211 const auto &p =
212 model().get_participant<model::participant>(activity_id);
213
214 if (p.has_value()) {
215 ostr << "hnote over " << generate_alias(p.value()) << " : *\n";
216 // This is necessary to keep the hnote over the activity life line
217 ostr << generate_alias(p.value()) << "-[hidden]->"
218 << generate_alias(p.value()) << '\n';
219 }
220
221 return;
222 }
223
224 for (const auto &m : a.messages()) {
225 if (m.in_static_declaration_context()) {
227 continue;
228
230 }
231
232 if (m.type() == message_t::kCall || m.type() == message_t::kCoAwait) {
233 const auto &to =
234 model().get_participant<model::participant>(m.to());
235
236 if (!to.has_value()) {
237 LOG_DBG("Skipping activity {} due to missing target paricipant "
238 "in the diagram",
239 m.from());
240 continue;
241 }
242
243 visited.push_back(m.from());
244
245 LOG_DBG("Generating message [{}] --> [{}]", m.from(), m.to());
246
247 generate_call(m, ostr);
248
249 std::string to_alias = generate_alias(to.value());
250
251 ostr << "activate " << to_alias << '\n';
252
253 if (model().sequences().find(m.to()) != model().sequences().end()) {
254 if (std::find(visited.begin(), visited.end(), m.to()) ==
255 visited
256 .end()) { // break infinite recursion on recursive calls
257
258 LOG_DBG("Generating activity {} (called from {})", m.to(),
259 m.from());
260
261 generate_activity(m.to(), ostr, visited);
262 }
263 }
264 else
265 LOG_DBG("Skipping activity {} --> {} - missing sequence {}",
266 m.from(), m.to(), m.to());
267
268 ostr << "deactivate " << to_alias << '\n';
269
270 visited.pop_back();
271 }
272 else if (m.type() == message_t::kReturn) {
273 print_debug(m, ostr);
275 auto return_message = m;
276 if (!visited.empty()) {
277 return_message.set_to(visited.back());
278 }
279 generate_return(return_message, ostr);
280 }
281 else if (m.type() == message_t::kCoReturn) {
282 print_debug(m, ostr);
284 auto return_message = m;
285 if (!visited.empty()) {
286 return_message.set_to(visited.back());
287 }
288 generate_return(return_message, ostr);
289 }
290 else if (m.type() == message_t::kCoYield) {
291 print_debug(m, ostr);
293 auto return_message = m;
294 if (!visited.empty()) {
295 return_message.set_to(visited.back());
296 }
297 generate_return(return_message, ostr);
298 }
299 else if (m.type() == message_t::kIf) {
300 print_debug(m, ostr);
302 ostr << "alt";
303 if (const auto &text = m.condition_text(); text.has_value())
304 ostr << " " << text.value();
305 ostr << '\n';
306 }
307 else if (m.type() == message_t::kElseIf) {
308 print_debug(m, ostr);
309 ostr << "else";
310 if (const auto &text = m.condition_text(); text.has_value())
311 ostr << " " << text.value();
312 ostr << '\n';
313 }
314 else if (m.type() == message_t::kElse) {
315 print_debug(m, ostr);
316 ostr << "else\n";
317 }
318 else if (m.type() == message_t::kIfEnd) {
319 ostr << "end\n";
320 }
321 else if (m.type() == message_t::kWhile) {
322 print_debug(m, ostr);
324 ostr << "loop";
325 if (const auto &text = m.condition_text(); text.has_value())
326 ostr << " " << text.value();
327 ostr << '\n';
328 }
329 else if (m.type() == message_t::kWhileEnd) {
330 ostr << "end\n";
331 }
332 else if (m.type() == message_t::kFor) {
333 print_debug(m, ostr);
335 ostr << "loop";
336 if (const auto &text = m.condition_text(); text.has_value())
337 ostr << " " << text.value();
338 ostr << '\n';
339 }
340 else if (m.type() == message_t::kForEnd) {
341 ostr << "end\n";
342 }
343 else if (m.type() == message_t::kDo) {
344 print_debug(m, ostr);
346 ostr << "loop";
347 if (const auto &text = m.condition_text(); text.has_value())
348 ostr << " " << text.value();
349 ostr << '\n';
350 }
351 else if (m.type() == message_t::kDoEnd) {
352 ostr << "end\n";
353 }
354 else if (m.type() == message_t::kTry) {
355 print_debug(m, ostr);
357 ostr << "group try\n";
358 }
359 else if (m.type() == message_t::kCatch) {
360 print_debug(m, ostr);
361 ostr << "else " << render_message_name(m.message_name()) << '\n';
362 }
363 else if (m.type() == message_t::kTryEnd) {
364 print_debug(m, ostr);
365 ostr << "end\n";
366 }
367 else if (m.type() == message_t::kSwitch) {
368 print_debug(m, ostr);
370 ostr << "group switch\n";
371 }
372 else if (m.type() == message_t::kCase) {
373 print_debug(m, ostr);
374 ostr << "else " << render_message_name(m.message_name()) << '\n';
375 }
376 else if (m.type() == message_t::kSwitchEnd) {
377 ostr << "end\n";
378 }
379 else if (m.type() == message_t::kConditional) {
380 print_debug(m, ostr);
382 ostr << "alt";
383 if (const auto &text = m.condition_text(); text.has_value())
384 ostr << " " << text.value();
385 ostr << '\n';
386 }
387 else if (m.type() == message_t::kConditionalElse) {
388 print_debug(m, ostr);
389 ostr << "else\n";
390 }
391 else if (m.type() == message_t::kConditionalEnd) {
392 ostr << "end\n";
393 }
394 }
395}
396
398 std::ostream &ostr, const model::message &m) const
399{
400 const auto &from = model().get_participant<model::participant>(m.from());
401 if (!from)
402 return;
403
404 // First generate message comments from \note directives in comments
405 bool comment_generated_from_note_decorators{false};
406 for (const auto &decorator : m.decorators()) {
407 auto note = std::dynamic_pointer_cast<decorators::note>(decorator);
408 if (note && note->applies_to_diagram(config().name)) {
409 comment_generated_from_note_decorators = true;
410
411 ostr << "note over " << generate_alias(from.value()) << '\n';
412
414 note->text, config().message_comment_width())
415 << '\n';
416
417 ostr << "end note" << '\n';
418 }
419 }
420
421 if (comment_generated_from_note_decorators)
422 return;
423
424 if (!config().generate_message_comments())
425 return;
426
427 // Now generate message notes from raw comments if enabled
428 if (const auto &comment = m.comment(); comment &&
429 generated_comment_ids_.emplace(comment.value().at("id")).second) {
430
431 ostr << "note over " << generate_alias(from.value()) << '\n';
432
433 ostr << util::format_message_comment(comment.value().at("comment"),
434 config().message_comment_width())
435 << '\n';
436
437 ostr << "end note" << '\n';
438 }
439}
440
442 std::ostream &ostr, const std::string &name) const
443{
444 auto p = model().get(name);
445
446 if (!p.has_value()) {
447 LOG_WARN("Cannot find participant {} from `participants_order` "
448 "option",
449 name);
450 return;
451 }
452
453 generate_participant(ostr, p.value().id(), true);
454}
455
457 std::ostream &ostr, eid_t id, bool force) const
458{
459 eid_t participant_id{};
460
461 if (!force) {
462 for (const auto pid : model().active_participants()) {
463 if (pid == id) {
464 participant_id = pid;
465 break;
466 }
467 }
468 }
469 else
470 participant_id = id;
471
472 if (participant_id == 0)
473 return;
474
475 if (is_participant_generated(participant_id))
476 return;
477
478 const auto &participant =
479 model().get_participant<model::participant>(participant_id).value();
480
481 if (participant.type_name() == "method") {
482 const auto class_id =
483 model()
484 .get_participant<model::method>(participant_id)
485 .value()
486 .class_id();
487
488 if (is_participant_generated(class_id))
489 return;
490
491 const auto &class_participant =
492 model().get_participant<model::participant>(class_id).value();
493
494 print_debug(class_participant, ostr);
495
496 auto participant_name = config().simplify_template_type(
497 display_name_adapter(class_participant).full_name(false));
498 participant_name =
499 config().using_namespace().relative(participant_name);
500
502
503 ostr << "participant \"" << participant_name << "\" as "
504 << class_participant.alias();
505
506 if (config().generate_links) {
508 ostr, class_participant);
509 }
510
511 ostr << '\n';
512
513 generated_participants_.emplace(class_id);
514 }
515 else if (participant.type_name() == "objc_method") {
516 const auto class_id =
517 model()
518 .get_participant<model::objc_method>(participant_id)
519 .value()
520 .class_id();
521
522 if (is_participant_generated(class_id))
523 return;
524
525 const auto &class_participant =
526 model().get_participant<model::participant>(class_id).value();
527
528 print_debug(class_participant, ostr);
529
530 auto participant_name = config().simplify_template_type(
531 display_name_adapter(class_participant).full_name(false));
532 participant_name =
533 config().using_namespace().relative(participant_name);
534
536
537 ostr << "participant \"" << participant_name << "\" as "
538 << class_participant.alias() << " <<ObjC Interface>>";
539
540 if (config().generate_links) {
542 ostr, class_participant);
543 }
544
545 ostr << '\n';
546
547 generated_participants_.emplace(class_id);
548 }
549 else if ((participant.type_name() == "function" ||
550 participant.type_name() == "function_template") &&
551 config().combine_free_functions_into_file_participants()) {
552 // Create a single participant for all functions declared in a
553 // single file
554 const auto &file_path =
555 model()
556 .get_participant<model::function>(participant_id)
557 .value()
558 .file();
559
560 assert(!file_path.empty());
561
562 const auto file_id = common::to_id(file_path);
563
564 if (is_participant_generated(file_id))
565 return;
566
567 auto participant_name = util::path_to_url(relative(
568 std::filesystem::path{file_path}, config().root_directory())
569 .string());
570
571 ostr << "participant \"" << participant_name << "\" as "
572 << fmt::format("C_{:022}", file_id.value());
573
574 ostr << '\n';
575
576 generated_participants_.emplace(file_id);
577 }
578 else {
579 print_debug(participant, ostr);
580
581 auto participant_name =
582 config().using_namespace().relative(config().simplify_template_type(
583 display_name_adapter(participant).full_name(false)));
585
586 ostr << "participant \"" << participant_name << "\" as "
587 << participant.alias();
588
589 if (const auto *function_ptr =
590 dynamic_cast<const model::function *>(&participant);
591 function_ptr) {
592 if (function_ptr->is_cuda_kernel())
593 ostr << " << CUDA Kernel >>";
594 else if (function_ptr->is_cuda_device())
595 ostr << " << CUDA Device >>";
596 else if (function_ptr->is_coroutine())
597 ostr << fmt::format("<< Coroutine >>");
598 }
599
600 if (config().generate_links) {
602 ostr, participant);
603 }
604
605 ostr << '\n';
606
607 generated_participants_.emplace(participant_id);
608 }
609}
610
612{
613 return std::find(generated_participants_.begin(),
615 id) != generated_participants_.end();
616}
617
619 const model::participant &participant) const
620{
621 if ((participant.type_name() == "function" ||
622 participant.type_name() == "function_template") &&
623 config().combine_free_functions_into_file_participants()) {
624 const auto file_id = common::to_id(participant.file());
625
626 return fmt::format("C_{:022}", file_id.value());
627 }
628
629 return participant.alias();
630}
631
632void generator::generate_diagram(std::ostream &ostr) const
633{
634 model().print();
635
636 if (config().participants_order.has_value) {
637 for (const auto &p : config().participants_order()) {
638 LOG_DBG("Pregenerating participant {}", p);
639 generate_participant(ostr, p);
640 }
641 }
642
644
646
648}
649
650void generator::generate_from_sequences(std::ostream &ostr) const
651{
652 std::vector<eid_t> start_from = find_from_activities();
653
654 // Use this to break out of recurrent loops
655 std::vector<eid_t> visited_participants;
656
657 for (const auto from_id : start_from) {
658 if (model().participants().count(from_id) == 0)
659 continue;
660
661 const auto &from = model().get_participant<model::function>(from_id);
662
663 if (!from.has_value()) {
664 LOG_WARN("Failed to find participant {} for 'from' "
665 "condition");
666 continue;
667 }
668
669 generate_participant(ostr, from_id);
670
671 std::string from_alias = generate_alias(from.value());
672
675
676 // For methods or functions in diagrams where they are
677 // combined into file participants, we need to add an
678 // 'entry' point call to know which method relates to the
679 // first activity for this 'start_from' condition
680 if (from.value().type_name() == "method" ||
681 from.value().type_name() == "objc_method" ||
682 config().combine_free_functions_into_file_participants()) {
683 ostr << "[->" << " " << from_alias << " : "
684 << render_message_name(from.value().message_name(render_mode))
685 << '\n';
686 }
687
688 ostr << "activate " << from_alias << '\n';
689
690 generate_activity(from_id, ostr, visited_participants);
691
692 ostr << "deactivate " << from_alias << '\n';
693 }
694}
695
696std::vector<eid_t> generator::find_from_activities() const
697{
698 std::vector<eid_t> start_from;
699 for (const auto &sf : config().from()) {
700 if (sf.location_type == location_t::function) {
701 bool found{false};
702 for (const auto &[k, v] : model().sequences()) {
703 if (model().participants().count(v.from()) == 0)
704 continue;
705
706 const auto &caller = *model().participants().at(v.from());
707 std::string vfrom = caller.full_name(false);
708 if (sf.location == vfrom) {
709 LOG_DBG("Found sequence diagram start point: {}", k);
710 start_from.push_back(k);
711 found = true;
712 }
713 }
714
715 if (!found)
717 model().name(),
718 fmt::format("Failed to find participant matching '{}' for "
719 "'from' condition: ",
720 sf.location.to_string()));
721 }
722 }
723
724 return start_from;
725}
726
727std::vector<model::message_chain_t> generator::find_to_message_chains() const
728{
729 std::vector<model::message_chain_t> result;
730
731 for (const auto &to_location : config().to()) {
732 auto to_activity_ids = model().get_to_activity_ids(to_location);
733
734 if (to_activity_ids.empty()) {
735 LOG_WARN("Failed to find participant matching '{}' for "
736 "'to' condition: ",
737 to_location.location.to_string());
738 }
739
740 for (const auto &to_activity_id : to_activity_ids) {
741 std::vector<model::message_chain_t> message_chains_unique =
742 model().get_all_from_to_message_chains(eid_t{}, to_activity_id);
743
744 result.insert(result.end(), message_chains_unique.begin(),
745 message_chains_unique.end());
746 }
747 }
748
749 return result;
750}
751
752std::string generator::render_message_name(const std::string &m) const
753{
754 return util::abbreviate(m, config().message_name_width());
755}
756
757void generator::generate_to_sequences(std::ostream &ostr) const
758{
759 std::vector<model::message_chain_t> message_chains =
761
762 bool first_separator_skipped{false};
763 for (const auto &mc : message_chains) {
764 if (!first_separator_skipped)
765 first_separator_skipped = true;
766 else
767 ostr << "====\n";
768
769 const auto from_activity_id = mc.front().from();
770
771 if (model().participants().count(from_activity_id) == 0)
772 continue;
773
774 const auto &from =
775 model().get_participant<model::function>(from_activity_id);
776
777 if (from.value().type_name() == "method" ||
778 from.value().type_name() == "objc_method" ||
779 config().combine_free_functions_into_file_participants()) {
780 generate_participant(ostr, from_activity_id);
781 ostr << "[->" << " " << generate_alias(from.value()) << " : "
782 << render_message_name(from.value().message_name(
784 << '\n';
785 }
786
787 for (const auto &m : mc) {
788 generate_call(m, ostr);
789 }
790 }
791}
792
793void generator::generate_from_to_sequences(std::ostream &ostr) const
794{
795 for (const auto &ft : config().from_to()) {
796 // First, find the sequence of activities from 'from' location
797 // to 'to' location
798 assert(ft.size() == 2);
799
800 const auto &from_location = ft.front();
801 const auto &to_location = ft.back();
802
803 const auto from_activity_ids =
804 model().get_from_activity_ids(from_location);
805
806 const auto to_activity_ids = model().get_to_activity_ids(to_location);
807
808 if (from_activity_ids.empty()) {
810 model().name(),
811 fmt::format("Failed to find participant matching '{}' for "
812 "'from' condition: ",
813 from_location.location.to_string()));
814 }
815
816 if (from_activity_ids.empty() || to_activity_ids.empty()) {
818 model().name(),
819 fmt::format("Failed to find participant matching '{}' for "
820 "'to' condition: ",
821 to_location.location.to_string()));
822 }
823
824 bool first_separator_skipped{false};
825
826 for (const auto from_activity_id : from_activity_ids) {
827 if (model().participants().count(from_activity_id) == 0)
828 continue;
829
830 for (const auto to_activity_id : to_activity_ids) {
831 if (model().participants().count(to_activity_id) == 0)
832 continue;
833
834 auto message_chains_unique =
835 model().get_all_from_to_message_chains(
836 from_activity_id, to_activity_id);
837
838 for (const auto &mc : message_chains_unique) {
839 if (!first_separator_skipped)
840 first_separator_skipped = true;
841 else
842 ostr << "====\n";
843
844 const auto &from = model().get_participant<model::function>(
845 from_activity_id);
846
847 if (from.value().type_name() == "method" ||
848 from.value().type_name() == "objc_method" ||
849 config()
850 .combine_free_functions_into_file_participants()) {
851 generate_participant(ostr, from_activity_id);
852 ostr << "[->" << " " << generate_alias(from.value())
853 << " : "
854 << render_message_name(from.value().message_name(
856 << '\n';
857 }
858
859 for (const auto &m : mc) {
860 generate_call(m, ostr);
861 }
862 }
863 }
864 }
865 }
866}
867
870{
871 if (config().generate_method_arguments() ==
874
875 if (config().generate_method_arguments() == config::method_arguments::none)
877
879}
880
881} // namespace clanguml::sequence_diagram::generators::plantuml