47 ostr <<
"sequenceDiagram\n";
57 bool comment_generated_from_note_decorators{
false};
59 auto note = std::dynamic_pointer_cast<decorators::note>(decorator);
60 if (note && note->applies_to_diagram(
config().name)) {
61 comment_generated_from_note_decorators =
true;
67 note->text,
config().message_comment_width());
70 ostr << formatted_message <<
'\n';
74 if (comment_generated_from_note_decorators)
77 if (
const auto &cmt = m.
comment();
config().generate_message_comments() &&
85 cmt.value().at(
"comment"),
config().message_comment_width());
89 ostr << formatted_message <<
'\n';
99 LOG_DBG(
"Skipping empty call from '{}' to '{}'", m.
from(), m.
to());
111 if (to.value().
type_name() ==
"method") {
112 const auto &f =
dynamic_cast<const model::method &
>(to.value());
113 if (m.
type() == message_t::kCoAwait)
115 "<< co_await >><br>{}", f.message_name(render_mode));
119 else if (to.value().
type_name() ==
"objc_method") {
123 else if (
config().combine_free_functions_into_file_participants()) {
124 if (to.value().
type_name() ==
"function") {
129 if (f.is_cuda_kernel())
131 else if (f.is_cuda_device())
133 else if (f.is_coroutine())
136 else if (to.value().
type_name() ==
"function_template") {
140 if (f.is_cuda_kernel())
142 else if (f.is_cuda_device())
144 else if (f.is_coroutine())
158 ostr << indent(1) << from_alias <<
" "
187 std::string message_stereotype;
188 if (m.
type() == message_t::kCoReturn) {
189 message_stereotype =
"<< co_return >>";
191 else if (m.
type() == message_t::kCoYield) {
192 message_stereotype =
"<< co_yield >>";
195 std::string message_label;
199 if (to.has_value() && from.has_value() && !from.value().
is_void()) {
204 ostr << indent(1) << from_alias <<
" "
206 <<
" " << to_alias <<
" : ";
208 if (
config().generate_return_types()) {
211 else if (
config().generate_return_values()) {
215 else if (from.has_value() && !from.value().is_void() &&
216 (from.value().type_name() ==
"method" ||
217 from.value().type_name() ==
"objc_method" ||
218 config().combine_free_functions_into_file_participants())) {
221 ostr << indent(1) << from_alias <<
" "
225 if (
config().generate_return_types())
227 else if (
config().generate_return_values())
231 if (!message_stereotype.empty()) {
232 if (message_label.empty())
233 message_label = fmt::format(
"{}", message_stereotype);
236 fmt::format(
"{}<br>{}", message_stereotype, message_label);
239 if (!message_label.empty())
240 message_label = fmt::format(
"{}", message_label);
243 if (!message_label.empty())
244 ostr << message_label;
256 eid_t activity_id, std::ostream &ostr, std::vector<eid_t> &visited)
const
258 const auto &a =
model().get_activity(activity_id);
262 if (
config().fold_repeated_activities() && !inserted &&
263 !a.messages().empty()) {
275 for (
const auto &m : a.messages()) {
276 if (m.in_static_declaration_context()) {
283 if (m.type() == message_t::kCall || m.type() == message_t::kCoAwait) {
287 if (!to.has_value()) {
288 LOG_DBG(
"Skipping activity {} due to missing target paricipant "
294 visited.push_back(m.from());
296 LOG_DBG(
"Generating message [{}] --> [{}]", m.from(), m.to());
302 ostr << indent(1) <<
"activate " << to_alias <<
'\n';
304 if (
model().sequences().find(m.to()) !=
model().sequences().end()) {
305 if (std::find(visited.begin(), visited.end(), m.to()) ==
308 LOG_DBG(
"Creating activity {} --> {} - missing sequence {}",
309 m.from(), m.to(), m.to());
314 LOG_DBG(
"Skipping activity {} --> {} - missing sequence {}",
315 m.from(), m.to(), m.to());
317 ostr << indent(1) <<
"deactivate " << to_alias <<
'\n';
321 else if (m.type() == message_t::kReturn) {
324 auto return_message = m;
325 if (!visited.empty()) {
326 return_message.set_to(visited.back());
330 else if (m.type() == message_t::kCoReturn) {
333 auto return_message = m;
334 if (!visited.empty()) {
335 return_message.set_to(visited.back());
339 else if (m.type() == message_t::kCoYield) {
342 auto return_message = m;
343 if (!visited.empty()) {
344 return_message.set_to(visited.back());
348 else if (m.type() == message_t::kIf) {
351 ostr << indent(1) <<
"alt";
352 if (
const auto &text = m.condition_text(); text.has_value())
356 else if (m.type() == message_t::kElseIf) {
358 ostr << indent(1) <<
"else";
359 if (
const auto &text = m.condition_text(); text.has_value())
363 else if (m.type() == message_t::kElse) {
365 ostr << indent(1) <<
"else\n";
367 else if (m.type() == message_t::kIfEnd) {
368 ostr << indent(1) <<
"end\n";
370 else if (m.type() == message_t::kWhile) {
373 ostr << indent(1) <<
"loop";
374 if (
const auto &text = m.condition_text(); text.has_value())
378 else if (m.type() == message_t::kWhileEnd) {
379 ostr << indent(1) <<
"end\n";
381 else if (m.type() == message_t::kFor) {
384 ostr << indent(1) <<
"loop";
385 if (
const auto &text = m.condition_text(); text.has_value())
389 else if (m.type() == message_t::kForEnd) {
390 ostr << indent(1) <<
"end\n";
392 else if (m.type() == message_t::kDo) {
395 ostr << indent(1) <<
"loop";
396 if (
const auto &text = m.condition_text(); text.has_value())
400 else if (m.type() == message_t::kDoEnd) {
401 ostr << indent(1) <<
"end\n";
403 else if (m.type() == message_t::kTry) {
406 ostr << indent(1) <<
"critical\n";
408 else if (m.type() == message_t::kCatch) {
410 ostr << indent(1) <<
"option "
413 else if (m.type() == message_t::kTryEnd) {
415 ostr << indent(1) <<
"end\n";
417 else if (m.type() == message_t::kSwitch) {
420 ostr << indent(1) <<
"alt\n";
422 else if (m.type() == message_t::kCase) {
424 ostr << indent(1) <<
"else "
427 else if (m.type() == message_t::kSwitchEnd) {
428 ostr << indent(1) <<
"end\n";
430 else if (m.type() == message_t::kConditional) {
433 ostr << indent(1) <<
"alt";
434 if (
const auto &text = m.condition_text(); text.has_value())
438 else if (m.type() == message_t::kConditionalElse) {
440 ostr << indent(1) <<
"else\n";
442 else if (m.type() == message_t::kConditionalEnd) {
443 ostr << indent(1) <<
"end\n";
449 std::ostream &ostr,
const std::string &name)
const
451 auto p =
model().get(name);
453 if (!p.has_value()) {
454 LOG_WARN(
"Cannot find participant {} from `participants_order` option",
463 std::ostream &ostr,
eid_t id,
bool force)
const
465 eid_t participant_id{};
468 for (
const auto pid :
model().active_participants()) {
470 participant_id = pid;
478 if (participant_id == 0)
484 const auto &participant =
487 if (participant.type_name() ==
"method") {
488 const auto class_id =
497 const auto &class_participant =
502 auto participant_name =
503 config().using_namespace().relative(
config().simplify_template_type(
507 ostr << indent(1) <<
"participant " << class_participant.alias()
508 <<
" as " << participant_name;
514 else if (participant.type_name() ==
"objc_method") {
515 const auto class_id =
524 const auto &class_participant =
529 auto participant_name =
530 config().using_namespace().relative(
config().simplify_template_type(
534 ostr << indent(1) <<
"participant " << class_participant.alias()
535 <<
" as " <<
"<< ObjC Interface >><br>" << participant_name;
541 else if ((participant.type_name() ==
"function" ||
542 participant.type_name() ==
"function_template") &&
543 config().combine_free_functions_into_file_participants()) {
549 const auto &file_path = f.file();
551 assert(!file_path.empty());
559 std::filesystem::path{file_path},
config().root_directory())
562 ostr << indent(1) <<
"participant "
563 << fmt::format(
"C_{:022}", file_id.value()) <<
" as "
572 auto participant_name =
573 config().using_namespace().relative(
config().simplify_template_type(
577 ostr << indent(1) <<
"participant " << participant.alias() <<
" as ";
579 if (participant.type_name() ==
"function" ||
580 participant.type_name() ==
"function_template") {
586 if (f.is_cuda_kernel())
587 ostr <<
"<< CUDA Kernel >><br>";
588 else if (f.is_cuda_device())
589 ostr <<
"<< CUDA Device >><br>";
592 ostr << participant_name;
609 if ((participant.
type_name() ==
"function" ||
610 participant.
type_name() ==
"function_template") &&
611 config().combine_free_functions_into_file_participants()) {
614 return fmt::format(
"C_{:022}", file_id.value());
617 return participant.
alias();
624 if (
config().participants_order.has_value) {
625 for (
const auto &p :
config().participants_order()) {
626 LOG_DBG(
"Pregenerating participant {}", p);
631 bool star_participant_generated{
false};
645 std::vector<eid_t> visited_participants;
646 for (
const auto from_id : start_from) {
647 if (
model().participants().count(from_id) == 0)
652 if (!from.has_value()) {
653 LOG_WARN(
"Failed to find participant {} for 'from' "
669 if (from.value().type_name() ==
"method" ||
670 from.value().type_name() ==
"objc_method" ||
671 config().combine_free_functions_into_file_participants()) {
672 ostr << indent(1) <<
"* "
674 <<
" " << from_alias <<
" : "
675 << from.value().message_name(render_mode) <<
'\n';
678 ostr << indent(1) <<
"activate " << from_alias <<
'\n';
682 ostr << indent(1) <<
"deactivate " << from_alias <<
'\n';
688 std::vector<model::message_chain_t> result;
690 for (
const auto &to_location :
config().to()) {
691 auto to_activity_ids =
model().get_to_activity_ids(to_location);
693 if (to_activity_ids.empty()) {
694 LOG_WARN(
"Failed to find participant matching '{}' for "
696 to_location.location.to_string());
699 for (
const auto &to_activity_id : to_activity_ids) {
700 std::vector<model::message_chain_t> message_chains_unique =
701 model().get_all_from_to_message_chains(
eid_t{}, to_activity_id);
703 result.insert(result.end(), message_chains_unique.begin(),
704 message_chains_unique.end());
713 std::vector<model::message_chain_t> message_chains =
716 for (
const auto &mc : message_chains) {
717 const auto from_activity_id = mc.front().from();
719 if (
model().participants().count(from_activity_id) == 0)
725 if (from.value().type_name() ==
"method" ||
726 config().combine_free_functions_into_file_participants()) {
728 ostr << indent(1) <<
"* "
736 for (
const auto &m : mc) {
743 std::ostream &ostr,
bool star_participant_generated)
const
745 for (
const auto &ft :
config().from_to()) {
748 assert(ft.size() == 2);
750 const auto &from_location = ft.front();
751 const auto &to_location = ft.back();
753 auto from_activity_ids =
model().get_from_activity_ids(from_location);
754 auto to_activity_ids =
model().get_to_activity_ids(to_location);
756 if (from_activity_ids.empty()) {
759 fmt::format(
"Failed to find participant matching '{}' for "
760 "'from' condition: ",
761 from_location.location.to_string()));
764 if (from_activity_ids.empty() || to_activity_ids.empty()) {
767 fmt::format(
"Failed to find participant matching '{}' for "
769 to_location.location.to_string()));
772 for (
const auto from_activity_id : from_activity_ids) {
773 if (
model().participants().count(from_activity_id) == 0)
776 for (
const auto to_activity_id : to_activity_ids) {
777 if (
model().participants().count(to_activity_id) == 0)
780 auto message_chains_unique =
781 model().get_all_from_to_message_chains(
782 from_activity_id, to_activity_id);
784 for (
const auto &mc : message_chains_unique) {
788 if (from.value().type_name() ==
"method" ||
790 .combine_free_functions_into_file_participants()) {
791 if (!star_participant_generated) {
792 ostr << indent(1) <<
"participant *\n";
793 star_participant_generated =
true;
796 ostr << indent(1) <<
"* "
805 for (
const auto &m : mc) {
816 std::vector<eid_t> start_from;
817 for (
const auto &sf :
config().from()) {
818 if (sf.location_type == location_t::function) {
820 for (
const auto &[k, v] :
model().sequences()) {
821 if (
model().participants().count(v.from()) == 0)
824 const auto &caller = *
model().participants().at(v.from());
825 std::string vfrom = caller.full_name(
false);
826 if (sf.location == vfrom) {
827 LOG_DBG(
"Found sequence diagram start point: {}", k);
828 start_from.push_back(k);
836 fmt::format(
"Failed to find participant matching '{}' for "
837 "'from' condition: ",
838 sf.location.to_string()));
848 if (
config().generate_method_arguments() ==