renderer.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. import functools
  2. from uuid import UUID
  3. from api.od import ODAPI
  4. from services import scd, od
  5. from services.bottom.V0 import Bottom
  6. from concrete_syntax.common import display_value, display_name, indent
  7. # Turn ModelVerse/muMLE ID into GraphViz ID
  8. def make_graphviz_id(uuid, prefix="") -> str:
  9. result = 'n'+(prefix+str(uuid).replace('-',''))[24:] # we assume that the first 24 characters are always zero...
  10. return result
  11. def render_object_diagram(state, m, mm,
  12. render_attributes=True, # doesn't do anything (yet)
  13. prefix_ids="",
  14. reify=False, # If true, will create a node in the middle of every link. This allows links to be the src/tgt of other links (which muMLE supports), but will result in a larger diagram.
  15. only_render=None, # List of type names or None. If specified, only render instances of these types. E.g., ["Place", "connection"]
  16. type_to_style={}, # Dictionary. Mapping from type-name to graphviz style. E.g., { "generic_link": ",color=purple" }
  17. type_to_label={}, # Dictionary. Mapping from type-name to callback for custom label creation.
  18. ):
  19. bottom = Bottom(state)
  20. mm_scd = scd.SCD(mm, state)
  21. m_od = od.OD(mm, m, state)
  22. odapi = ODAPI(state, m, mm)
  23. make_id = functools.partial(make_graphviz_id, prefix=prefix_ids)
  24. output = ""
  25. # Render objects
  26. for class_name, class_node in mm_scd.get_classes().items():
  27. if only_render != None and class_name not in only_render:
  28. continue
  29. make_label = type_to_label.get(class_name,
  30. # default, if not found:
  31. lambda obj_name, obj, odapi: f"{display_name(obj_name)} : {class_name}")
  32. output += f"\nsubgraph {class_name} {{"
  33. if render_attributes:
  34. attributes = od.get_attributes(bottom, class_node)
  35. custom_style = type_to_style.get(class_name, "")
  36. if custom_style == "":
  37. output += f"\nnode [shape=rect]"
  38. else:
  39. output += f"\nnode [shape=rect,{custom_style}]"
  40. for obj_name, obj_node in m_od.get_objects(class_node).items():
  41. output += f"\n{make_id(obj_node)} [label=\"{make_label(obj_name, obj_node, odapi)}\"] ;"
  42. #" {{"
  43. # if render_attributes:
  44. # for attr_name, attr_edge in attributes:
  45. # slot = m_od.get_slot(obj_node, attr_name)
  46. # if slot != None:
  47. # val, type_name = od.read_primitive_value(bottom, slot, mm)
  48. # output += f"\n{attr_name} => {display_value(val, type_name)}"
  49. # output += '\n}'
  50. output += '\n}'
  51. output += '\n'
  52. # Render links
  53. for assoc_name, assoc_edge in mm_scd.get_associations().items():
  54. if only_render != None and assoc_name not in only_render:
  55. continue
  56. make_label = type_to_label.get(assoc_name,
  57. # default, if not found:
  58. lambda lnk_name, lnk, odapi: f"{display_name(lnk_name)} : {assoc_name}")
  59. output += f"\nsubgraph {assoc_name} {{"
  60. custom_style = type_to_style.get(assoc_name, "")
  61. if custom_style != "":
  62. output += f"\nedge [{custom_style}]"
  63. if reify:
  64. if custom_style != "":
  65. # created nodes will be points of matching style:
  66. output += f"\nnode [{custom_style},shape=point]"
  67. else:
  68. output += "\nnode [shape=point]"
  69. for link_name, link_edge in m_od.get_objects(assoc_edge).items():
  70. src_obj = bottom.read_edge_source(link_edge)
  71. tgt_obj = bottom.read_edge_target(link_edge)
  72. src_name = m_od.get_object_name(src_obj)
  73. tgt_name = m_od.get_object_name(tgt_obj)
  74. if reify:
  75. # intermediary node:
  76. output += f"\n{make_id(src_obj)} -> {make_id(link_edge)} [arrowhead=none]"
  77. output += f"\n{make_id(link_edge)} -> {make_id(tgt_obj)}"
  78. output += f"\n{make_id(link_edge)} [xlabel=\"{make_label(link_name, link_edge, odapi)}\"]"
  79. else:
  80. output += f"\n{make_id(src_obj)} -> {make_id(tgt_obj)} [label=\"{make_label(link_name, link_edge, odapi)}\", {custom_style}] ;"
  81. output += '\n}'
  82. return output
  83. def render_trace_match(state, name_mapping: dict, pattern_m: UUID, host_m: UUID, color="grey", prefix_pattern_ids="", prefix_host_ids=""):
  84. bottom = Bottom(state)
  85. class_type = od.get_scd_mm_class_node(bottom)
  86. attr_link_type = od.get_scd_mm_attributelink_node(bottom)
  87. make_pattern_id = functools.partial(make_graphviz_id, prefix=prefix_pattern_ids)
  88. make_host_id = functools.partial(make_graphviz_id, prefix=prefix_host_ids)
  89. output = ""
  90. # render_suffix = f"#line:{color};line.dotted;text:{color} : matchedWith"
  91. render_suffix = f"[label=\"\",style=dashed,color={color}] ;"
  92. for pattern_el_name, host_el_name in name_mapping.items():
  93. # print(pattern_el_name, host_el_name)
  94. try:
  95. pattern_el, = bottom.read_outgoing_elements(pattern_m, pattern_el_name)
  96. host_el, = bottom.read_outgoing_elements(host_m, host_el_name)
  97. except:
  98. continue
  99. # only render 'match'-edges between objects (= those elements where the type of the type is 'Class'):
  100. pattern_el_type = od.get_type(bottom, pattern_el)
  101. pattern_el_type_type = od.get_type(bottom, pattern_el_type)
  102. if pattern_el_type_type == class_type:
  103. output += f"\n{make_pattern_id(pattern_el)} -> {make_host_id(host_el)} {render_suffix}"
  104. # elif pattern_el_type_type == attr_link_type:
  105. # pattern_obj = bottom.read_edge_source(pattern_el)
  106. # pattern_attr_name = od.get_attr_name(bottom, pattern_el_type)
  107. # host_obj = bottom.read_edge_source(host_el)
  108. # host_el_type = od.get_type(bottom, host_el)
  109. # host_attr_name = od.get_attr_name(bottom, host_el_type)
  110. # output += f"\n{make_pattern_id(pattern_obj)}::{pattern_attr_name} -> {make_host_id(host_obj)}::{host_attr_name} {render_suffix}"
  111. return output
  112. def render_package(name, contents):
  113. output = f"subgraph cluster_{name} {{\n label=\"{name}\";"
  114. output += indent(contents, 2)
  115. output += "\n}\n"
  116. return output