renderer.py 3.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  1. from uuid import UUID
  2. from services import scd, od
  3. from services.bottom.V0 import Bottom
  4. from concrete_syntax.common import display_value, display_name, indent
  5. def render_object_diagram(state, m, mm, render_attributes=True, prefix_ids=""):
  6. bottom = Bottom(state)
  7. mm_scd = scd.SCD(mm, state)
  8. m_od = od.OD(mm, m, state)
  9. def make_id(uuid) -> str:
  10. return 'n'+(prefix_ids+str(uuid).replace('-',''))[24:]
  11. output = ""
  12. # Render objects
  13. for class_name, class_node in mm_scd.get_classes().items():
  14. if render_attributes:
  15. attributes = od.get_attributes(bottom, class_node)
  16. for obj_name, obj_node in m_od.get_objects(class_node).items():
  17. output += f"\n{make_id(obj_node)} [label=\"{display_name(obj_name)} : {class_name}\", shape=rect] ;"
  18. #" {{"
  19. # if render_attributes:
  20. # for attr_name, attr_edge in attributes:
  21. # slot = m_od.get_slot(obj_node, attr_name)
  22. # if slot != None:
  23. # val, type_name = od.read_primitive_value(bottom, slot, mm)
  24. # output += f"\n{attr_name} => {display_value(val, type_name)}"
  25. # output += '\n}'
  26. output += '\n'
  27. # Render links
  28. for assoc_name, assoc_edge in mm_scd.get_associations().items():
  29. for link_name, link_edge in m_od.get_objects(assoc_edge).items():
  30. src_obj = bottom.read_edge_source(link_edge)
  31. tgt_obj = bottom.read_edge_target(link_edge)
  32. src_name = m_od.get_object_name(src_obj)
  33. tgt_name = m_od.get_object_name(tgt_obj)
  34. output += f"\n{make_id(src_obj)} -> {make_id(tgt_obj)} [label=\"{display_name(link_name)}:{assoc_name}\"] ;"
  35. return output
  36. def render_trace_match(state, name_mapping: dict, pattern_m: UUID, host_m: UUID, color="grey", prefix_pattern_ids="", prefix_host_ids=""):
  37. bottom = Bottom(state)
  38. class_type = od.get_scd_mm_class_node(bottom)
  39. attr_link_type = od.get_scd_mm_attributelink_node(bottom)
  40. def make_pattern_id(uuid) -> str:
  41. return 'n'+(prefix_pattern_ids+str(uuid).replace('-',''))[24:]
  42. def make_host_id(uuid) -> str:
  43. return 'n'+(prefix_host_ids+str(uuid).replace('-',''))[24:]
  44. output = ""
  45. # render_suffix = f"#line:{color};line.dotted;text:{color} : matchedWith"
  46. render_suffix = f"[label=\"\",style=dashed,color={color}] ;"
  47. for pattern_el_name, host_el_name in name_mapping.items():
  48. # print(pattern_el_name, host_el_name)
  49. try:
  50. pattern_el, = bottom.read_outgoing_elements(pattern_m, pattern_el_name)
  51. host_el, = bottom.read_outgoing_elements(host_m, host_el_name)
  52. except:
  53. continue
  54. # only render 'match'-edges between objects (= those elements where the type of the type is 'Class'):
  55. pattern_el_type = od.get_type(bottom, pattern_el)
  56. pattern_el_type_type = od.get_type(bottom, pattern_el_type)
  57. if pattern_el_type_type == class_type:
  58. output += f"\n{make_pattern_id(pattern_el)} -> {make_host_id(host_el)} {render_suffix}"
  59. # elif pattern_el_type_type == attr_link_type:
  60. # pattern_obj = bottom.read_edge_source(pattern_el)
  61. # pattern_attr_name = od.get_attr_name(bottom, pattern_el_type)
  62. # host_obj = bottom.read_edge_source(host_el)
  63. # host_el_type = od.get_type(bottom, host_el)
  64. # host_attr_name = od.get_attr_name(bottom, host_el_type)
  65. # output += f"\n{make_pattern_id(pattern_obj)}::{pattern_attr_name} -> {make_host_id(host_obj)}::{host_attr_name} {render_suffix}"
  66. return output
  67. def render_package(name, contents):
  68. output = f"subgraph cluster_{name} {{\n label=\"{name}\";"
  69. output += indent(contents, 2)
  70. output += "\n}\n"
  71. return output