0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
Public Member Functions | Private Member Functions | Private Attributes | List of all members
clanguml::sequence_diagram::generators::json::generator Class Reference

Sequence diagram JSON generator. More...

Detailed Description

Sequence diagram JSON generator.

Definition at line 46 of file sequence_diagram_generator.h.

#include <sequence_diagram_generator.h>

Public Member Functions

 generator (diagram_config &config, diagram_model &model)
 
void generate_diagram (nlohmann::json &parent) const override
 Main generator method.
 
void generate_call (const sequence_diagram::model::message &m, nlohmann::json &parent) const
 Generate sequence diagram message.
 
std::optional< eid_tgenerate_participant (nlohmann::json &parent, eid_t id, bool force=false) const
 Generate sequence diagram participant.
 
void generate_participant (nlohmann::json &parent, const std::string &name) const
 Generate sequence diagram participant by name.
 
void generate_activity (const sequence_diagram::model::activity &a, std::vector< eid_t > &visited) const
 Generate sequence diagram activity.
 
nlohmann::json & current_block_statement () const
 Get reference to the current block statement.
 
std::string make_display_name (const std::string &full_name) const
 
- Public Member Functions inherited from clanguml::common::generators::json::generator< ConfigType, DiagramType >
 ~generator () override=default
 
void generate (std::ostream &ostr) const override
 Generate diagram.
 
virtual void generate_diagram (nlohmann::json &parent) const =0
 Generate diagram model.
 
void generate_metadata (nlohmann::json &parent) const
 Generate metadata element with diagram metadata.
 
- Public Member Functions inherited from clanguml::common::generators::generator< ConfigType, DiagramType >
 generator (ConfigType &config, DiagramType &model)
 Constructor.
 
virtual ~generator ()=default
 
virtual void generate (std::ostream &ostr) const =0
 Generate diagram.
 
const ConfigType & config () const
 Get reference to diagram config.
 
const DiagramType & model () const
 Get reference to diagram model.
 
std::optional< std::pair< std::string, std::string > > get_link_pattern (const common::model::source_location &sl) const
 
std::optional< std::pair< std::string, std::string > > get_tooltip_pattern (const common::model::source_location &sl) const
 
std::optional< std::string > render_link (const common::model::diagram_element &e) const
 
std::optional< std::string > render_link (const common::model::relationship &e) const
 
std::optional< std::string > render_tooltip (const common::model::diagram_element &e) const
 
std::optional< std::string > render_tooltip (const common::model::relationship &e) const
 
void init_context ()
 Initialize diagram Jinja context.
 
void update_context () const
 Update diagram Jinja context.
 
void init_env ()
 
const inja::json & context () const
 
inja::Environment & env () const
 

Private Member Functions

bool is_participant_generated (eid_t id) const
 Check if specified participant has already been generated.
 
void process_call_message (const model::message &m, std::vector< eid_t > &visited) const
 Process call message.
 
void process_return_message (const model::message &m) const
 Process return message.
 
void process_if_message (const model::message &m) const
 Process if statement message.
 
void process_else_if_message () const
 Process else if statement message.
 
void process_end_if_message () const
 Process end if statement message.
 
void process_conditional_message (const model::message &m) const
 Process :? statement message.
 
void process_end_conditional_message () const
 Process end of conditional statement message.
 
void process_conditional_else_message (const model::message &m) const
 Process conditional else statement message.
 
void process_switch_message (const model::message &m) const
 Process switch statement message.
 
void process_end_switch_message () const
 Process switch end statement message.
 
void process_case_message (const model::message &m) const
 Process switch case statement message.
 
void process_try_message (const model::message &m) const
 Process try statement message.
 
void process_end_try_message () const
 Process try end statement message.
 
void process_catch_message () const
 Process catch statement message.
 
void process_do_message (const model::message &m) const
 Process do loop statement message.
 
void process_end_do_message () const
 Process do end statement message.
 
void process_for_message (const model::message &m) const
 Process for loop statement message.
 
void process_end_for_message () const
 Process for end statement message.
 
void process_while_message (const model::message &m) const
 Process while loop message.
 
void process_end_while_message () const
 Process while end loop message.
 
void generate_from_to_sequences (nlohmann::json &parent) const
 
void generate_to_sequences (nlohmann::json &parent) const
 
void generate_from_sequences (nlohmann::json &parent) const
 
void generate_from_activity (const model::message &m, const common::optional_ref< model::participant > &from, nlohmann::json &msg) const
 
void generate_to_activity (const common::optional_ref< model::participant > &to, nlohmann::json &msg) const
 
std::vector< eid_tfind_from_activities () const
 

Private Attributes

std::set< eid_tgenerated_participants_
 
nlohmann::json json_
 
std::vector< std::reference_wrapper< nlohmann::json > > block_statements_stack_
 
std::vector< model::messagealready_generated_in_static_context_
 

Additional Inherited Members

- Protected Attributes inherited from clanguml::common::generators::generator< ConfigType, DiagramType >
inja::json m_context
 
inja::Environment m_env
 

Constructor & Destructor Documentation

◆ generator()

clanguml::sequence_diagram::generators::json::generator::generator ( diagram_config config,
diagram_model model 
)

Definition at line 71 of file sequence_diagram_generator.cc.

73 : common_generator<diagram_config, diagram_model>{config, model}
74{
75}

Member Function Documentation

◆ current_block_statement()

nlohmann::json & clanguml::sequence_diagram::generators::json::generator::current_block_statement ( ) const

Get reference to the current block statement.

This method returns a reference to the last block statement (e.g if statement or for loop) in the call stack.

Returns
Reference to the current block statement.

Definition at line 306 of file sequence_diagram_generator.cc.

307{
308 assert(!block_statements_stack_.empty());
309
310 return block_statements_stack_.back().get();
311}

◆ find_from_activities()

std::vector< eid_t > clanguml::sequence_diagram::generators::json::generator::find_from_activities ( ) const
private

Definition at line 1036 of file sequence_diagram_generator.cc.

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)
1056 throw error::invalid_sequence_from_condition(model().type(),
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}

◆ generate_activity()

void clanguml::sequence_diagram::generators::json::generator::generate_activity ( const sequence_diagram::model::activity a,
std::vector< eid_t > &  visited 
) const

Generate sequence diagram activity.

Parameters
aActivity model
visitedList of already visited participants, this is necessary for breaking infinite recursion on recursive calls

Definition at line 227 of file sequence_diagram_generator.cc.

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}

◆ generate_call()

void clanguml::sequence_diagram::generators::json::generator::generate_call ( const sequence_diagram::model::message m,
nlohmann::json &  parent 
) const

Generate sequence diagram message.

Parameters
mMessage model
parentJSON node

Definition at line 77 of file sequence_diagram_generator.cc.

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
87 generate_participant(json_, m.from());
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}

◆ generate_diagram()

void clanguml::sequence_diagram::generators::json::generator::generate_diagram ( nlohmann::json &  parent) const
overridevirtual

Main generator method.

This method is called first and coordinates the entire diagram generation.

Parameters
ostrOutput stream.

Implements clanguml::common::generators::json::generator< ConfigType, DiagramType >.

Definition at line 829 of file sequence_diagram_generator.cc.

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}

◆ generate_from_activity()

void clanguml::sequence_diagram::generators::json::generator::generate_from_activity ( const model::message m,
const common::optional_ref< model::participant > &  from,
nlohmann::json &  msg 
) const
private

Definition at line 182 of file sequence_diagram_generator.cc.

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}

◆ generate_from_sequences()

void clanguml::sequence_diagram::generators::json::generator::generate_from_sequences ( nlohmann::json &  parent) const
private

Definition at line 859 of file sequence_diagram_generator.cc.

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}

◆ generate_from_to_sequences()

void clanguml::sequence_diagram::generators::json::generator::generate_from_to_sequences ( nlohmann::json &  parent) const
private

