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/json/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
22
24
25void to_json(nlohmann::json &j, const participant &c)
26{
27 to_json(j, dynamic_cast<const participant::element &>(c));
28 j["type"] = c.type_name();
29
30 if (c.type_name() == "method") {
31 j["name"] = dynamic_cast<const method &>(c).method_name();
32 }
33 else if (c.type_name() == "objc_method") {
34 j["name"] = dynamic_cast<const objc_method &>(c).method_name();
35 }
36
37 j["full_name"] = display_name_adapter(c).full_name(false);
38
39 if (c.type_name() == "function" || c.type_name() == "function_template") {
40 const auto &f = dynamic_cast<const function &>(c);
41 if (f.is_cuda_kernel())
42 j["is_cuda_kernel"] = true;
43 if (f.is_cuda_device())
44 j["is_cuda_device"] = true;
45 if (f.is_coroutine())
46 j["is_coroutine"] = true;
47 }
48}
49
50void to_json(nlohmann::json &j, const activity &c)
51{
52 j["participant_id"] = std::to_string(c.from().value());
53}
54
55} // namespace clanguml::sequence_diagram::model
56
58
64
65using namespace clanguml::util;
66
67//
68// generator
69//
70
74{
75}
76
77void generator::generate_call(const message &m, nlohmann::json &parent) const
78{
79 const auto &from = model().get_participant<model::participant>(m.from());
80 const auto &to = model().get_participant<model::participant>(m.to());
81
82 if (!from || !to) {
83 LOG_DBG("Skipping empty call from '{}' to '{}'", m.from(), m.to());
84 return;
85 }
86
89
90 std::string message;
91
94
95 if (to.value().type_name() == "method") {
96 message = dynamic_cast<const model::method &>(to.value())
97 .message_name(render_mode);
98 }
99 else if (to.value().type_name() == "objc_method") {
100 message = dynamic_cast<const model::objc_method &>(to.value())
101 .message_name(render_mode);
102 }
103 else if (config().combine_free_functions_into_file_participants()) {
104 if (to.value().type_name() == "function") {
105 message = dynamic_cast<const model::function &>(to.value())
106 .message_name(render_mode);
107 }
108 else if (to.value().type_name() == "function_template") {
109 message = dynamic_cast<const model::function_template &>(to.value())
110 .message_name(render_mode);
111 }
112 }
113
114 message = config().simplify_template_type(message);
115
116 nlohmann::json msg;
117
118 msg["name"] = message;
119 if (m.type() == message_t::kCoAwait)
120 msg["type"] = "co_await";
121 else
122 msg["type"] = "message";
123
124 generate_from_activity(m, from, msg);
125
126 generate_to_activity(to, msg);
127
128 msg["source_location"] =
129 dynamic_cast<const clanguml::common::model::source_location &>(m);
130
131 msg["scope"] = to_string(m.message_scope());
132 msg["return_type"] = config().simplify_template_type(m.return_type());
133
134 parent["messages"].push_back(std::move(msg));
135
136 LOG_DBG("Generated call '{}' from {} [{}] to {} [{}]", message,
137 from.value().full_name(false), m.from(), to.value().full_name(false),
138 m.to());
139}
140
143 nlohmann::json &msg) const
144{
145 msg["to"]["activity_id"] = std::to_string(to.value().id().value());
146 if (to.value().type_name() == "method") {
147 const auto &class_participant =
148 model().get_participant<model::method>(to.value().id()).value();
149
150 msg["to"]["participant_id"] =
151 std::to_string(class_participant.class_id().value());
152 }
153 else if (to.value().type_name() == "objc_method") {
154 const auto &class_participant =
155 model()
156 .get_participant<model::objc_method>(to.value().id())
157 .value();
158
159 msg["to"]["participant_id"] =
160 std::to_string(class_participant.class_id().value());
161 }
162 else if (to.value().type_name() == "function" ||
163 to.value().type_name() == "function_template") {
164 if (config().combine_free_functions_into_file_participants()) {
165 const auto &file_participant =
166 model()
167 .get_participant<model::function>(to.value().id())
168 .value();
169 msg["to"]["participant_id"] = std::to_string(
170 common::to_id(file_participant.file_relative()).value());
171 }
172 else {
173 msg["to"]["participant_id"] =
174 std::to_string(to.value().id().value());
175 }
176 }
177 else if (to.value().type_name() == "lambda") {
178 msg["to"]["participant_id"] = std::to_string(to.value().id().value());
179 }
180}
181
184 nlohmann::json &msg) const
185{
186 msg["from"]["activity_id"] = std::to_string(from.value().id().value());
187 if (const auto &cmt = m.comment(); cmt.has_value())
188 msg["comment"] = cmt.value().at("comment");
189
190 if (from.value().type_name() == "method") {
191 const auto &class_participant =
192 model().get_participant<model::method>(from.value().id()).value();
193
194 msg["from"]["participant_id"] =
195 std::to_string(class_participant.class_id().value());
196 }
197 else if (from.value().type_name() == "objc_method") {
198 const auto &class_participant =
199 model()
200 .get_participant<model::objc_method>(from.value().id())
201 .value();
202
203 msg["from"]["participant_id"] =
204 std::to_string(class_participant.class_id().value());
205 }
206 else if (from.value().type_name() == "function" ||
207 from.value().type_name() == "function_template") {
208 if (config().combine_free_functions_into_file_participants()) {
209 const auto &file_participant =
210 model()
211 .get_participant<model::function>(from.value().id())
212 .value();
213 msg["from"]["participant_id"] = std::to_string(
214 common::to_id(file_participant.file_relative()).value());
215 }
216 else {
217 msg["from"]["participant_id"] =
218 std::to_string(from.value().id().value());
219 }
220 }
221 else if (from.value().type_name() == "lambda") {
222 msg["from"]["participant_id"] =
223 std::to_string(from.value().id().value());
224 }
225}
226
228 const activity &a, std::vector<eid_t> &visited) const
229{
230 // Generate calls from this activity to other activities
231 for (const auto &m : a.messages()) {
232 switch (m.type()) {
233 case message_t::kCall:
234 case message_t::kCoAwait:
235 process_call_message(m, visited);
236 break;
237 case message_t::kIf:
239 break;
240 case message_t::kElseIf:
241 case message_t::kElse:
243 break;
244 case message_t::kIfEnd:
246 break;
247 case message_t::kWhile:
249 break;
250 case message_t::kWhileEnd:
252 break;
253 case message_t::kFor:
255 break;
256 case message_t::kForEnd:
258 break;
259 case message_t::kDo:
261 break;
262 case message_t::kDoEnd:
264 break;
265 case message_t::kTry:
267 break;
268 case message_t::kCatch:
270 break;
271 case message_t::kTryEnd:
273 break;
274 case message_t::kSwitch:
276 break;
277 case message_t::kCase:
279 break;
280 case message_t::kSwitchEnd:
282 break;
283 case message_t::kConditional:
285 break;
286 case message_t::kConditionalElse:
288 break;
289 case message_t::kConditionalEnd:
291 break;
292 case message_t::kReturn:
293 case message_t::kCoReturn:
294 case message_t::kCoYield: {
295 auto return_message = m;
296 if (!visited.empty()) {
297 return_message.set_to(visited.back());
298 }
299 process_return_message(return_message);
300 }
301 case message_t::kNone:;
302 }
303 }
304}
305
307{
308 assert(!block_statements_stack_.empty());
309
310 return block_statements_stack_.back().get();
311}
312
314 const model::message &m, std::vector<eid_t> &visited) const
315{
316 visited.push_back(m.from());
317
320 visited.pop_back();
321 return;
322 }
323
325 }
326
327 LOG_DBG("Generating message {} --> {}", m.from(), m.to());
328
330
331 if (model().sequences().find(m.to()) != model().sequences().end()) {
332 if (std::find(visited.begin(), visited.end(), m.to()) ==
333 visited.end()) { // break infinite recursion on recursive calls
334
335 LOG_DBG(
336 "Generating activity {} (called from {})", m.to(), m.from());
337
338 generate_activity(model().get_activity(m.to()), visited);
339 }
340 }
341 else
342 LOG_DBG("Skipping activity {} --> {} - missing sequence {}", m.from(),
343 m.to(), m.to());
344
345 visited.pop_back();
346}
347
349{
350 const auto &from = model().get_participant<model::participant>(m.from());
351 const auto &to = model().get_participant<model::participant>(m.to());
352
353 if (!from) {
354 LOG_DBG("Skipping return call - no participant for id {}", m.from());
355 return;
356 }
357
358 if (!to) {
359 LOG_DBG("Skipping return call - no participant for id {}", m.to());
360 return;
361 }
362
365
366 std::string message;
367
368 if (config().generate_return_types()) {
369 message = m.return_type();
370 }
371 else if (config().generate_return_values()) {
372 message = m.message_name();
373 }
374
375 nlohmann::json msg;
376 msg["name"] = message;
377 msg["type"] = to_string(m.type());
378
379 generate_from_activity(m, from, msg);
380
381 generate_to_activity(to, msg);
382
383 msg["source_location"] =
384 dynamic_cast<const clanguml::common::model::source_location &>(m);
385
386 msg["scope"] = to_string(m.message_scope());
387 msg["return_type"] = config().simplify_template_type(m.return_type());
388
389 current_block_statement()["messages"].push_back(std::move(msg));
390
391 LOG_DBG("Generated return call '{}' from {} [{}] to {} [{}]", message,
392 from.value().full_name(false), m.from(), to.value().full_name(false),
393 m.to());
394}
395
397{
398 nlohmann::json while_block;
399 while_block["type"] = "loop";
400 while_block["name"] = "while";
401 while_block["activity_id"] = std::to_string(m.from().value());
402 if (auto text = m.condition_text(); text.has_value())
403 while_block["condition_text"] = *text;
404
405 current_block_statement()["messages"].push_back(std::move(while_block));
406
407 block_statements_stack_.push_back(
408 std::ref(current_block_statement()["messages"].back()));
409}
410
412{
413 // Remove the while statement block from the stack
414 block_statements_stack_.pop_back();
415}
416
418{
419 nlohmann::json for_block;
420 for_block["type"] = "loop";
421 for_block["name"] = "for";
422 for_block["activity_id"] = std::to_string(m.from().value());
423 if (auto text = m.condition_text(); text.has_value())
424 for_block["condition_text"] = *text;
425
426 current_block_statement()["messages"].push_back(std::move(for_block));
427
428 block_statements_stack_.push_back(
429 std::ref(current_block_statement()["messages"].back()));
430}
431
433{
434 // Remove the while statement block from the stack
435 block_statements_stack_.pop_back();
436}
437
439{
440 nlohmann::json do_block;
441 do_block["type"] = "loop";
442 do_block["name"] = "do";
443 do_block["activity_id"] = std::to_string(m.from().value());
444 if (auto text = m.condition_text(); text.has_value())
445 do_block["condition_text"] = *text;
446
447 current_block_statement()["messages"].push_back(std::move(do_block));
448
449 block_statements_stack_.push_back(
450 std::ref(current_block_statement()["messages"].back()));
451}
452
454{
455 // Remove the do statement block from the stack
456 block_statements_stack_.pop_back();
457}
458
460{
461 nlohmann::json try_block;
462 try_block["type"] = "break";
463 try_block["name"] = "try";
464 try_block["activity_id"] = std::to_string(m.from().value());
465
466 current_block_statement()["messages"].push_back(std::move(try_block));
467
468 block_statements_stack_.push_back(
469 std::ref(current_block_statement()["messages"].back()));
470
471 nlohmann::json branch;
472 branch["type"] = "main";
473 current_block_statement()["branches"].push_back(std::move(branch));
474
475 block_statements_stack_.push_back(
476 std::ref(current_block_statement()["branches"].back()));
477}
478
480{
481 // remove previous block from the stack
482 block_statements_stack_.pop_back();
483
484 nlohmann::json branch;
485 branch["type"] = "catch";
486 current_block_statement()["branches"].push_back(std::move(branch));
487
488 block_statements_stack_.push_back(
489 std::ref(current_block_statement()["branches"].back()));
490}
491
493{
494 // Remove last if block from the stack
495 block_statements_stack_.pop_back();
496
497 // Remove the try statement block from the stack
498 block_statements_stack_.pop_back();
499}
500
502{
503 nlohmann::json if_block;
504 if_block["type"] = "alt";
505 if_block["name"] = "switch";
506 if_block["activity_id"] = std::to_string(m.from().value());
507
508 current_block_statement()["messages"].push_back(std::move(if_block));
509
510 block_statements_stack_.push_back(
511 std::ref(current_block_statement()["messages"].back()));
512}
513
515{
516 if (current_block_statement()["type"] == "case")
517 block_statements_stack_.pop_back();
518
519 nlohmann::json case_block;
520 case_block["type"] = "case";
521 case_block["name"] = m.message_name();
522 current_block_statement()["branches"].push_back(std::move(case_block));
523
524 block_statements_stack_.push_back(
525 std::ref(current_block_statement()["branches"].back()));
526}
527
529{ // Remove last case block from the stack
530 block_statements_stack_.pop_back();
531
532 // Remove the switch statement block from the stack
533 block_statements_stack_.pop_back();
534}
535
537{
538 nlohmann::json if_block;
539 if_block["type"] = "alt";
540 if_block["name"] = "conditional";
541 if_block["activity_id"] = std::to_string(m.from().value());
542 if (auto text = m.condition_text(); text.has_value())
543 if_block["condition_text"] = *text;
544
545 current_block_statement()["messages"].push_back(std::move(if_block));
546
547 block_statements_stack_.push_back(
548 std::ref(current_block_statement()["messages"].back()));
549
550 nlohmann::json branch;
551 branch["type"] = "consequent";
552 current_block_statement()["branches"].push_back(std::move(branch));
553
554 block_statements_stack_.push_back(
555 std::ref(current_block_statement()["branches"].back()));
556}
557
559{
560 // remove previous branch from the stack
561 block_statements_stack_.pop_back();
562
563 nlohmann::json branch;
564 branch["type"] = "alternative";
565 if (auto text = m.condition_text(); text.has_value())
566 branch["condition_text"] = *text;
567 current_block_statement()["branches"].push_back(std::move(branch));
568
569 block_statements_stack_.push_back(
570 std::ref(current_block_statement()["branches"].back()));
571}
572
574{
575 // Remove last if branch from the stack
576 block_statements_stack_.pop_back();
577
578 // Remove the if statement block from the stack
579 block_statements_stack_.pop_back();
580}
581
583{
584 // Remove last if branch from the stack
585 block_statements_stack_.pop_back();
586
587 // Remove the if statement block from the stack
588 block_statements_stack_.pop_back();
589}
590
592{
593 // remove previous branch from the stack
594 block_statements_stack_.pop_back();
595
596 nlohmann::json branch;
597 branch["type"] = "alternative";
598 current_block_statement()["branches"].push_back(std::move(branch));
599
600 block_statements_stack_.push_back(
601 std::ref(current_block_statement()["branches"].back()));
602}
603
605{
606 nlohmann::json if_block;
607 if_block["type"] = "alt";
608 if_block["name"] = "if";
609 if_block["activity_id"] = std::to_string(m.from().value());
610 if (auto text = m.condition_text(); text.has_value())
611 if_block["condition_text"] = *text;
612
613 current_block_statement()["messages"].push_back(std::move(if_block));
614
615 block_statements_stack_.push_back(
616 std::ref(current_block_statement()["messages"].back()));
617
618 nlohmann::json branch;
619 branch["type"] = "consequent";
620 current_block_statement()["branches"].push_back(std::move(branch));
621
622 block_statements_stack_.push_back(
623 std::ref(current_block_statement()["branches"].back()));
624}
625
627 nlohmann::json &parent, const std::string &name) const
628{
629 auto p = model().get(name);
630
631 if (!p.has_value()) {
632 LOG_WARN("Cannot find participant {} from `participants_order` option",
633 name);
634 return;
635 }
636
637 generate_participant(parent, p.value().id(), true);
638}
639
641 nlohmann::json & /*parent*/, eid_t id, bool force) const
642{
643 std::optional<eid_t> participant_id{};
644
645 if (!force) {
646 for (const auto pid : model().active_participants()) {
647 if (pid == id) {
648 participant_id = pid;
649 break;
650 }
651 }
652 }
653 else
654 participant_id = id;
655
656 if (!participant_id.has_value())
657 return participant_id;
658
659 if (is_participant_generated(*participant_id))
660 return participant_id;
661
662 const auto &participant =
663 model().get_participant<model::participant>(*participant_id).value();
664
665 const auto participant_type = participant.type_name();
666
667 if (participant_type == "method") {
668 auto class_participant_id =
669 model()
670 .get_participant<model::method>(*participant_id)
671 .value()
672 .class_id();
673
674 LOG_DBG("Generating JSON method participant: {}",
675 model()
676 .get_participant<model::method>(*participant_id)
677 .value()
678 .full_name(false));
679
680 if (!is_participant_generated(class_participant_id)) {
681 const auto &class_participant =
682 model()
683 .get_participant<model::participant>(class_participant_id)
684 .value();
685
686 generated_participants_.emplace(*participant_id);
687 generated_participants_.emplace(class_participant_id);
688
689 json_["participants"].push_back(class_participant);
690 json_["participants"].back()["activities"].push_back(participant);
691
692 // Perform config dependent postprocessing on generated class
693 const auto class_participant_full_name =
694 display_name_adapter(class_participant).full_name(false);
695
696 json_["participants"].back().at("display_name") =
697 make_display_name(class_participant_full_name);
698
699 return class_participant_id;
700 }
701
702 if (!is_participant_generated(*participant_id)) {
703 for (auto &p : json_["participants"]) {
704 if (p.at("id") ==
705 std::to_string(class_participant_id.value())) {
706 generated_participants_.emplace(*participant_id);
707 p["activities"].push_back(participant);
708 return class_participant_id;
709 }
710 }
711 }
712 }
713 if (participant_type == "objc_method") {
714 auto class_participant_id =
715 model()
716 .get_participant<model::objc_method>(*participant_id)
717 .value()
718 .class_id();
719
720 LOG_DBG("Generating JSON ObjC method participant: {}",
721 model()
722 .get_participant<model::objc_method>(*participant_id)
723 .value()
724 .full_name(false));
725
726 if (!is_participant_generated(class_participant_id)) {
727 const auto &class_participant =
728 model()
729 .get_participant<model::participant>(class_participant_id)
730 .value();
731
732 generated_participants_.emplace(*participant_id);
733 generated_participants_.emplace(class_participant_id);
734
735 json_["participants"].push_back(class_participant);
736 json_["participants"].back()["activities"].push_back(participant);
737
738 // Perform config dependent postprocessing on generated class
739 const auto class_participant_full_name =
740 display_name_adapter(class_participant).full_name(false);
741
742 json_["participants"].back().at("display_name") =
743 make_display_name(class_participant_full_name);
744
745 return class_participant_id;
746 }
747
748 if (!is_participant_generated(*participant_id)) {
749 for (auto &p : json_["participants"]) {
750 if (p.at("id") ==
751 std::to_string(class_participant_id.value())) {
752 generated_participants_.emplace(*participant_id);
753 p["activities"].push_back(participant);
754 return class_participant_id;
755 }
756 }
757 }
758 }
759 else if ((participant_type == "function" ||
760 participant_type == "function_template") &&
761 config().combine_free_functions_into_file_participants()) {
762 // Create a single participant for all functions declared in a
763 // single file
764 // participant_id will become activity_id within a file participant
765 const auto &function_participant =
766 model().get_participant<model::function>(*participant_id).value();
767
768 const auto file_participant_id =
769 common::to_id(function_participant.file_relative());
770
771 if (!is_participant_generated(file_participant_id)) {
772 nlohmann::json p = function_participant;
773
774 const auto file_path =
775 config().make_path_relative(function_participant.file());
776
777 p["display_name"] = util::path_to_url(file_path.string());
778 p["name"] = file_path.filename();
779
780 if (is_participant_generated(file_participant_id))
781 return participant_id;
782
783 p["id"] = std::to_string(file_participant_id.value());
784 p["type"] = "file";
785 p.erase("source_location");
786
787 generated_participants_.emplace(participant_id.value());
788
789 p["activities"].push_back(participant);
790 json_["participants"].push_back(p);
791
792 generated_participants_.emplace(file_participant_id);
793
794 return file_participant_id;
795 }
796
797 if (!is_participant_generated(*participant_id)) {
798 for (auto &p : json_["participants"]) {
799 if (p.at("id") == std::to_string(file_participant_id.value())) {
800 generated_participants_.emplace(*participant_id);
801 p["activities"].push_back(participant);
802 }
803 }
804 }
805
806 return file_participant_id;
807 }
808 else {
809 json_["participants"].push_back(participant);
810 const auto function_participant_full_name =
811 display_name_adapter(participant).full_name(false);
812
813 json_["participants"].back().at("display_name") =
814 make_display_name(function_participant_full_name);
815 }
816
817 generated_participants_.emplace(*participant_id);
818
819 return participant_id;
820}
821
823{
824 return std::find(generated_participants_.begin(),
826 id) != generated_participants_.end();
827}
828
829void generator::generate_diagram(nlohmann::json &parent) const
830{
831 model().print();
832
833 if (config().using_namespace)
834 parent["using_namespace"] = config().using_namespace().to_string();
835
836 if (config().participants_order.has_value) {
837 for (const auto &p : config().participants_order()) {
838 LOG_DBG("Pregenerating participant {}", p);
840 }
841 }
842
844
845 generate_to_sequences(parent);
846
848
849 // Perform config dependent postprocessing on generated participants
850 for (auto &p : json_["participants"]) {
851 if (p.contains("display_name")) {
852 p["display_name"] = make_display_name(p["display_name"]);
853 }
854 }
855
856 parent["participants"] = json_["participants"];
857}
858
859void generator::generate_from_sequences(nlohmann::json &parent) const
860{
861 std::vector<eid_t> start_from = find_from_activities();
862
863 // Use this to break out of recurrent loops
864 std::vector<eid_t> visited_participants;
865 for (const auto from_id : start_from) {
866
867 const auto &from = model().get_participant<model::function>(from_id);
868
869 if (!from.has_value()) {
870 LOG_WARN("Failed to find participant {} for 'from' "
871 "condition");
872 continue;
873 }
874
875 generate_participant(json_, from_id);
876
877 [[maybe_unused]] model::function::message_render_mode render_mode =
879
880 nlohmann::json sequence;
881 sequence["from"]["location"] = from.value().full_name(false);
882 sequence["from"]["id"] = std::to_string(from_id.value());
883
884 block_statements_stack_.push_back(std::ref(sequence));
885
886 generate_activity(model().get_activity(from_id), visited_participants);
887
888 block_statements_stack_.pop_back();
889
890 if (from.value().type_name() == "method" ||
891 config().combine_free_functions_into_file_participants()) {
892
893 sequence["return_type"] =
894 make_display_name(from.value().return_type());
895 }
896
897 parent["sequences"].push_back(std::move(sequence));
898 }
899}
900
901void generator::generate_to_sequences(nlohmann::json &parent) const
902{
903 for (const auto &to_location : config().to()) {
904 auto to_activity_ids = model().get_to_activity_ids(to_location);
905
906 if (to_activity_ids.empty()) {
907 LOG_WARN("Failed to find participant matching '{}' for "
908 "'to' condition: ",
909 to_location.location.to_string());
910 }
911
912 for (const auto to_activity_id : to_activity_ids) {
913 const auto &to =
914 model().get_participant<model::function>(to_activity_id);
915
916 nlohmann::json sequence;
917 sequence["to"]["location"] = to.value().full_name(false);
918 sequence["to"]["id"] = std::to_string(to_activity_id.value());
919 sequence["message_chains"] = nlohmann::json::array();
920
921 block_statements_stack_.push_back(std::ref(sequence));
922
923 std::vector<model::message_chain_t> message_chains =
924 model().get_all_from_to_message_chains(eid_t{}, to_activity_id);
925
926 for (const auto &mc : message_chains) {
927 const auto from_activity_id = mc.front().from();
928
929 if (model().participants().count(from_activity_id) == 0)
930 continue;
931
932 nlohmann::json message_chain;
933
934 block_statements_stack_.push_back(std::ref(message_chain));
935
936 for (const auto &m : mc) {
938 }
939
940 block_statements_stack_.pop_back();
941
942 sequence["message_chains"].push_back(std::move(message_chain));
943 }
944
945 block_statements_stack_.pop_back();
946
947 parent["sequences"].push_back(std::move(sequence));
948 }
949 }
950}
951
952void generator::generate_from_to_sequences(nlohmann::json &parent) const
953{
954 for (const auto &ft : config().from_to()) {
955 // First, find the sequence of activities from 'from' location
956 // to 'to' location
957 assert(ft.size() == 2);
958
959 const auto &from_location = ft.front();
960 const auto &to_location = ft.back();
961
962 auto from_activity_ids = model().get_from_activity_ids(from_location);
963 auto to_activity_ids = model().get_to_activity_ids(to_location);
964
965 if (from_activity_ids.empty()) {
967 model().name(),
968 fmt::format("Failed to find participant matching '{}' for "
969 "'from' condition: ",
970 from_location.location.to_string()));
971 }
972
973 if (from_activity_ids.empty() || to_activity_ids.empty()) {
975 model().name(),
976 fmt::format("Failed to find participant matching '{}' for "
977 "'to' condition: ",
978 to_location.location.to_string()));
979 }
980
981 for (const auto from_activity_id : from_activity_ids) {
982 if (model().participants().count(from_activity_id) == 0)
983 continue;
984
985 const auto &from =
986 model().get_participant<model::function>(from_activity_id);
987
988 for (const auto to_activity_id : to_activity_ids) {
989 if (model().participants().count(to_activity_id) == 0)
990 continue;
991
992 const auto &to =
993 model().get_participant<model::function>(to_activity_id);
994
995 auto message_chains_unique =
996 model().get_all_from_to_message_chains(
997 from_activity_id, to_activity_id);
998
999 nlohmann::json sequence;
1000 sequence["from_to"]["from"]["location"] =
1001 from.value().full_name(false);
1002 sequence["from_to"]["from"]["id"] =
1003 std::to_string(from_activity_id.value());
1004 sequence["from_to"]["to"]["location"] =
1005 to.value().full_name(false);
1006 sequence["from_to"]["to"]["id"] =
1007 std::to_string(to_activity_id.value());
1008
1009 block_statements_stack_.push_back(std::ref(sequence));
1010
1011 sequence["message_chains"] = nlohmann::json::array();
1012
1013 for (const auto &mc : message_chains_unique) {
1014 nlohmann::json message_chain;
1015
1016 block_statements_stack_.push_back(std::ref(message_chain));
1017
1018 for (const auto &m : mc) {
1020 }
1021
1022 block_statements_stack_.pop_back();
1023
1024 sequence["message_chains"].push_back(
1025 std::move(message_chain));
1026 }
1027
1028 block_statements_stack_.pop_back();
1029
1030 parent["sequences"].push_back(std::move(sequence));
1031 }
1032 }
1033 }
1034}
1035
1036std::vector<eid_t> generator::find_from_activities() const
1037{
1038 std::vector<eid_t> start_from;
1039 for (const auto &sf : config().from()) {
1040 if (sf.location_type == location_t::function) {
1041 bool found{false};
1042 for (const auto &[k, v] : model().sequences()) {
1043 if (model().participants().count(v.from()) == 0)
1044 continue;
1045
1046 const auto &caller = *model().participants().at(v.from());
1047 std::string vfrom = caller.full_name(false);
1048 if (sf.location == vfrom) {
1049 LOG_DBG("Found sequence diagram start point: {}", k);
1050 start_from.push_back(k);
1051 found = true;
1052 }
1053 }
1054
1055 if (!found)
1057 model().name(),
1058 fmt::format("Failed to find participant matching '{}' for "
1059 "'from' condition: ",
1060 sf.location.to_string()));
1061 }
1062 }
1063
1064 return start_from;
1065}
1066
1067std::string generator::make_display_name(const std::string &full_name) const
1068{
1069 auto result = config().simplify_template_type(full_name);
1070 result = config().using_namespace().relative(result);
1072
1073 return result;
1074}
1075
1076} // namespace clanguml::sequence_diagram::generators::json