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::plantuml::generator Class Reference

Sequence diagram PlantUML generator. More...

Detailed Description

Sequence diagram PlantUML generator.

Definition at line 50 of file sequence_diagram_generator.h.

#include <sequence_diagram_generator.h>

Public Member Functions

 generator (diagram_config &config, diagram_model &model)
 
void generate_diagram (std::ostream &ostr) const override
 Main generator method.
 
void generate_call (const clanguml::sequence_diagram::model::message &m, std::ostream &ostr) const
 Generate sequence diagram message.
 
void generate_return (const clanguml::sequence_diagram::model::message &m, std::ostream &ostr) const
 Generate sequence diagram return message.
 
void generate_participant (std::ostream &ostr, eid_t id, bool force=false) const
 Generate sequence diagram participant.
 
void generate_participant (std::ostream &ostr, const std::string &name) const
 Generate sequence diagram participant by name.
 
void generate_activity (eid_t activity_id, std::ostream &ostr, std::vector< eid_t > &visited) const
 Generate sequence diagram activity.
 
- Public Member Functions inherited from clanguml::common::generators::plantuml::generator< ConfigType, DiagramType >
 generator (ConfigType &config, DiagramType &model)
 Constructor.
 
 ~generator () override=default
 
void generate (std::ostream &ostr) const override
 Generate diagram.
 
virtual void generate_diagram (std::ostream &ostr) const =0
 Generate diagram specific part.
 
void generate_config_layout_hints (std::ostream &ostr) const
 Generate diagram layout hints.
 
void generate_plantuml_directives (std::ostream &ostr, const std::vector< std::string > &directives) const
 Generate PlantUML directives from config file.
 
void generate_notes (std::ostream &ostr, const model::element &element) const
 Generate diagram notes.
 
void generate_style (std::ostream &ostr, const std::string &element_type, const model::stylable_element &el) const
 Generate diagram element PlantUML style.
 
void generate_metadata (std::ostream &ostr) const
 Generate comment with diagram metadata.
 
void generate_title (std::ostream &ostr) const
 Generate diagram title.
 
template<typename E >
void generate_link (std::ostream &ostr, const E &e) const
 Generate hyper link to element.
 
void generate_link (std::ostream &ostr, const relationship &e) const
 generate_link specialization for relationship
 
void print_debug (const common::model::source_location &e, std::ostream &ostr) const
 Print debug information in diagram comments.
 
- 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.
 
std::string generate_alias (const model::participant &participant) const
 Generate PlantUML alias for participant.
 
void generate_message_comment (std::ostream &ostr, const model::message &m) const
 Generate message call note.
 
void generate_from_to_sequences (std::ostream &ostr) const
 
void generate_to_sequences (std::ostream &ostr) const
 
void generate_from_sequences (std::ostream &ostr) const
 
std::vector< eid_tfind_from_activities () const
 
std::vector< model::message_chain_tfind_to_message_chains () const
 
std::string render_message_name (const std::string &m) const
 
model::function::message_render_mode select_method_arguments_render_mode () const
 Convert config to model message render mode.
 

Private Attributes

std::set< eid_tgenerated_participants_
 
std::set< unsigned int > generated_comment_ids_
 
std::vector< model::messagealready_generated_in_static_context_
 
std::set< eid_tgenerated_activities_
 

Additional Inherited Members

- Protected Attributes inherited from clanguml::common::generators::plantuml::generator< ConfigType, DiagramType >
std::set< std::string > m_generated_aliases
 
- 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::plantuml::generator::generator ( diagram_config config,
diagram_model model 
)

Definition at line 37 of file sequence_diagram_generator.cc.

39 : common_generator<diagram_config, diagram_model>{config, model}
40{
41}

Member Function Documentation

◆ find_from_activities()

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

Definition at line 696 of file sequence_diagram_generator.cc.