Definition at line 952 of file sequence_diagram_generator.cc.

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()) {
966 throw error::invalid_sequence_from_condition(model().type(),
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()) {
974 throw error::invalid_sequence_to_condition(model().type(),
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}

◆ generate_participant() [1/2]

void clanguml::sequence_diagram::generators::json::generator::generate_participant ( nlohmann::json &  parent,
const std::string &  name 
) const

Generate sequence diagram participant by name.

This is convienience wrapper over generate_participant() by id.

Parameters
parentJSON node
nameFull participant name

Definition at line 626 of file sequence_diagram_generator.cc.

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}

◆ generate_participant() [2/2]

std::optional< eid_t > clanguml::sequence_diagram::generators::json::generator::generate_participant ( nlohmann::json &  parent,
eid_t  id,
bool  force = false 
) const

Generate sequence diagram participant.

Parameters
parentJSON node
idParticipant id
forceIf true, generate the participant even if its not in the set of active participants
Returns
Id of the generated participant

Definition at line 640 of file sequence_diagram_generator.cc.

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}

◆ generate_to_activity()

void clanguml::sequence_diagram::generators::json::generator::generate_to_activity ( const common::optional_ref< model::participant > &  to,
nlohmann::json &  msg 
) const
private

Definition at line 141 of file sequence_diagram_generator.cc.

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}

◆ generate_to_sequences()

void clanguml::sequence_diagram::generators::json::generator::generate_to_sequences ( nlohmann::json &  parent) const
private

Definition at line 901 of file sequence_diagram_generator.cc.

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}

◆ is_participant_generated()

bool clanguml::sequence_diagram::generators::json::generator::is_participant_generated ( eid_t  id) const
private

Check if specified participant has already been generated.

Parameters
idParticipant id.
Returns
True, if participant has already been generated.

Definition at line 822 of file sequence_diagram_generator.cc.

823{
824 return std::find(generated_participants_.begin(),
826 id) != generated_participants_.end();
827}

◆ make_display_name()

std::string clanguml::sequence_diagram::generators::json::generator::make_display_name ( const std::string &  full_name) const

Definition at line 1067 of file sequence_diagram_generator.cc.

1068{
1069 auto result = config().simplify_template_type(full_name);
1070 result = config().using_namespace().relative(result);
1072
1073 return result;
1074}

◆ process_call_message()

void clanguml::sequence_diagram::generators::json::generator::process_call_message ( const model::message m,
std::vector< eid_t > &  visited 
) const
private

Process call message.

Parameters
mMessage model
visitedList of already visited participants

Definition at line 313 of file sequence_diagram_generator.cc.

315{
316 visited.push_back(m.from());
317
318 if (m.in_static_declaration_context()) {
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}

◆ process_case_message()

void clanguml::sequence_diagram::generators::json::generator::process_case_message ( const model::message m) const
private

Process switch case statement message.

Parameters
mMessage model

Definition at line 514 of file sequence_diagram_generator.cc.

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}

◆ process_catch_message()

void clanguml::sequence_diagram::generators::json::generator::process_catch_message ( ) const
private

Process catch statement message.

Definition at line 479 of file sequence_diagram_generator.cc.

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}

◆ process_conditional_else_message()

void clanguml::sequence_diagram::generators::json::generator::process_conditional_else_message ( const model::message m) const
private

Process conditional else statement message.

Parameters
mMessage model

Definition at line 558 of file sequence_diagram_generator.cc.

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}

◆ process_conditional_message()

void clanguml::sequence_diagram::generators::json::generator::process_conditional_message ( const model::message m) const
private

Process :? statement message.

Parameters
mMessage model

Definition at line 536 of file sequence_diagram_generator.cc.

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}

◆ process_do_message()

void clanguml::sequence_diagram::generators::json::generator::process_do_message ( const model::message m) const
private

Process do loop statement message.

Parameters
mMessage model

Definition at line 438 of file sequence_diagram_generator.cc.

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}

◆ process_else_if_message()

void clanguml::sequence_diagram::generators::json::generator::process_else_if_message ( ) const
private

Process else if statement message.

Definition at line 591 of file sequence_diagram_generator.cc.

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}

◆ process_end_conditional_message()

void clanguml::sequence_diagram::generators::json::generator::process_end_conditional_message ( ) const
private

