# PlantUML renderer from services import scd, od from services.bottom.V0 import Bottom from transformation import ramify from concrete_syntax.common import display_value, display_name from uuid import UUID def render_class_diagram(state, model, prefix_ids=""): bottom = Bottom(state) model_scd = scd.SCD(model, state) model_od = od.OD(od.get_scd_mm(bottom), model, state) def make_id(uuid) -> str: return prefix_ids+str(uuid).replace('-','_') output = "" # Render classes for name, class_node in model_scd.get_classes().items(): is_abstract = False slot = model_od.get_slot(class_node, "abstract") if slot != None: is_abstract, _ = od.read_primitive_value(bottom, slot, model_od.type_model) lower_card, upper_card = model_scd.get_class_cardinalities(class_node) if lower_card == None and upper_card == None: card_spec = "" else: low = "0" if lower_card == None else lower_card upp = "*" if upper_card == None else upper_card card_spec = f"{low}..{upp}" if is_abstract: output += f"\nabstract class \"{name} {card_spec}\" as {make_id(class_node)}" else: output += f"\nclass \"{name} {card_spec}\" as {make_id(class_node)}" # Render attributes output += " {" for attr_name, attr_edge in od.get_attributes(bottom, class_node): tgt_name = model_scd.get_class_name(bottom.read_edge_target(attr_edge)) output += f"\n {attr_name} : {tgt_name}" output += "\n}" output += "\n" # Render inheritance links for inh_node in model_scd.get_inheritances().values(): src_node = bottom.read_edge_source(inh_node) tgt_node = bottom.read_edge_target(inh_node) output += f"\n{make_id(tgt_node)} <|-- {make_id(src_node)}" output += "\n" # Render associations for assoc_name, assoc_edge in model_scd.get_associations().items(): src_node = bottom.read_edge_source(assoc_edge) tgt_node = bottom.read_edge_target(assoc_edge) src_lower_card, src_upper_card, tgt_lower_card, tgt_upper_card = model_scd.get_assoc_cardinalities(assoc_edge) # default cardinalities if src_lower_card == None: src_lower_card = 0 if src_upper_card == None: src_upper_card = "*" if tgt_lower_card == None: tgt_lower_card = 0 if tgt_upper_card == None: tgt_upper_card = "*" src_card = f"{src_lower_card} .. {src_upper_card}" tgt_card = f"{tgt_lower_card} .. {tgt_upper_card}" if src_card == "0 .. *": src_card = " " # hide cardinality if tgt_card == "1 .. 1": tgt_card = " " # hide cardinality output += f'\n{make_id(src_node)} "{src_card}" --> "{tgt_card}" {make_id(tgt_node)} : {assoc_name}' return output def render_object_diagram(state, m, mm, render_attributes=True, prefix_ids=""): bottom = Bottom(state) mm_scd = scd.SCD(mm, state) m_od = od.OD(mm, m, state) def make_id(uuid) -> str: return prefix_ids+str(uuid).replace('-','_') output = "" # Render objects for class_name, class_node in mm_scd.get_classes().items(): if render_attributes: attributes = od.get_attributes(bottom, class_node) for obj_name, obj_node in m_od.get_objects(class_node).items(): output += f"\nmap \"{display_name(obj_name)} : {class_name}\" as {make_id(obj_node)} {{" if render_attributes: for attr_name, attr_edge in attributes: slot = m_od.get_slot(obj_node, attr_name) if slot != None: val, type_name = od.read_primitive_value(bottom, slot, mm) escaped_newline = ";" output += f"\n{attr_name} => {display_value(val, type_name, newline_character=escaped_newline)}" output += '\n}' output += '\n' # Render links for assoc_name, assoc_edge in mm_scd.get_associations().items(): for link_name, link_edge in m_od.get_objects(assoc_edge).items(): src_obj = bottom.read_edge_source(link_edge) tgt_obj = bottom.read_edge_target(link_edge) src_name = m_od.get_object_name(src_obj) tgt_name = m_od.get_object_name(tgt_obj) output += f"\n{make_id(src_obj)} -> {make_id(tgt_obj)} : {display_name(link_name)}:{assoc_name}" return output def render_package(name, contents): output = "" output += f'\npackage "{name}" {{' output += contents output += '\n}' return output def render_trace_ramifies(state, mm, ramified_mm, render_attributes=True, prefix_ram_ids="", prefix_orig_ids=""): bottom = Bottom(state) mm_scd = scd.SCD(mm, state) ramified_mm_scd = scd.SCD(ramified_mm, state) def make_ram_id(uuid) -> str: return prefix_ram_ids+str(uuid).replace('-','_') def make_orig_id(uuid) -> str: return prefix_orig_ids+str(uuid).replace('-','_') output = "" # Render RAMifies-edges between classes for ram_name, ram_class_node in ramified_mm_scd.get_classes().items(): original_class = ramify.get_original_type(bottom, ram_class_node) if original_class == None: continue # not all classes come from original (e.g., 'GlobalCondition') original_name = mm_scd.get_class_name(original_class) output += f"\n{make_ram_id(ram_class_node)} ..> {make_orig_id(original_class)} #line:green;text:green : RAMifies" if render_attributes: # and between attributes for (ram_attr_name, ram_attr_edge) in od.get_attributes(bottom, ram_class_node): orig_attr_edge = ramify.get_original_type(bottom, ram_attr_edge) if orig_attr_edge == None: continue # not all attributes come from original (e.g., 'condition') orig_class_node = bottom.read_edge_source(orig_attr_edge) # dirty AF: orig_attr_name = mm_scd.get_class_name(orig_attr_edge)[len(original_name)+1:] 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" return output def render_trace_conformance(state, m, mm, render_attributes=True, prefix_inst_ids="", prefix_type_ids=""): bottom = Bottom(state) mm_scd = scd.SCD(mm, state) m_od = od.OD(mm, m, state) def make_inst_id(uuid) -> str: return prefix_inst_ids+str(uuid).replace('-','_') def make_type_id(uuid) -> str: return prefix_type_ids+str(uuid).replace('-','_') output = "" # Render objects for class_name, class_node in mm_scd.get_classes().items(): if render_attributes: attributes = od.get_attributes(bottom, class_node) for obj_name, obj_node in m_od.get_objects(class_node).items(): output += f"\n{make_inst_id(obj_node)} ..> {make_type_id(class_node)} #line:blue;text:blue : instanceOf" if render_attributes: for attr_name, attr_edge in attributes: slot = m_od.get_slot(obj_node, attr_name) if slot != None: output += f"\n{make_inst_id(obj_node)}::{attr_name} ..> {make_type_id(class_node)}::{attr_name} #line:blue;text:blue : instanceOf" output += '\n' return output def render_trace_match(state, name_mapping: dict, pattern_m: UUID, host_m: UUID, color="grey", prefix_pattern_ids="", prefix_host_ids=""): bottom = Bottom(state) class_type = od.get_scd_mm_class_node(bottom) attr_link_type = od.get_scd_mm_attributelink_node(bottom) def make_pattern_id(uuid) -> str: return prefix_pattern_ids+str(uuid).replace('-','_') def make_host_id(uuid) -> str: return prefix_host_ids+str(uuid).replace('-','_') output = "" render_suffix = f"#line:{color};line.dotted;text:{color} : matchedWith" for pattern_el_name, host_el_name in name_mapping.items(): # print(pattern_el_name, host_el_name) try: pattern_el, = bottom.read_outgoing_elements(pattern_m, pattern_el_name) host_el, = bottom.read_outgoing_elements(host_m, host_el_name) except: continue # only render 'match'-edges between objects (= those elements where the type of the type is 'Class'): pattern_el_type = od.get_type(bottom, pattern_el) pattern_el_type_type = od.get_type(bottom, pattern_el_type) if pattern_el_type_type == class_type: output += f"\n{make_pattern_id(pattern_el)} ..> {make_host_id(host_el)} {render_suffix}" elif pattern_el_type_type == attr_link_type: pattern_obj = bottom.read_edge_source(pattern_el) pattern_attr_name = od.get_attr_name(bottom, pattern_el_type) host_obj = bottom.read_edge_source(host_el) host_el_type = od.get_type(bottom, host_el) host_attr_name = od.get_attr_name(bottom, host_el_type) output += f"\n{make_pattern_id(pattern_obj)}::{pattern_attr_name} ..> {make_host_id(host_obj)}::{host_attr_name} {render_suffix}" return output