697{
698 std::vector<eid_t> start_from;
699 for (const auto &sf : config().from()) {
700 if (sf.location_type == location_t::function) {
701 bool found{false};
702 for (const auto &[k, v] : model().sequences()) {
703 if (model().participants().count(v.from()) == 0)
704 continue;
705
706 const auto &caller = *model().participants().at(v.from());
707 std::string vfrom = caller.full_name(false);
708 if (sf.location == vfrom) {
709 LOG_DBG("Found sequence diagram start point: {}", k);
710 start_from.push_back(k);
711 found = true;
712 }
713 }
714
715 if (!found)
716 throw error::invalid_sequence_from_condition(model().type(),
717 model().name(),
718 fmt::format("Failed to find participant matching '{}' for "
719 "'from' condition: ",
720 sf.location.to_string()));
721 }
722 }
723
724 return start_from;
725}

◆ find_to_message_chains()

std::vector< model::message_chain_t > clanguml::sequence_diagram::generators::plantuml::generator::find_to_message_chains ( ) const
private

Definition at line 727 of file sequence_diagram_generator.cc.

728{
729 std::vector<model::message_chain_t> result;
730
731 for (const auto &to_location : config().to()) {
732 auto to_activity_ids = model().get_to_activity_ids(to_location);
733
734 if (to_activity_ids.empty()) {
735 LOG_WARN("Failed to find participant matching '{}' for "
736 "'to' condition: ",
737 to_location.location.to_string());
738 }
739
740 for (const auto &to_activity_id : to_activity_ids) {
741 std::vector<model::message_chain_t> message_chains_unique =
742 model().get_all_from_to_message_chains(eid_t{}, to_activity_id);
743
744 result.insert(result.end(), message_chains_unique.begin(),
745 message_chains_unique.end());
746 }
747 }
748
749 return result;
750}

◆ generate_activity()

void clanguml::sequence_diagram::generators::plantuml::generator::generate_activity ( eid_t  activity_id,
std::ostream &  ostr,
std::vector< eid_t > &  visited 
) const

Generate sequence diagram activity.

Parameters
activity_idActivity id
ostrOutput stream
visitedList of already visited participants, this is necessary for breaking infinite recursion on recursive calls

Definition at line 202 of file sequence_diagram_generator.cc.

