(WIP) implementing CBD language... Meta-meta-model: Association inherits from Class. Matcher accepts pivot. Add generic graphviz renderer.
This commit is contained in:
parent
a26ceef10f
commit
1eb8a84553
25 changed files with 542 additions and 170 deletions
|
|
@ -17,6 +17,11 @@ def display_value(val: any, type_name: str, indentation=0):
|
|||
else:
|
||||
raise Exception("don't know how to display value" + type_name)
|
||||
|
||||
def display_name(raw_name: str) -> str:
|
||||
if raw_name[0:2] == "__":
|
||||
return "" # hide names that start with '__', they are anonymous (by convention)
|
||||
else:
|
||||
return raw_name
|
||||
|
||||
# internal use only
|
||||
# just a dumb wrapper to distinguish between code and string
|
||||
|
|
|
|||
12
concrete_syntax/graphviz/make_url.py
Normal file
12
concrete_syntax/graphviz/make_url.py
Normal file
|
|
@ -0,0 +1,12 @@
|
|||
from concrete_syntax.common import indent
|
||||
import urllib.parse
|
||||
|
||||
def make_url(graphviz_txt: str) -> str:
|
||||
|
||||
as_digraph = f"digraph {{\n{indent(graphviz_txt, 2)}\n}}"
|
||||
|
||||
# This one seems much faster:
|
||||
return "https://edotor.net/?engine=dot#"+urllib.parse.quote(as_digraph)
|
||||
|
||||
# Keeping this one here just in case:
|
||||
# return "https://dreampuf.github.io/GraphvizOnline/#"+urllib.parse.quote(graphviz)
|
||||
88
concrete_syntax/graphviz/renderer.py
Normal file
88
concrete_syntax/graphviz/renderer.py
Normal file
|
|
@ -0,0 +1,88 @@
|
|||
from uuid import UUID
|
||||
from services import scd, od
|
||||
from services.bottom.V0 import Bottom
|
||||
from concrete_syntax.common import display_value, display_name, indent
|
||||
|
||||
|
||||
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 'n'+(prefix_ids+str(uuid).replace('-',''))[24:]
|
||||
|
||||
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_id(obj_node)} [label=\"{display_name(obj_name)} : {class_name}\", shape=rect] ;"
|
||||
#" {{"
|
||||
|
||||
# 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)
|
||||
# output += f"\n{attr_name} => {display_value(val, type_name)}"
|
||||
# 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)} [label=\"{display_name(link_name)}:{assoc_name}\"] ;"
|
||||
|
||||
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 'n'+(prefix_pattern_ids+str(uuid).replace('-',''))[24:]
|
||||
def make_host_id(uuid) -> str:
|
||||
return 'n'+(prefix_host_ids+str(uuid).replace('-',''))[24:]
|
||||
|
||||
output = ""
|
||||
|
||||
# render_suffix = f"#line:{color};line.dotted;text:{color} : matchedWith"
|
||||
render_suffix = f"[label=\"\",style=dashed,color={color}] ;"
|
||||
|
||||
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
|
||||
|
||||
def render_package(name, contents):
|
||||
output = f"subgraph cluster_{name} {{\n label=\"{name}\";"
|
||||
output += indent(contents, 2)
|
||||
output += "\n}\n"
|
||||
return output
|
||||
21
concrete_syntax/plantuml/make_url.py
Normal file
21
concrete_syntax/plantuml/make_url.py
Normal file
|
|
@ -0,0 +1,21 @@
|
|||
from zlib import compress
|
||||
import base64
|
||||
import string
|
||||
|
||||
maketrans = bytes.maketrans
|
||||
|
||||
# Includes code fragments from: https://github.com/dougn/python-plantuml/blob/bb5407e87aabbac9e8baef5a6726b03f72afca16/plantuml.py
|
||||
# Copyright (c) 2013, Doug Napoleone and then Copyright (c) 2015, Samuel Marks
|
||||
|
||||
plantuml_alphabet = string.digits + string.ascii_uppercase + string.ascii_lowercase + '-_'
|
||||
base64_alphabet = string.ascii_uppercase + string.ascii_lowercase + string.digits + '+/'
|
||||
b64_to_plantuml = maketrans(base64_alphabet.encode('utf-8'), plantuml_alphabet.encode('utf-8'))
|
||||
|
||||
def encode(plantuml_text: str) -> str:
|
||||
zlibbed_str = compress(plantuml_text.encode('utf-8'))
|
||||
compressed_string = zlibbed_str[2:-4]
|
||||
return base64.b64encode(compressed_string).translate(b64_to_plantuml).decode('utf-8')
|
||||
|
||||
def make_url(plantuml_text: str) -> str:
|
||||
encoded = encode(plantuml_text)
|
||||
return f"https://deemz.org/plantuml/pdf/{encoded}"
|
||||
|
|
@ -3,9 +3,10 @@
|
|||
from services import scd, od
|
||||
from services.bottom.V0 import Bottom
|
||||
from transformation import ramify
|
||||
from concrete_syntax.common import display_value
|
||||
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)
|
||||
|
|
@ -102,7 +103,7 @@ def render_object_diagram(state, m, mm, render_attributes=True, prefix_ids=""):
|
|||
attributes = od.get_attributes(bottom, class_node)
|
||||
|
||||
for obj_name, obj_node in m_od.get_objects(class_node).items():
|
||||
output += f"\nmap \"{obj_name} : {class_name}\" as {make_id(obj_node)} {{"
|
||||
output += f"\nmap \"{display_name(obj_name)} : {class_name}\" as {make_id(obj_node)} {{"
|
||||
|
||||
if render_attributes:
|
||||
for attr_name, attr_edge in attributes:
|
||||
|
|
@ -122,7 +123,7 @@ def render_object_diagram(state, m, mm, render_attributes=True, prefix_ids=""):
|
|||
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)} : :{assoc_name}"
|
||||
output += f"\n{make_id(src_obj)} -> {make_id(tgt_obj)} : {display_name(link_name)}:{assoc_name}"
|
||||
|
||||
return output
|
||||
|
||||
|
|
@ -229,4 +230,3 @@ def render_trace_match(state, name_mapping: dict, pattern_m: UUID, host_m: UUID,
|
|||
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
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue