renderer.py 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. # PlantUML renderer
  2. from services import scd, od
  3. from services.bottom.V0 import Bottom
  4. from transformation import ramify
  5. from concrete_syntax.common import display_value
  6. from uuid import UUID
  7. def render_class_diagram(state, model, prefix_ids=""):
  8. bottom = Bottom(state)
  9. model_scd = scd.SCD(model, state)
  10. model_od = od.OD(od.get_scd_mm(bottom), model, state)
  11. def make_id(uuid) -> str:
  12. return prefix_ids+str(uuid).replace('-','_')
  13. output = ""
  14. # Render classes
  15. for name, class_node in model_scd.get_classes().items():
  16. is_abstract = False
  17. slot = model_od.get_slot(class_node, "abstract")
  18. if slot != None:
  19. is_abstract, _ = od.read_primitive_value(bottom, slot, model_od.type_model)
  20. lower_card, upper_card = model_scd.get_class_cardinalities(class_node)
  21. if lower_card == None and upper_card == None:
  22. card_spec = ""
  23. else:
  24. card_spec = f"{0 if lower_card == None else lower_card}..{"*" if upper_card == None else upper_card}"
  25. if is_abstract:
  26. output += f"\nabstract class \"{name} {card_spec}\" as {make_id(class_node)}"
  27. else:
  28. output += f"\nclass \"{name} {card_spec}\" as {make_id(class_node)}"
  29. # Render attributes
  30. output += " {"
  31. for attr_name, attr_edge in od.get_attributes(bottom, class_node):
  32. tgt_name = model_scd.get_class_name(bottom.read_edge_target(attr_edge))
  33. output += f"\n {attr_name} : {tgt_name}"
  34. output += "\n}"
  35. output += "\n"
  36. # Render inheritance links
  37. for inh_node in model_scd.get_inheritances().values():
  38. src_node = bottom.read_edge_source(inh_node)
  39. tgt_node = bottom.read_edge_target(inh_node)
  40. output += f"\n{make_id(tgt_node)} <|-- {make_id(src_node)}"
  41. output += "\n"
  42. # Render associations
  43. for assoc_name, assoc_edge in model_scd.get_associations().items():
  44. src_node = bottom.read_edge_source(assoc_edge)
  45. tgt_node = bottom.read_edge_target(assoc_edge)
  46. src_lower_card, src_upper_card, tgt_lower_card, tgt_upper_card = model_scd.get_assoc_cardinalities(assoc_edge)
  47. # default cardinalities
  48. if src_lower_card == None:
  49. src_lower_card = 0
  50. if src_upper_card == None:
  51. src_upper_card = "*"
  52. if tgt_lower_card == None:
  53. tgt_lower_card = 0
  54. if tgt_upper_card == None:
  55. tgt_upper_card = "*"
  56. src_card = f"{src_lower_card} .. {src_upper_card}"
  57. tgt_card = f"{tgt_lower_card} .. {tgt_upper_card}"
  58. if src_card == "0 .. *":
  59. src_card = " " # hide cardinality
  60. if tgt_card == "1 .. 1":
  61. tgt_card = " " # hide cardinality
  62. output += f'\n{make_id(src_node)} "{src_card}" --> "{tgt_card}" {make_id(tgt_node)} : {assoc_name}'
  63. return output
  64. def render_object_diagram(state, m, mm, render_attributes=True, prefix_ids=""):
  65. bottom = Bottom(state)
  66. mm_scd = scd.SCD(mm, state)
  67. m_od = od.OD(mm, m, state)
  68. def make_id(uuid) -> str:
  69. return prefix_ids+str(uuid).replace('-','_')
  70. output = ""
  71. # Render objects
  72. for class_name, class_node in mm_scd.get_classes().items():
  73. if render_attributes:
  74. attributes = od.get_attributes(bottom, class_node)
  75. for obj_name, obj_node in m_od.get_objects(class_node).items():
  76. output += f"\nmap \"{obj_name} : {class_name}\" as {make_id(obj_node)} {{"
  77. if render_attributes:
  78. for attr_name, attr_edge in attributes:
  79. slot = m_od.get_slot(obj_node, attr_name)
  80. if slot != None:
  81. val, type_name = od.read_primitive_value(bottom, slot, mm)
  82. output += f"\n{attr_name} => {display_value(val, type_name)}"
  83. output += '\n}'
  84. output += '\n'
  85. # Render links
  86. for assoc_name, assoc_edge in mm_scd.get_associations().items():
  87. for link_name, link_edge in m_od.get_objects(assoc_edge).items():
  88. src_obj = bottom.read_edge_source(link_edge)
  89. tgt_obj = bottom.read_edge_target(link_edge)
  90. src_name = m_od.get_object_name(src_obj)
  91. tgt_name = m_od.get_object_name(tgt_obj)
  92. output += f"\n{make_id(src_obj)} -> {make_id(tgt_obj)} : :{assoc_name}"
  93. return output
  94. def render_package(name, contents):
  95. output = ""
  96. output += f'\npackage "{name}" {{'
  97. output += contents
  98. output += '\n}'
  99. return output
  100. def render_trace_ramifies(state, mm, ramified_mm, render_attributes=True, prefix_ram_ids="", prefix_orig_ids=""):
  101. bottom = Bottom(state)
  102. mm_scd = scd.SCD(mm, state)
  103. ramified_mm_scd = scd.SCD(ramified_mm, state)
  104. def make_ram_id(uuid) -> str:
  105. return prefix_ram_ids+str(uuid).replace('-','_')
  106. def make_orig_id(uuid) -> str:
  107. return prefix_orig_ids+str(uuid).replace('-','_')
  108. output = ""
  109. # Render RAMifies-edges between classes
  110. for ram_name, ram_class_node in ramified_mm_scd.get_classes().items():
  111. original_class, = bottom.read_outgoing_elements(ram_class_node, ramify.RAMIFIES_LABEL)
  112. original_name = mm_scd.get_class_name(original_class)
  113. output += f"\n{make_ram_id(ram_class_node)} ..> {make_orig_id(original_class)} #line:green;text:green : RAMifies"
  114. if render_attributes:
  115. # and between attributes
  116. for (ram_attr_name, ram_attr_edge) in od.get_attributes(bottom, ram_class_node):
  117. orig_attr_edge, = bottom.read_outgoing_elements(ram_attr_edge, ramify.RAMIFIES_LABEL)
  118. orig_class_node = bottom.read_edge_source(orig_attr_edge)
  119. # dirty AF:
  120. orig_attr_name = mm_scd.get_class_name(orig_attr_edge)[len(original_name)+1:]
  121. output += f"\n{make_ram_id(ram_class_node)}::{ram_attr_name} ..> {make_orig_id(orig_class_node)}::{orig_attr_name} #line:green;text:green : RAMifies"
  122. return output
  123. def render_trace_conformance(state, m, mm, render_attributes=True, prefix_inst_ids="", prefix_type_ids=""):
  124. bottom = Bottom(state)
  125. mm_scd = scd.SCD(mm, state)
  126. m_od = od.OD(mm, m, state)
  127. def make_inst_id(uuid) -> str:
  128. return prefix_inst_ids+str(uuid).replace('-','_')
  129. def make_type_id(uuid) -> str:
  130. return prefix_type_ids+str(uuid).replace('-','_')
  131. output = ""
  132. # Render objects
  133. for class_name, class_node in mm_scd.get_classes().items():
  134. if render_attributes:
  135. attributes = od.get_attributes(bottom, class_node)
  136. for obj_name, obj_node in m_od.get_objects(class_node).items():
  137. output += f"\n{make_inst_id(obj_node)} ..> {make_type_id(class_node)} #line:blue;text:blue : instanceOf"
  138. if render_attributes:
  139. for attr_name, attr_edge in attributes:
  140. slot = m_od.get_slot(obj_node, attr_name)
  141. if slot != None:
  142. output += f"\n{make_inst_id(obj_node)}::{attr_name} ..> {make_type_id(class_node)}::{attr_name} #line:blue;text:blue : instanceOf"
  143. output += '\n'
  144. return output
  145. def render_trace_match(state, name_mapping: dict, pattern_m: UUID, host_m: UUID, color="grey", prefix_pattern_ids="", prefix_host_ids=""):
  146. bottom = Bottom(state)
  147. class_type = od.get_scd_mm_class_node(bottom)
  148. attr_link_type = od.get_scd_mm_attributelink_node(bottom)
  149. def make_pattern_id(uuid) -> str:
  150. return prefix_pattern_ids+str(uuid).replace('-','_')
  151. def make_host_id(uuid) -> str:
  152. return prefix_host_ids+str(uuid).replace('-','_')
  153. output = ""
  154. render_suffix = f"#line:{color};line.dotted;text:{color} : matchedWith"
  155. for pattern_el_name, host_el_name in name_mapping.items():
  156. # print(pattern_el_name, host_el_name)
  157. try:
  158. pattern_el, = bottom.read_outgoing_elements(pattern_m, pattern_el_name)
  159. host_el, = bottom.read_outgoing_elements(host_m, host_el_name)
  160. except:
  161. continue
  162. # only render 'match'-edges between objects (= those elements where the type of the type is 'Class'):
  163. pattern_el_type = od.get_type(bottom, pattern_el)
  164. pattern_el_type_type = od.get_type(bottom, pattern_el_type)
  165. if pattern_el_type_type == class_type:
  166. output += f"\n{make_pattern_id(pattern_el)} ..> {make_host_id(host_el)} {render_suffix}"
  167. elif pattern_el_type_type == attr_link_type:
  168. pattern_obj = bottom.read_edge_source(pattern_el)
  169. pattern_attr_name = od.get_attr_name(bottom, pattern_el_type)
  170. host_obj = bottom.read_edge_source(host_el)
  171. host_el_type = od.get_type(bottom, host_el)
  172. host_attr_name = od.get_attr_name(bottom, host_el_type)
  173. output += f"\n{make_pattern_id(pattern_obj)}::{pattern_attr_name} ..> {make_host_id(host_obj)}::{host_attr_name} {render_suffix}"
  174. return output