204{
205 const auto &a = model().get_activity(activity_id);
206
207 const auto [it, inserted] = generated_activities_.emplace(activity_id);
208
209 if (config().fold_repeated_activities() && !inserted &&
210 !a.messages().empty()) {
211 const auto &p =
212 model().get_participant<model::participant>(activity_id);
213
214 if (p.has_value()) {
215 ostr << "hnote over " << generate_alias(p.value()) << " : *\n";
216 // This is necessary to keep the hnote over the activity life line
217 ostr << generate_alias(p.value()) << "-[hidden]->"
218 << generate_alias(p.value()) << '\n';
219 }
220
221 return;
222 }
223
224 for (const auto &m : a.messages()) {
225 if (m.in_static_declaration_context()) {
227 continue;
228
230 }
231
232 if (m.type() == message_t::kCall || m.type() == message_t::kCoAwait) {
233 const auto &to =
234 model().get_participant<model::participant>(m.to());
235
236 if (!to.has_value()) {
237 LOG_DBG("Skipping activity {} due to missing target paricipant "
238 "in the diagram",
239 m.from());
240 continue;
241 }
242
243 visited.push_back(m.from());
244
245 LOG_DBG("Generating message [{}] --> [{}]", m.from(), m.to());
246
247 generate_call(m, ostr);
248
249 std::string to_alias = generate_alias(to.value());
250
251 ostr << "activate " << to_alias << '\n';
252
253 if (model().sequences().find(m.to()) != model().sequences().end()) {
254 if (std::find(visited.begin(), visited.end(), m.to()) ==
255 visited
256 .end()) { // break infinite recursion on recursive calls
257
258 LOG_DBG("Generating activity {} (called from {})", m.to(),
259 m.from());
260
261 generate_activity(m.to(), ostr, visited);
262 }
263 }
264 else
265 LOG_DBG("Skipping activity {} --> {} - missing sequence {}",
266 m.from(), m.to(), m.to());
267
268 ostr << "deactivate " << to_alias << '\n';
269
270 visited.pop_back();
271 }
272 else if (m.type() == message_t::kReturn) {
273 print_debug(m, ostr);
275 auto return_message = m;
276 if (!visited.empty()) {
277 return_message.set_to(visited.back());
278 }
279 generate_return(return_message, ostr);
280 }
281 else if (m.type() == message_t::kCoReturn) {
282 print_debug(m, ostr);
284 auto return_message = m;
285 if (!visited.empty()) {
286 return_message.set_to(visited.back());
287 }
288 generate_return(return_message, ostr);
289 }
290 else if (m.type() == message_t::kCoYield) {
291 print_debug(m, ostr);
293 auto return_message = m;
294 if (!visited.empty()) {
295 return_message.set_to(visited.back());
296 }
297 generate_return(return_message, ostr);
298 }
299 else if (m.type() == message_t::kIf) {
300 print_debug(m, ostr);
302 ostr << "alt";
303 if (const auto &text = m.condition_text(); text.has_value())
304 ostr << " " << text.value();
305 ostr << '\n';
306 }
307 else if (m.type() == message_t::kElseIf) {
308 print_debug(m, ostr);
309 ostr << "else";
310 if (const auto &text = m.condition_text(); text.has_value())
311 ostr << " " << text.value();
312 ostr << '\n';
313 }
314 else if (m.type() == message_t::kElse) {
315 print_debug(m, ostr);
316 ostr << "else\n";
317 }
318 else if (m.type() == message_t::kIfEnd) {
319 ostr << "end\n";
320 }
321 else if (m.type() == message_t::kWhile) {
322 print_debug(m, ostr);
324 ostr << "loop";
325 if (const auto &text = m.condition_text(); text.has_value())
326 ostr << " " << text.value();
327 ostr << '\n';
328 }
329 else if (m.type() == message_t::kWhileEnd) {
330 ostr << "end\n";
331 }
332 else if (m.type() == message_t::kFor) {
333 print_debug(m, ostr);
335 ostr << "loop";
336 if (const auto &text = m.condition_text(); text.has_value())
337 ostr << " " << text.value();
338 ostr << '\n';
339 }
340 else if (m.type() == message_t::kForEnd) {
341 ostr << "end\n";
342 }
343 else if (m.type() == message_t::kDo) {
344 print_debug(m, ostr);
346 ostr << "loop";
347 if (const auto &text = m.condition_text(); text.has_value())
348 ostr << " " << text.value();
349 ostr << '\n';
350 }
351 else if (m.type() == message_t::kDoEnd) {
352 ostr << "end\n";
353 }
354 else if (m.type() == message_t::kTry) {
355 print_debug(m, ostr);
357 ostr << "group try\n";
358 }
359 else if (m.type() == message_t::kCatch) {
360 print_debug(m, ostr);
361 ostr << "else " << render_message_name(m.message_name()) << '\n';
362 }
363 else if (m.type() == message_t::kTryEnd) {
364 print_debug(m, ostr);
365 ostr << "end\n";
366 }
367 else if (m.type() == message_t::kSwitch) {
368 print_debug(m, ostr);
370 ostr << "group switch\n";
371 }
372 else if (m.type() == message_t::kCase) {
373 print_debug(m, ostr);
374 ostr << "else " << render_message_name(m.message_name()) << '\n';
375 }
376 else if (m.type() == message_t::kSwitchEnd) {
377 ostr << "end\n";
378 }
379 else if (m.type() == message_t::kConditional) {
380 print_debug(m, ostr);
382 ostr << "alt";
383 if (const auto &text = m.condition_text(); text.has_value())
384 ostr << " " << text.value();
385 ostr << '\n';
386 }
387 else if (m.type() == message_t::kConditionalElse) {
388 print_debug(m, ostr);
389 ostr << "else\n";
390 }
391 else if (m.type() == message_t::kConditionalEnd) {
392 ostr << "end\n";
393 }
394 }
395}

◆ generate_alias()