Process end of conditional statement message.

Definition at line 573 of file sequence_diagram_generator.cc.

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}

◆ process_end_do_message()

void clanguml::sequence_diagram::generators::json::generator::process_end_do_message ( ) const
private

Process do end statement message.

Definition at line 453 of file sequence_diagram_generator.cc.

454{
455 // Remove the do statement block from the stack
456 block_statements_stack_.pop_back();
457}

◆ process_end_for_message()

void clanguml::sequence_diagram::generators::json::generator::process_end_for_message ( ) const
private

Process for end statement message.

Definition at line 432 of file sequence_diagram_generator.cc.

433{
434 // Remove the while statement block from the stack
435 block_statements_stack_.pop_back();
436}

◆ process_end_if_message()

void clanguml::sequence_diagram::generators::json::generator::process_end_if_message ( ) const
private

Process end if statement message.

Definition at line 582 of file sequence_diagram_generator.cc.

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}

◆ process_end_switch_message()

void clanguml::sequence_diagram::generators::json::generator::process_end_switch_message ( ) const
private

Process switch end statement message.

Definition at line 528 of file sequence_diagram_generator.cc.

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}

◆ process_end_try_message()

void clanguml::sequence_diagram::generators::json::generator::process_end_try_message ( ) const
private

Process try end statement message.

Definition at line 492 of file sequence_diagram_generator.cc.

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}

◆ process_end_while_message()

void clanguml::sequence_diagram::generators::json::generator::process_end_while_message ( ) const
private

Process while end loop message.

Definition at line 411 of file sequence_diagram_generator.cc.

412{
413 // Remove the while statement block from the stack
414 block_statements_stack_.pop_back();
415}

◆ process_for_message()

void clanguml::sequence_diagram::generators::json::generator::process_for_message ( const model::message m) const
private

Process for loop statement message.

Parameters
mMessage model

Definition at line 417 of file sequence_diagram_generator.cc.

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}

◆ process_if_message()

void clanguml::sequence_diagram::generators::json::generator::process_if_message ( const model::message m) const
private

Process if statement message.

Parameters
mMessage model

Definition at line 604 of file sequence_diagram_generator.cc.

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}

◆ process_return_message()

void clanguml::sequence_diagram::generators::json::generator::process_return_message ( const model::message m) const
private

Process return message.

Parameters
mMessage model

Definition at line 348 of file sequence_diagram_generator.cc.

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
363 generate_participant(json_, m.from());
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}

◆ process_switch_message()

void clanguml::sequence_diagram::generators::json::generator::process_switch_message ( const model::message m) const
private

Process switch statement message.

Parameters
mMessage model

Definition at line 501 of file sequence_diagram_generator.cc.

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}

◆ process_try_message()

void clanguml::sequence_diagram::generators::json::generator::process_try_message ( const model::message m) const
private

Process try statement message.

Parameters
mMessage model

Definition at line 459 of file sequence_diagram_generator.cc.

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}

◆ process_while_message()

void clanguml::sequence_diagram::generators::json::generator::process_while_message ( const model::message m) const
private

Process while loop message.

Parameters
mMessage model

Definition at line 396 of file sequence_diagram_generator.cc.

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}

Member Data Documentation

◆ already_generated_in_static_context_

std::vector<model::message> clanguml::sequence_diagram::generators::json::generator::already_generated_in_static_context_
mutableprivate

Definition at line 274 of file sequence_diagram_generator.h.

◆ block_statements_stack_

std::vector<std::reference_wrapper<nlohmann::json> > clanguml::sequence_diagram::generators::json::generator::block_statements_stack_
mutableprivate

Definition at line 272 of file sequence_diagram_generator.h.

◆ generated_participants_

std::set<eid_t> clanguml::sequence_diagram::generators::json::generator::generated_participants_
mutableprivate

Definition at line 265 of file sequence_diagram_generator.h.

◆ json_

nlohmann::json clanguml::sequence_diagram::generators::json::generator::json_
mutableprivate

Definition at line 269 of file sequence_diagram_generator.h.


The documentation for this class was generated from the following files: