31 j[
"name"] =
dynamic_cast<const method &
>(c).method_name();
33 else if (c.
type_name() ==
"objc_method") {
34 j[
"name"] =
dynamic_cast<const objc_method &
>(c).method_name();
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;
46 j[
"is_coroutine"] =
true;
52 j[
"participant_id"] = std::to_string(c.
from().
value());
83 LOG_DBG(
"Skipping empty call from '{}' to '{}'", m.
from(), m.
to());
97 .message_name(render_mode);
99 else if (to.value().
type_name() ==
"objc_method") {
101 .message_name(render_mode);
103 else if (
config().combine_free_functions_into_file_participants()) {
104 if (to.value().
type_name() ==
"function") {
106 .message_name(render_mode);
108 else if (to.value().
type_name() ==
"function_template") {
110 .message_name(render_mode);
119 if (m.
type() == message_t::kCoAwait)
120 msg[
"type"] =
"co_await";
122 msg[
"type"] =
"message";
128 msg[
"source_location"] =
134 parent[
"messages"].push_back(std::move(msg));
143 nlohmann::json &msg)
const
145 msg[
"to"][
"activity_id"] = std::to_string(to.
value().id().value());
146 if (to.
value().type_name() ==
"method") {
147 const auto &class_participant =
150 msg[
"to"][
"participant_id"] =
151 std::to_string(class_participant.class_id().value());
153 else if (to.
value().type_name() ==
"objc_method") {
154 const auto &class_participant =
159 msg[
"to"][
"participant_id"] =
160 std::to_string(class_participant.class_id().value());
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 =
169 msg[
"to"][
"participant_id"] = std::to_string(
173 msg[
"to"][
"participant_id"] =
174 std::to_string(to.
value().id().value());
177 else if (to.
value().type_name() ==
"lambda") {
178 msg[
"to"][
"participant_id"] = std::to_string(to.
value().id().value());
184 nlohmann::json &msg)
const
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");
190 if (from.
value().type_name() ==
"method") {
191 const auto &class_participant =
194 msg[
"from"][
"participant_id"] =
195 std::to_string(class_participant.class_id().value());
197 else if (from.
value().type_name() ==
"objc_method") {
198 const auto &class_participant =
203 msg[
"from"][
"participant_id"] =
204 std::to_string(class_participant.class_id().value());
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 =
213 msg[
"from"][
"participant_id"] = std::to_string(
217 msg[
"from"][
"participant_id"] =
218 std::to_string(from.
value().id().value());
221 else if (from.
value().type_name() ==
"lambda") {
222 msg[
"from"][
"participant_id"] =
223 std::to_string(from.
value().id().value());
228 const activity &a, std::vector<eid_t> &visited)
const
231 for (
const auto &m : a.
messages()) {
233 case message_t::kCall:
234 case message_t::kCoAwait:
240 case message_t::kElseIf:
241 case message_t::kElse:
244 case message_t::kIfEnd:
247 case message_t::kWhile:
250 case message_t::kWhileEnd:
253 case message_t::kFor:
256 case message_t::kForEnd:
262 case message_t::kDoEnd:
265 case message_t::kTry:
268 case message_t::kCatch:
271 case message_t::kTryEnd:
274 case message_t::kSwitch:
277 case message_t::kCase:
280 case message_t::kSwitchEnd:
283 case message_t::kConditional:
286 case message_t::kConditionalElse:
289 case message_t::kConditionalEnd:
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());
301 case message_t::kNone:;
316 visited.push_back(m.
from());
331 if (
model().sequences().find(m.
to()) !=
model().sequences().end()) {
332 if (std::find(visited.begin(), visited.end(), m.
to()) ==
336 "Generating activity {} (called from {})", m.
to(), m.
from());
342 LOG_DBG(
"Skipping activity {} --> {} - missing sequence {}", m.
from(),
354 LOG_DBG(
"Skipping return call - no participant for id {}", m.
from());
359 LOG_DBG(
"Skipping return call - no participant for id {}", m.
to());
368 if (
config().generate_return_types()) {
371 else if (
config().generate_return_values()) {
377 msg[
"type"] = to_string(m.
type());
383 msg[
"source_location"] =
391 LOG_DBG(
"Generated return call '{}' from {} [{}] to {} [{}]",
message,
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());
403 while_block[
"condition_text"] = *text;
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());
424 for_block[
"condition_text"] = *text;
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());
445 do_block[
"condition_text"] = *text;
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());
471 nlohmann::json branch;
472 branch[
"type"] =
"main";
484 nlohmann::json branch;
485 branch[
"type"] =
"catch";
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());
519 nlohmann::json case_block;
520 case_block[
"type"] =
"case";
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());
543 if_block[
"condition_text"] = *text;
550 nlohmann::json branch;
551 branch[
"type"] =
"consequent";
563 nlohmann::json branch;
564 branch[
"type"] =
"alternative";
566 branch[
"condition_text"] = *text;
596 nlohmann::json branch;
597 branch[
"type"] =
"alternative";
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());
611 if_block[
"condition_text"] = *text;
618 nlohmann::json branch;
619 branch[
"type"] =
"consequent";
627 nlohmann::json &parent,
const std::string &name)
const
629 auto p =
model().get(name);
631 if (!p.has_value()) {
632 LOG_WARN(
"Cannot find participant {} from `participants_order` option",
641 nlohmann::json & ,
eid_t id,
bool force)
const
643 std::optional<eid_t> participant_id{};
646 for (
const auto pid :
model().active_participants()) {
648 participant_id = pid;
656 if (!participant_id.has_value())
657 return participant_id;
660 return participant_id;
662 const auto &participant =
665 const auto participant_type = participant.
type_name();
667 if (participant_type ==
"method") {
668 auto class_participant_id =
674 LOG_DBG(
"Generating JSON method participant: {}",
676 .get_participant<model::method>(*participant_id)
681 const auto &class_participant =
689 json_[
"participants"].push_back(class_participant);
690 json_[
"participants"].back()[
"activities"].push_back(participant);
693 const auto class_participant_full_name =
696 json_[
"participants"].back().at(
"display_name") =
699 return class_participant_id;
703 for (
auto &p :
json_[
"participants"]) {
705 std::to_string(class_participant_id.value())) {
707 p[
"activities"].push_back(participant);
708 return class_participant_id;
713 if (participant_type ==
"objc_method") {
714 auto class_participant_id =
720 LOG_DBG(
"Generating JSON ObjC method participant: {}",
722 .get_participant<model::objc_method>(*participant_id)
727 const auto &class_participant =
735 json_[
"participants"].push_back(class_participant);
736 json_[
"participants"].back()[
"activities"].push_back(participant);
739 const auto class_participant_full_name =
742 json_[
"participants"].back().at(
"display_name") =
745 return class_participant_id;
749 for (
auto &p :
json_[
"participants"]) {
751 std::to_string(class_participant_id.value())) {
753 p[
"activities"].push_back(participant);
754 return class_participant_id;
759 else if ((participant_type ==
"function" ||
760 participant_type ==
"function_template") &&
761 config().combine_free_functions_into_file_participants()) {
765 const auto &function_participant =
768 const auto file_participant_id =
772 nlohmann::json p = function_participant;
774 const auto file_path =
775 config().make_path_relative(function_participant.file());
778 p[
"name"] = file_path.filename();
781 return participant_id;
783 p[
"id"] = std::to_string(file_participant_id.value());
785 p.erase(
"source_location");
789 p[
"activities"].push_back(participant);
790 json_[
"participants"].push_back(p);
794 return file_participant_id;
798 for (
auto &p :
json_[
"participants"]) {
799 if (p.at(
"id") == std::to_string(file_participant_id.value())) {
801 p[
"activities"].push_back(participant);
806 return file_participant_id;
809 json_[
"participants"].push_back(participant);
810 const auto function_participant_full_name =
813 json_[
"participants"].back().at(
"display_name") =
819 return participant_id;
833 if (
config().using_namespace)
834 parent[
"using_namespace"] =
config().using_namespace().to_string();
836 if (
config().participants_order.has_value) {
837 for (
const auto &p :
config().participants_order()) {
838 LOG_DBG(
"Pregenerating participant {}", p);
850 for (
auto &p :
json_[
"participants"]) {
851 if (p.contains(
"display_name")) {
856 parent[
"participants"] =
json_[
"participants"];
864 std::vector<eid_t> visited_participants;
865 for (
const auto from_id : start_from) {
869 if (!from.has_value()) {
870 LOG_WARN(
"Failed to find participant {} for 'from' "
880 nlohmann::json sequence;
881 sequence[
"from"][
"location"] = from.value().full_name(
false);
882 sequence[
"from"][
"id"] = std::to_string(from_id.value());
890 if (from.value().type_name() ==
"method" ||
891 config().combine_free_functions_into_file_participants()) {
893 sequence[
"return_type"] =
897 parent[
"sequences"].push_back(std::move(sequence));
903 for (
const auto &to_location :
config().to()) {
904 auto to_activity_ids =
model().get_to_activity_ids(to_location);
906 if (to_activity_ids.empty()) {
907 LOG_WARN(
"Failed to find participant matching '{}' for "
909 to_location.location.to_string());
912 for (
const auto to_activity_id : to_activity_ids) {
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();
923 std::vector<model::message_chain_t> message_chains =
924 model().get_all_from_to_message_chains(
eid_t{}, to_activity_id);
926 for (
const auto &mc : message_chains) {
927 const auto from_activity_id = mc.front().from();
929 if (
model().participants().count(from_activity_id) == 0)
932 nlohmann::json message_chain;
936 for (
const auto &m : mc) {
942 sequence[
"message_chains"].push_back(std::move(message_chain));
947 parent[
"sequences"].push_back(std::move(sequence));
954 for (
const auto &ft :
config().from_to()) {
957 assert(ft.size() == 2);
959 const auto &from_location = ft.front();
960 const auto &to_location = ft.back();
962 auto from_activity_ids =
model().get_from_activity_ids(from_location);
963 auto to_activity_ids =
model().get_to_activity_ids(to_location);
965 if (from_activity_ids.empty()) {
968 fmt::format(
"Failed to find participant matching '{}' for "
969 "'from' condition: ",
970 from_location.location.to_string()));
973 if (from_activity_ids.empty() || to_activity_ids.empty()) {
976 fmt::format(
"Failed to find participant matching '{}' for "
978 to_location.location.to_string()));
981 for (
const auto from_activity_id : from_activity_ids) {
982 if (
model().participants().count(from_activity_id) == 0)
988 for (
const auto to_activity_id : to_activity_ids) {
989 if (
model().participants().count(to_activity_id) == 0)
995 auto message_chains_unique =
996 model().get_all_from_to_message_chains(
997 from_activity_id, to_activity_id);
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());
1011 sequence[
"message_chains"] = nlohmann::json::array();
1013 for (
const auto &mc : message_chains_unique) {
1014 nlohmann::json message_chain;
1018 for (
const auto &m : mc) {
1024 sequence[
"message_chains"].push_back(
1025 std::move(message_chain));
1030 parent[
"sequences"].push_back(std::move(sequence));
1038 std::vector<eid_t> start_from;
1039 for (
const auto &sf :
config().from()) {
1040 if (sf.location_type == location_t::function) {
1042 for (
const auto &[k, v] :
model().sequences()) {
1043 if (
model().participants().count(v.from()) == 0)
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);
1058 fmt::format(
"Failed to find participant matching '{}' for "
1059 "'from' condition: ",
1060 sf.location.to_string()));
1069 auto result =
config().simplify_template_type(full_name);
1070 result =
config().using_namespace().relative(result);