std::string clanguml::sequence_diagram::generators::plantuml::generator::generate_alias ( const model::participant participant) const
private

Generate PlantUML alias for participant.

Parameters
participantSequence diagram participant model
Returns
Particpant alias

Definition at line 618 of file sequence_diagram_generator.cc.

620{
621 if ((participant.type_name() == "function" ||
622 participant.type_name() == "function_template") &&
623 config().combine_free_functions_into_file_participants()) {
624 const auto file_id = common::to_id(participant.file());
625
626 return fmt::format("C_{:022}", file_id.value());
627 }
628
629 return participant.alias();
630}

◆ generate_call()

void clanguml::sequence_diagram::generators::plantuml::generator::generate_call ( const clanguml::sequence_diagram::model::message m,
std::ostream &  ostr 
) const

Generate sequence diagram message.

Parameters
mMessage model
ostrOutput stream

Definition at line 43 of file sequence_diagram_generator.cc.

44{
45 const auto &from = model().get_participant<model::participant>(m.from());
46 const auto &to = model().get_participant<model::participant>(m.to());
47
48 if (!from || !to) {
49 LOG_DBG("Skipping empty call from '{}' to '{}'", m.from(), m.to());
50 return;
51 }
52
53 generate_participant(ostr, m.from());
54 generate_participant(ostr, m.to());
55
56 std::string message;
57
60
61 if (to.value().type_name() == "method") {
62 const auto &f = dynamic_cast<const model::method &>(to.value());
63 const std::string_view style = f.is_static() ? "__" : "";
64
65 if (m.type() == message_t::kCoAwait)
66 message = fmt::format("{}<< co_await >>\\n{}{}", style,
67 f.message_name(render_mode), style);
68 else
69 message = fmt::format(
70 "{}{}{}", style, f.message_name(render_mode), style);
71 }
72 else if (to.value().type_name() == "objc_method") {
73 const auto &f = dynamic_cast<const model::objc_method &>(to.value());
74 const std::string_view style = f.is_static() ? "__" : "";
75 message =
76 fmt::format("{}{}{}", style, f.message_name(render_mode), style);
77 }
78 else if (config().combine_free_functions_into_file_participants()) {
79 if (to.value().type_name() == "function") {
80 const auto &f = dynamic_cast<const model::function &>(to.value());
81 message = f.message_name(render_mode);
82
83 if (f.is_cuda_kernel())
84 message = fmt::format("<< CUDA Kernel >>\\n{}", message);
85 else if (f.is_cuda_device())
86 message = fmt::format("<< CUDA Device >>\\n{}", message);
87 else if (f.is_coroutine())
88 message = fmt::format("<< Coroutine >>\\n{}", message);
89 }
90 else if (to.value().type_name() == "function_template") {
91 const auto &f = dynamic_cast<const model::function &>(to.value());
92 message = f.message_name(render_mode);
93
94 if (f.is_cuda_kernel())
95 message = fmt::format("<< CUDA Kernel >>\\n{}", message);
96 else if (f.is_cuda_device())
97 message = fmt::format("<< CUDA Device >>\\n{}", message);
98 else if (f.is_coroutine())
99 message = fmt::format("<< Coroutine >>\\n{}", message);
100 }
101 }
102
103 message = config().simplify_template_type(message);
104
105 const std::string from_alias = generate_alias(from.value());
106 const std::string to_alias = generate_alias(to.value());
107
108 print_debug(m, ostr);
109
111
112 ostr << from_alias << " "
113 << common::generators::plantuml::to_plantuml(message_t::kCall) << " ";
114
115 ostr << to_alias;
116
117 if (config().generate_links) {
119 }
120
121 ostr << " : ";
122
124 ostr << "**[**";
125
126 ostr << render_message_name(message);
127
129 ostr << "**]**";
130
131 ostr << '\n';
132
133 LOG_DBG("Generated call '{}' from {} [{}] to {} [{}]", message,
134 from.value().full_name(false), m.from(), to.value().full_name(false),
135 m.to());
136}

◆ generate_diagram()

void clanguml::sequence_diagram::generators::plantuml::generator::generate_diagram ( std::ostream &  ostr) const
overridevirtual

Main generator method.

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

Parameters
ostrOutput stream.

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

Definition at line 632 of file sequence_diagram_generator.cc.

633{
634 model().print();
635
636 if (config().participants_order.has_value) {
637 for (const auto &p : config().participants_order()) {
638 LOG_DBG("Pregenerating participant {}", p);
639 generate_participant(ostr, p);
640 }
641 }
642
644
646
648}

◆ generate_from_sequences()

void clanguml::sequence_diagram::generators::plantuml::generator::generate_from_sequences ( std::ostream &  ostr) const
private

Definition at line 650 of file sequence_diagram_generator.cc.

651{
652 std::vector<eid_t> start_from = find_from_activities();
653
654 // Use this to break out of recurrent loops
655 std::vector<eid_t> visited_participants;
656
657 for (const auto from_id : start_from) {
658 if (model().participants().count(from_id) == 0)
659 continue;
660
661 const auto &from = model().get_participant<model::function>(from_id);
662
663 if (!from.has_value()) {
664 LOG_WARN("Failed to find participant {} for 'from' "
665 "condition");
666 continue;
667 }
668
669 generate_participant(ostr, from_id);
670
671 std::string from_alias = generate_alias(from.value());
672
675
676 // For methods or functions in diagrams where they are
677 // combined into file participants, we need to add an
678 // 'entry' point call to know which method relates to the
679 // first activity for this 'start_from' condition
680 if (from.value().type_name() == "method" ||
681 from.value().type_name() == "objc_method" ||
682 config().combine_free_functions_into_file_participants()) {
683 ostr << "[->" << " " << from_alias << " : "
684 << render_message_name(from.value().message_name(render_mode))
685 << '\n';
686 }
687
688 ostr << "activate " << from_alias << '\n';
689
690 generate_activity(from_id, ostr, visited_participants);
691
692 ostr << "deactivate " << from_alias << '\n';
693 }
694}

◆ generate_from_to_sequences()

void clanguml::sequence_diagram::generators::plantuml::generator::generate_from_to_sequences ( std::ostream &  ostr) const
private

Definition at line 793 of file sequence_diagram_generator.cc.

794{
795 for (const auto &ft : config().from_to()) {
796 // First, find the sequence of activities from 'from' location
797 // to 'to' location
798 assert(ft.size() == 2);
799
800 const auto &from_location = ft.front();
801 const auto &to_location = ft.back();
802
803 const auto from_activity_ids =
804 model().get_from_activity_ids(from_location);
805
806 const auto to_activity_ids = model().get_to_activity_ids(to_location);
807
808 if (from_activity_ids.empty()) {
809 throw error::invalid_sequence_from_condition(model().type(),
810 model().name(),
811 fmt::format("Failed to find participant matching '{}' for "
812 "'from' condition: ",
813 from_location.location.to_string()));
814 }
815
816 if (from_activity_ids.empty() || to_activity_ids.empty()) {
817 throw error::invalid_sequence_to_condition(model().type(),
818 model().name(),
819 fmt::format("Failed to find participant matching '{}' for "
820 "'to' condition: ",
821 to_location.location.to_string()));
822 }
823
824 bool first_separator_skipped{false};
825
826 for (const auto from_activity_id : from_activity_ids) {
827 if (model().participants().count(from_activity_id) == 0)
828 continue;
829
830 for (const auto to_activity_id : to_activity_ids) {
831 if (model().participants().count(to_activity_id) == 0)
832 continue;
833
834 auto message_chains_unique =
835 model().get_all_from_to_message_chains(
836 from_activity_id, to_activity_id);
837
838 for (const auto &mc : message_chains_unique) {
839 if (!first_separator_skipped)
840 first_separator_skipped = true;
841 else
842 ostr << "====\n";
843
844 const auto &from = model().get_participant<model::function>(
845 from_activity_id);
846
847 if (from.value().type_name() == "method" ||
848 from.value().type_name() == "objc_method" ||
849 config()
850 .combine_free_functions_into_file_participants()) {
851 generate_participant(ostr, from_activity_id);
852 ostr << "[->" << " " << generate_alias(from.value())
853 << " : "
854 << render_message_name(from.value().message_name(
856 << '\n';
857 }
858
859 for (const auto &m : mc) {
860 generate_call(m, ostr);
861 }
862 }
863 }
864 }
865 }
866}

◆ generate_message_comment()

void clanguml::sequence_diagram::generators::plantuml::generator::generate_message_comment ( std::ostream &  ostr,
const model::message m 
) const
private

Generate message call note.

Parameters
ostrOutput stream
mMessage

Definition at line 397 of file sequence_diagram_generator.cc.

399{
400 const auto &from = model().get_participant<model::participant>(m.from());
401 if (!from)
402 return;
403
404 // First generate message comments from \note directives in comments
405 bool comment_generated_from_note_decorators{false};
406 for (const auto &decorator : m.decorators()) {
407 auto note = std::dynamic_pointer_cast<decorators::note>(decorator);
408 if (note && note->applies_to_diagram(config().name)) {
409 comment_generated_from_note_decorators = true;
410
411 ostr << "note over " << generate_alias(from.value()) << '\n';
412
414 note->text, config().message_comment_width())
415 << '\n';
416
417 ostr << "end note" << '\n';
418 }
419 }
420
421 if (comment_generated_from_note_decorators)
422 return;
423
424 if (!config().generate_message_comments())
425 return;
426
427 // Now generate message notes from raw comments if enabled
428 if (const auto &comment = m.comment(); comment &&
429 generated_comment_ids_.emplace(comment.value().at("id")).second) {
430
431 ostr << "note over " << generate_alias(from.value()) << '\n';
432
433 ostr << util::format_message_comment(comment.value().at("comment"),
434 config().message_comment_width())
435 << '\n';
436
437 ostr << "end note" << '\n';
438 }
439}

◆ generate_participant() [1/2]

void clanguml::sequence_diagram::generators::plantuml::generator::generate_participant ( std::ostream &  ostr,
const std::string &  name 
) const

Generate sequence diagram participant by name.

This is convienience wrapper over generate_participant() by id.

Parameters
ostrOutput stream
nameFull participant name

Definition at line 441 of file sequence_diagram_generator.cc.

443{
444 auto p = model().get(name);
445
446 if (!p.has_value()) {
447 LOG_WARN("Cannot find participant {} from `participants_order` "
448 "option",
449 name);
450 return;
451 }
452
453 generate_participant(ostr, p.value().id(), true);
454}

◆ generate_participant() [2/2]

void clanguml::sequence_diagram::generators::plantuml::generator::generate_participant ( std::ostream &  ostr,
eid_t  id,
bool  force = false 
) const

Generate sequence diagram participant.

Parameters
ostrOutput stream
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 456 of file sequence_diagram_generator.cc.

458{
459 eid_t participant_id{};
460
461 if (!force) {
462 for (const auto pid : model().active_participants()) {
463 if (pid == id) {
464 participant_id = pid;
465 break;
466 }
467 }
468 }
469 else
470 participant_id = id;
471
472 if (participant_id == 0)
473 return;
474
475 if (is_participant_generated(participant_id))
476 return;
477
478 const auto &participant =
479 model().get_participant<model::participant>(participant_id).value();
480
481 if (participant.type_name() == "method") {
482 const auto class_id =
483 model()
484 .get_participant<model::method>(participant_id)
485 .value()
486 .class_id();
487
488 if (is_participant_generated(class_id))
489 return;
490
491 const auto &class_participant =
492 model().get_participant<model::participant>(class_id).value();
493
494 print_debug(class_participant, ostr);
495
496 auto participant_name = config().simplify_template_type(
497 display_name_adapter(class_participant).full_name(false));
498 participant_name =
499 config().using_namespace().relative(participant_name);
500
502
503 ostr << "participant \"" << participant_name << "\" as "
504 << class_participant.alias();
505
506 if (config().generate_links) {
508 ostr, class_participant);
509 }
510
511 ostr << '\n';
512
513 generated_participants_.emplace(class_id);
514 }
515 else if (participant.type_name() == "objc_method") {
516 const auto class_id =
517 model()
518 .get_participant<model::objc_method>(participant_id)
519 .value()
520 .class_id();
521
522 if (is_participant_generated(class_id))
523 return;
524
525 const auto &class_participant =
526 model().get_participant<model::participant>(class_id).value();
527
528 print_debug(class_participant, ostr);
529
530 auto participant_name = config().simplify_template_type(
531 display_name_adapter(class_participant).full_name(false));
532 participant_name =
533 config().using_namespace().relative(participant_name);
534
536
537 ostr << "participant \"" << participant_name << "\" as "
538 << class_participant.alias() << " <<ObjC Interface>>";
539
540 if (config().generate_links) {
542 ostr, class_participant);
543 }
544
545 ostr << '\n';
546
547 generated_participants_.emplace(class_id);
548 }
549 else if ((participant.type_name() == "function" ||
550 participant.type_name() == "function_template") &&
551 config().combine_free_functions_into_file_participants()) {
552 // Create a single participant for all functions declared in a
553 // single file
554 const auto &file_path =
555 model()
556 .get_participant<model::function>(participant_id)
557 .value()
558 .file();
559
560 assert(!file_path.empty());
561
562 const auto file_id = common::to_id(file_path);
563
564 if (is_participant_generated(file_id))
565 return;
566
567 auto participant_name = util::path_to_url(relative(
568 std::filesystem::path{file_path}, config().root_directory())
569 .string());
570
571 ostr << "participant \"" << participant_name << "\" as "
572 << fmt::format("C_{:022}", file_id.value());
573
574 ostr << '\n';
575
576 generated_participants_.emplace(file_id);
577 }
578 else {
579 print_debug(participant, ostr);
580
581 auto participant_name =
582 config().using_namespace().relative(config().simplify_template_type(
583 display_name_adapter(participant).full_name(false)));
585
586 ostr << "participant \"" << participant_name << "\" as "
587 << participant.alias();
588
589 if (const auto *function_ptr =
590 dynamic_cast<const model::function *>(&participant);
591 function_ptr) {
592 if (function_ptr->is_cuda_kernel())
593 ostr << " << CUDA Kernel >>";
594 else if (function_ptr->is_cuda_device())
595 ostr << " << CUDA Device >>";
596 else if (function_ptr->is_coroutine())
597 ostr << fmt::format("<< Coroutine >>");
598 }
599
600 if (config().generate_links) {
602 ostr, participant);
603 }
604
605 ostr << '\n';
606
607 generated_participants_.emplace(participant_id);
608 }
609}

◆ generate_return()

void clanguml::sequence_diagram::generators::plantuml::generator::generate_return ( const clanguml::sequence_diagram::model::message m,
std::ostream &  ostr 
) const

Generate sequence diagram return message.

Parameters
mMessage model
ostrOutput stream

Definition at line 138 of file sequence_diagram_generator.cc.

139{
140 // Add return activity only for messages between different actors
141 // and only if the return type is different than void
142 if (m.from() == m.to())
143 return;
144
145 std::string message_stereotype;
146 if (m.type() == message_t::kCoReturn) {
147 message_stereotype = "<< co_return >>";
148 }
149 else if (m.type() == message_t::kCoYield) {
150 message_stereotype = "<< co_yield >>";
151 }
152
153 std::string message_label;
154
155 const auto &from = model().get_participant<model::function>(m.from());
156 const auto &to = model().get_participant<model::participant>(m.to());
157 if (to.has_value() && from.has_value() && !from.value().is_void()) {
158 const std::string from_alias = generate_alias(from.value());
159
160 const std::string to_alias = generate_alias(to.value());
161
162 ostr << from_alias << " "
163 << common::generators::plantuml::to_plantuml(message_t::kReturn)
164 << " " << to_alias;
165
166 if (config().generate_return_types())
167 message_label = render_message_name(m.return_type());
168 else if (config().generate_return_values())
169 message_label = render_message_name(m.message_name());
170 }
171 else if (from.has_value() && !from.value().is_void() &&
172 (from.value().type_name() == "method" ||
173 from.value().type_name() == "objc_method" ||
174 config().combine_free_functions_into_file_participants())) {
175 const std::string from_alias = generate_alias(from.value());
176
177 ostr << "[<--" << " " << from_alias;
178 if (config().generate_return_types())
179 message_label = render_message_name(from.value().return_type());
180 else if (config().generate_return_values())
181 message_label = render_message_name(m.message_name());
182 }
183
184 if (!message_stereotype.empty()) {
185 if (message_label.empty())
186 message_label = fmt::format("//{}//", message_stereotype);
187 else
188 message_label = fmt::format(
189 "//{}//\\n//{}//", message_stereotype, message_label);
190 }
191 else {
192 if (!message_label.empty())
193 message_label = fmt::format("//{}//", message_label);
194 }
195
196 if (!message_label.empty())
197 ostr << " : " << message_label;
198
199 ostr << '\n';
200}

◆ generate_to_sequences()

void clanguml::sequence_diagram::generators::plantuml::generator::generate_to_sequences ( std::ostream &  ostr) const
private

Definition at line 757 of file sequence_diagram_generator.cc.

758{
759 std::vector<model::message_chain_t> message_chains =
761
762 bool first_separator_skipped{false};
763 for (const auto &mc : message_chains) {
764 if (!first_separator_skipped)
765 first_separator_skipped = true;
766 else
767 ostr << "====\n";
768
769 const auto from_activity_id = mc.front().from();
770
771 if (model().participants().count(from_activity_id) == 0)
772 continue;
773
774 const auto &from =
775 model().get_participant<model::function>(from_activity_id);
776
777 if (from.value().type_name() == "method" ||
778 from.value().type_name() == "objc_method" ||
779 config().combine_free_functions_into_file_participants()) {
780 generate_participant(ostr, from_activity_id);
781 ostr << "[->" << " " << generate_alias(from.value()) << " : "
782 << render_message_name(from.value().message_name(
784 << '\n';
785 }
786
787 for (const auto &m : mc) {
788 generate_call(m, ostr);
789 }
790 }
791}

◆ is_participant_generated()

bool clanguml::sequence_diagram::generators::plantuml::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 611 of file sequence_diagram_generator.cc.

612{
613 return std::find(generated_participants_.begin(),
615 id) != generated_participants_.end();
616}

◆ render_message_name()

std::string clanguml::sequence_diagram::generators::plantuml::generator::render_message_name ( const std::string &  m) const
private

Definition at line 752 of file sequence_diagram_generator.cc.

753{
754 return util::abbreviate(m, config().message_name_width());
755}

◆ select_method_arguments_render_mode()

model::function::message_render_mode clanguml::sequence_diagram::generators::plantuml::generator::select_method_arguments_render_mode ( ) const
private

Convert config to model message render mode.

Returns
Method render mode.

Definition at line 869 of file sequence_diagram_generator.cc.

870{
871 if (config().generate_method_arguments() ==
874
875 if (config().generate_method_arguments() == config::method_arguments::none)
877
879}

Member Data Documentation

◆ already_generated_in_static_context_

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

Definition at line 166 of file sequence_diagram_generator.h.

◆ generated_activities_

std::set<eid_t> clanguml::sequence_diagram::generators::plantuml::generator::generated_activities_
mutableprivate

Definition at line 167 of file sequence_diagram_generator.h.

◆ generated_comment_ids_

std::set<unsigned int> clanguml::sequence_diagram::generators::plantuml::generator::generated_comment_ids_
mutableprivate

Definition at line 165 of file sequence_diagram_generator.h.

◆ generated_participants_

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

Definition at line 164 of file sequence_diagram_generator.h.


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