conformance.py 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591
  1. from services.bottom.V0 import Bottom
  2. from uuid import UUID
  3. from state.base import State
  4. from typing import Dict, Tuple, Set, Any, List
  5. from pprint import pprint
  6. import functools
  7. class Conformance:
  8. def __init__(self, state: State, model: UUID, type_model: UUID):
  9. self.state = state
  10. self.bottom = Bottom(state)
  11. type_model_id = state.read_dict(state.read_root(), "SCD")
  12. self.scd_model = UUID(state.read_value(type_model_id))
  13. self.model = model
  14. self.type_model = type_model
  15. self.type_mapping: Dict[str, str] = {}
  16. self.model_names = {
  17. # map model elements to their names to prevent iterating too much
  18. self.bottom.read_outgoing_elements(self.model, e)[0]: e
  19. for e in self.bottom.read_keys(self.model)
  20. }
  21. self.type_model_names = {
  22. # map type model elements to their names to prevent iterating too much
  23. self.bottom.read_outgoing_elements(self.type_model, e)[0]
  24. : e for e in self.bottom.read_keys(self.type_model)
  25. }
  26. self.sub_types: Dict[str, Set[str]] = {
  27. k: set() for k in self.bottom.read_keys(self.type_model)
  28. }
  29. self.primitive_values: Dict[UUID, Any] = {}
  30. self.abstract_types: List[str] = []
  31. self.multiplicities: Dict[str, Tuple] = {}
  32. self.source_multiplicities: Dict[str, Tuple] = {}
  33. self.target_multiplicities: Dict[str, Tuple] = {}
  34. self.structures = {}
  35. self.matches = {}
  36. self.candidates = {}
  37. def check_nominal(self, *, log=False):
  38. """
  39. Perform a nominal conformance check
  40. Args:
  41. log: boolean indicating whether to log errors
  42. Returns:
  43. Boolean indicating whether the check has passed
  44. """
  45. try:
  46. self.check_typing()
  47. self.check_link_typing()
  48. self.check_multiplicities()
  49. self.check_constraints()
  50. return True
  51. except RuntimeError as e:
  52. if log:
  53. print(e)
  54. return False
  55. def check_structural(self, *, build_morphisms=True, log=False):
  56. """
  57. Perform a structural conformance check
  58. Args:
  59. build_morphisms: boolean indicating whether to create morpishm links
  60. log: boolean indicating whether to log errors
  61. Returns:
  62. Boolean indicating whether the check has passed
  63. """
  64. try:
  65. self.precompute_structures()
  66. self.match_structures()
  67. if build_morphisms:
  68. self.build_morphisms()
  69. self.check_nominal(log=log)
  70. return True
  71. except RuntimeError as e:
  72. if log:
  73. print(e)
  74. return False
  75. def read_attribute(self, element: UUID, attr_name: str):
  76. """
  77. Read an attribute value attached to an element
  78. Args:
  79. element: UUID of the element
  80. attr_name: name of the attribute to read
  81. Returns:
  82. The value of hte attribute, if no attribute with given name is found, returns None
  83. """
  84. if element in self.type_model_names:
  85. # type model element
  86. element_name = self.type_model_names[element]
  87. model = self.type_model
  88. else:
  89. # model element
  90. element_name = self.model_names[element]
  91. model = self.model
  92. try:
  93. attr_elem, = self.bottom.read_outgoing_elements(model, f"{element_name}.{attr_name}")
  94. return self.primitive_values.get(attr_elem, self.bottom.read_value(attr_elem))
  95. except ValueError:
  96. return None
  97. def precompute_sub_types(self):
  98. """
  99. Creates an internal representation of sub-type hierarchies that is
  100. more easily queryable that the state graph
  101. """
  102. # collect inheritance link instances
  103. inh_element, = self.bottom.read_outgoing_elements(self.scd_model, "Inheritance")
  104. inh_links = []
  105. for tm_element, tm_name in self.type_model_names.items():
  106. morphisms = self.bottom.read_outgoing_elements(tm_element, "Morphism")
  107. if inh_element in morphisms:
  108. # we have an instance of an inheritance link
  109. inh_links.append(tm_element)
  110. # for each inheritance link we add the parent and child to the sub types map
  111. for link in inh_links:
  112. tm_source = self.bottom.read_edge_source(link)
  113. tm_target = self.bottom.read_edge_target(link)
  114. parent_name = self.type_model_names[tm_target]
  115. child_name = self.type_model_names[tm_source]
  116. self.sub_types[parent_name].add(child_name)
  117. # iteratively expand the sub type hierarchies in the sub types map
  118. stop = False
  119. while not stop:
  120. stop = True
  121. for child_name, child_children in self.sub_types.items():
  122. for parent_name, parent_children in self.sub_types.items():
  123. if child_name in parent_children:
  124. original_size = len(parent_children)
  125. parent_children.update(child_children)
  126. if len(parent_children) != original_size:
  127. stop = False
  128. def deref_primitive_values(self):
  129. """
  130. Prefetch the values stored in referenced primitive type models
  131. """
  132. ref_element, = self.bottom.read_outgoing_elements(self.scd_model, "ModelRef")
  133. string_element, = self.bottom.read_outgoing_elements(self.scd_model, "String")
  134. boolean_element, = self.bottom.read_outgoing_elements(self.scd_model, "Boolean")
  135. integer_element, = self.bottom.read_outgoing_elements(self.scd_model, "Integer")
  136. t_deref = []
  137. t_refs = []
  138. for tm_element, tm_name in self.type_model_names.items():
  139. morphisms = self.bottom.read_outgoing_elements(tm_element, "Morphism")
  140. if ref_element in morphisms:
  141. t_refs.append(self.type_model_names[tm_element])
  142. elif string_element in morphisms:
  143. t_deref.append(tm_element)
  144. elif boolean_element in morphisms:
  145. t_deref.append(tm_element)
  146. elif integer_element in morphisms:
  147. t_deref.append(tm_element)
  148. for elem in t_deref:
  149. primitive_model = UUID(self.bottom.read_value(elem))
  150. primitive_value_node, = self.bottom.read_outgoing_elements(primitive_model)
  151. primitive_value = self.bottom.read_value(primitive_value_node)
  152. self.primitive_values[elem] = primitive_value
  153. for m_name, tm_name in self.type_mapping.items():
  154. if tm_name in t_refs:
  155. # dereference
  156. m_element, = self.bottom.read_outgoing_elements(self.model, m_name)
  157. primitive_model = UUID(self.bottom.read_value(m_element))
  158. try:
  159. primitive_value_node, = self.bottom.read_outgoing_elements(primitive_model)
  160. primitive_value = self.bottom.read_value(primitive_value_node)
  161. self.primitive_values[m_element] = primitive_value
  162. except ValueError:
  163. pass # multiple elements in model indicate that we're not dealing with a primitive
  164. def precompute_multiplicities(self):
  165. """
  166. Creates an internal representation of type multiplicities that is
  167. more easily queryable that the state graph
  168. """
  169. for tm_element, tm_name in self.type_model_names.items():
  170. # class abstract flags and multiplicities
  171. abstract = self.read_attribute(tm_element, "abstract")
  172. lc = self.read_attribute(tm_element, "lower_cardinality")
  173. uc = self.read_attribute(tm_element, "upper_cardinality")
  174. if abstract:
  175. self.abstract_types.append(tm_name)
  176. if lc or uc:
  177. mult = (
  178. lc if lc != None else float("-inf"),
  179. uc if uc != None else float("inf")
  180. )
  181. self.multiplicities[tm_name] = mult
  182. # multiplicities for associations
  183. slc = self.read_attribute(tm_element, "source_lower_cardinality")
  184. suc = self.read_attribute(tm_element, "source_upper_cardinality")
  185. if slc or suc:
  186. mult = (
  187. slc if slc != None else float("-inf"),
  188. suc if suc != None else float("inf")
  189. )
  190. self.source_multiplicities[tm_name] = mult
  191. tlc = self.read_attribute(tm_element, "target_lower_cardinality")
  192. tuc = self.read_attribute(tm_element, "target_upper_cardinality")
  193. if tlc or tuc:
  194. mult = (
  195. tlc if tlc != None else float("-inf"),
  196. tuc if tuc != None else float("inf")
  197. )
  198. self.target_multiplicities[tm_name] = mult
  199. # optional for attribute links
  200. opt = self.read_attribute(tm_element, "optional")
  201. if opt != None:
  202. self.source_multiplicities[tm_name] = (0 if opt else 1, 1)
  203. self.target_multiplicities[tm_name] = (0, 1)
  204. def get_type(self, element: UUID):
  205. """
  206. Retrieve the type of an element (wrt. current type model)
  207. """
  208. morphisms = self.bottom.read_outgoing_elements(element, "Morphism")
  209. tm_element, = [m for m in morphisms if m in self.type_model_names.keys()]
  210. return tm_element
  211. def check_typing(self):
  212. """
  213. for each element of model check whether a morphism
  214. link exists to some element of type_model
  215. """
  216. ref_element, = self.bottom.read_outgoing_elements(self.scd_model, "ModelRef")
  217. model_names = self.bottom.read_keys(self.model)
  218. for m_name in model_names:
  219. m_element, = self.bottom.read_outgoing_elements(self.model, m_name)
  220. try:
  221. tm_element = self.get_type(m_element)
  222. tm_name = self.type_model_names[tm_element]
  223. self.type_mapping[m_name] = tm_name
  224. if ref_element in self.bottom.read_outgoing_elements(tm_element, "Morphism"):
  225. sub_m = UUID(self.bottom.read_value(m_element))
  226. sub_tm = UUID(self.bottom.read_value(tm_element))
  227. if not Conformance(self.state, sub_m, sub_tm).check_nominal():
  228. raise RuntimeError(f"Incorrectly model reference: {m_name}")
  229. except ValueError as e:
  230. import traceback
  231. traceback.format_exc(e)
  232. # no or too many morphism links found
  233. raise RuntimeError(f"Incorrectly typed element: {m_name}")
  234. return True
  235. def check_link_typing(self):
  236. """
  237. for each link, check whether its source and target are of a valid type
  238. """
  239. self.precompute_sub_types()
  240. for m_name, tm_name in self.type_mapping.items():
  241. m_element, = self.bottom.read_outgoing_elements(self.model, m_name)
  242. m_source = self.bottom.read_edge_source(m_element)
  243. m_target = self.bottom.read_edge_target(m_element)
  244. if m_source == None or m_target == None:
  245. # element is not a link
  246. continue
  247. tm_element, = self.bottom.read_outgoing_elements(self.type_model, tm_name)
  248. tm_source = self.bottom.read_edge_source(tm_element)
  249. tm_target = self.bottom.read_edge_target(tm_element)
  250. # check if source is typed correctly
  251. source_name = self.model_names[m_source]
  252. source_type_actual = self.type_mapping[source_name]
  253. source_type_expected = self.type_model_names[tm_source]
  254. if source_type_actual != source_type_expected:
  255. if source_type_actual not in self.sub_types[source_type_expected]:
  256. raise RuntimeError(f"Invalid source type {source_type_actual} for element {m_name}")
  257. # check if target is typed correctly
  258. target_name = self.model_names[m_target]
  259. target_type_actual = self.type_mapping[target_name]
  260. target_type_expected = self.type_model_names[tm_target]
  261. if target_type_actual != target_type_expected:
  262. if target_type_actual not in self.sub_types[target_type_expected]:
  263. raise RuntimeError(f"Invalid target type {target_type_actual} for element {m_name}")
  264. return True
  265. def check_multiplicities(self):
  266. """
  267. Check whether multiplicities for all types are respected
  268. """
  269. self.deref_primitive_values()
  270. self.precompute_multiplicities()
  271. for tm_name in self.type_model_names.values():
  272. # abstract classes
  273. if tm_name in self.abstract_types:
  274. type_count = list(self.type_mapping.values()).count(tm_name)
  275. if type_count > 0:
  276. raise RuntimeError(f"Invalid instantiation of abstract class: {tm_name}")
  277. # class multiplicities
  278. if tm_name in self.multiplicities:
  279. lc, uc = self.multiplicities[tm_name]
  280. type_count = list(self.type_mapping.values()).count(tm_name)
  281. for sub_type in self.sub_types[tm_name]:
  282. type_count += list(self.type_mapping.values()).count(sub_type)
  283. if type_count < lc or type_count > uc:
  284. raise RuntimeError(f"Cardinality of type exceeds valid multiplicity range: {tm_name} ({type_count})")
  285. # association source multiplicities
  286. if tm_name in self.source_multiplicities:
  287. tm_element, = self.bottom.read_outgoing_elements(self.type_model, tm_name)
  288. tm_source_element = self.bottom.read_edge_source(tm_element)
  289. tm_source_name = self.type_model_names[tm_source_element]
  290. lc, uc = self.source_multiplicities[tm_name]
  291. for i, t in self.type_mapping.items():
  292. if t == tm_source_name or t in self.sub_types[tm_source_name]:
  293. count = 0
  294. i_element, = self.bottom.read_outgoing_elements(self.model, i)
  295. outgoing = self.bottom.read_outgoing_edges(i_element)
  296. for o in outgoing:
  297. try:
  298. if self.type_mapping[self.model_names[o]] == tm_name:
  299. count += 1
  300. except KeyError:
  301. pass # for elements not part of model, e.g. morphism links
  302. if count < lc or count > uc:
  303. raise RuntimeError(f"Source cardinality of type {tm_name} exceeds valid multiplicity range in {i}.")
  304. # association target multiplicities
  305. if tm_name in self.target_multiplicities:
  306. tm_element, = self.bottom.read_outgoing_elements(self.type_model, tm_name)
  307. tm_target_element = self.bottom.read_edge_target(tm_element)
  308. tm_target_name = self.type_model_names[tm_target_element]
  309. lc, uc = self.target_multiplicities[tm_name]
  310. for i, t in self.type_mapping.items():
  311. if t == tm_target_name or t in self.sub_types[tm_target_name]:
  312. count = 0
  313. i_element, = self.bottom.read_outgoing_elements(self.model, i)
  314. outgoing = self.bottom.read_incoming_edges(i_element)
  315. for o in outgoing:
  316. try:
  317. if self.type_mapping[self.model_names[o]] == tm_name:
  318. count += 1
  319. except KeyError:
  320. pass # for elements not part of model, e.g. morphism links
  321. if count < lc or count > uc:
  322. print(f"Target cardinality of type {tm_name} exceeds valid multiplicity range in {i}.")
  323. return False
  324. return True
  325. def evaluate_constraint(self, code, **kwargs):
  326. """
  327. Evaluate constraint code (Python code)
  328. """
  329. funcs = {
  330. 'read_value': self.state.read_value
  331. }
  332. return eval(
  333. code,
  334. {'__builtins__': {'isinstance': isinstance, 'print': print,
  335. 'int': int, 'float': float, 'bool': bool, 'str': str, 'tuple': tuple}
  336. }, # globals
  337. {**kwargs, **funcs} # locals
  338. )
  339. def check_constraints(self):
  340. """
  341. Check whether all constraints defined for a model are respected
  342. """
  343. # local constraints
  344. for m_name, tm_name in self.type_mapping.items():
  345. if tm_name != "GlobalConstraint":
  346. tm_element, = self.bottom.read_outgoing_elements(self.type_model, tm_name)
  347. code = self.read_attribute(tm_element, "constraint")
  348. if code != None:
  349. morphisms = self.bottom.read_incoming_elements(tm_element, "Morphism")
  350. morphisms = [m for m in morphisms if m in self.model_names]
  351. for m_element in morphisms:
  352. if not self.evaluate_constraint(code, element=m_element):
  353. raise RuntimeError(f"Local constraint of {tm_name} not satisfied in {m_name}.")
  354. # global constraints
  355. for m_name, tm_name in self.type_mapping.items():
  356. if tm_name == "GlobalConstraint":
  357. tm_element, = self.bottom.read_outgoing_elements(self.type_model, tm_name)
  358. code = self.read_attribute(tm_element, "constraint")
  359. if code != None:
  360. if not self.evaluate_constraint(code, model=self.model):
  361. raise RuntimeError(f"Global constraint {tm_name} not satisfied.")
  362. return True
  363. def precompute_structures(self):
  364. """
  365. Make an internal representation of type structures such that comparing type structures is easier
  366. """
  367. self.precompute_sub_types()
  368. scd_elements = self.bottom.read_outgoing_elements(self.scd_model)
  369. # collect types
  370. class_element, = self.bottom.read_outgoing_elements(self.scd_model, "Class")
  371. association_element, = self.bottom.read_outgoing_elements(self.scd_model, "Association")
  372. for tm_element, tm_name in self.type_model_names.items():
  373. # retrieve elements that tm_element is a morphism of
  374. morphisms = self.bottom.read_outgoing_elements(tm_element, "Morphism")
  375. morphism, = [m for m in morphisms if m in scd_elements]
  376. # check if tm_element is a morphism of AttributeLink
  377. if class_element == morphism or association_element == morphism:
  378. self.structures[tm_name] = set()
  379. # collect type structures
  380. # retrieve AttributeLink to check whether element is a morphism of AttributeLink
  381. attr_link_element, = self.bottom.read_outgoing_elements(self.scd_model, "AttributeLink")
  382. for tm_element, tm_name in self.type_model_names.items():
  383. # retrieve elements that tm_element is a morphism of
  384. morphisms = self.bottom.read_outgoing_elements(tm_element, "Morphism")
  385. morphism, = [m for m in morphisms if m in scd_elements]
  386. # check if tm_element is a morphism of AttributeLink
  387. if attr_link_element == morphism:
  388. # retrieve attributes of attribute link, i.e. 'name' and 'optional'
  389. attrs = self.bottom.read_outgoing_elements(tm_element)
  390. name_model_node, = filter(lambda x: self.type_model_names.get(x, "").endswith(".name"), attrs)
  391. opt_model_node, = filter(lambda x: self.type_model_names.get(x, "").endswith(".optional"), attrs)
  392. # get attr name value
  393. name_model = UUID(self.bottom.read_value(name_model_node))
  394. name_node, = self.bottom.read_outgoing_elements(name_model)
  395. name = self.bottom.read_value(name_node)
  396. # get attr opt value
  397. opt_model = UUID(self.bottom.read_value(opt_model_node))
  398. opt_node, = self.bottom.read_outgoing_elements(opt_model)
  399. opt = self.bottom.read_value(opt_node)
  400. # get attr type name
  401. source_type_node = self.bottom.read_edge_source(tm_element)
  402. source_type_name = self.type_model_names[source_type_node]
  403. target_type_node = self.bottom.read_edge_target(tm_element)
  404. target_type_name = self.type_model_names[target_type_node]
  405. # add attribute to the structure of its source type
  406. # attribute is stored as a (name, optional, type) triple
  407. self.structures.setdefault(source_type_name, set()).add((name, opt, target_type_name))
  408. # extend structures of sub types with attrs of super types
  409. for super_type, sub_types in self.sub_types.items():
  410. for sub_type in sub_types:
  411. self.structures.setdefault(sub_type, set()).update(self.structures[super_type])
  412. # filter out abstract types, as they cannot be instantiated
  413. # retrieve Class_abstract to check whether element is a morphism of Class_abstract
  414. class_abs_element, = self.bottom.read_outgoing_elements(self.scd_model, "Class_abstract")
  415. for tm_element, tm_name in self.type_model_names.items():
  416. # retrieve elements that tm_element is a morphism of
  417. morphisms = self.bottom.read_outgoing_elements(tm_element, "Morphism")
  418. morphism, = [m for m in morphisms if m in scd_elements]
  419. # check if tm_element is a morphism of Class_abstract
  420. if class_abs_element == morphism:
  421. # retrieve 'abstract' attribute value
  422. target_node = self.bottom.read_edge_target(tm_element)
  423. abst_model = UUID(self.bottom.read_value(target_node))
  424. abst_node, = self.bottom.read_outgoing_elements(abst_model)
  425. is_abstract = self.bottom.read_value(abst_node)
  426. # retrieve type name
  427. source_node = self.bottom.read_edge_source(tm_element)
  428. type_name = self.type_model_names[source_node]
  429. if is_abstract:
  430. self.structures.pop(type_name)
  431. def match_structures(self):
  432. """
  433. Try to match the structure of each element in the instance model to some element in the type model
  434. """
  435. ref_element, = self.bottom.read_outgoing_elements(self.scd_model, "ModelRef")
  436. # matching
  437. for m_element, m_name in self.model_names.items():
  438. is_edge = self.bottom.read_edge_source(m_element) != None
  439. print('element:', m_element, 'name:', m_name, 'is_edge', is_edge)
  440. for type_name, structure in self.structures.items():
  441. tm_element, = self.bottom.read_outgoing_elements(self.type_model, type_name)
  442. type_is_edge = self.bottom.read_edge_source(tm_element) != None
  443. if is_edge == type_is_edge:
  444. print(' type_name:', type_name, 'type_is_edge:', type_is_edge, "structure:", structure)
  445. mismatch = False
  446. matched = 0
  447. for name, optional, attr_type in structure:
  448. print(' name:', name, "optional:", optional, "attr_type:", attr_type)
  449. try:
  450. attr, = self.bottom.read_outgoing_elements(self.model, f"{m_name}.{name}")
  451. attr_tm, = self.bottom.read_outgoing_elements(self.type_model, attr_type)
  452. # if attribute is a modelref, we need to check whether it
  453. # linguistically conforms to the specified type
  454. # if its an internally defined attribute, this will be checked by constraints
  455. morphisms = self.bottom.read_outgoing_elements(attr_tm, "Morphism")
  456. attr_conforms = True
  457. if ref_element in morphisms:
  458. # check conformance of reference model
  459. type_model_uuid = UUID(self.bottom.read_value(attr_tm))
  460. model_uuid = UUID(self.bottom.read_value(attr))
  461. attr_conforms = Conformance(self.state, model_uuid, type_model_uuid)\
  462. .check_nominal()
  463. else:
  464. # eval constraints
  465. code = self.read_attribute(attr_tm, "constraint")
  466. if code != None:
  467. attr_conforms = self.evaluate_constraint(code, element=attr)
  468. if attr_conforms:
  469. matched += 1
  470. print(" attr_conforms -> matched:", matched)
  471. except ValueError as e:
  472. # attr not found or failed parsing UUID
  473. if optional:
  474. print(" skipping:", e)
  475. continue
  476. else:
  477. # did not match mandatory attribute
  478. print(" breaking:", e)
  479. mismatch = True
  480. break
  481. print(' matched:', matched, 'len(structure):', len(structure))
  482. # if matched == len(structure):
  483. if not mismatch:
  484. print(' add to candidates:', m_name, type_name)
  485. self.candidates.setdefault(m_name, set()).add(type_name)
  486. # filter out candidates for links based on source and target types
  487. for m_element, m_name in self.model_names.items():
  488. is_edge = self.bottom.read_edge_source(m_element) != None
  489. if is_edge and m_name in self.candidates:
  490. m_source = self.bottom.read_edge_source(m_element)
  491. m_target = self.bottom.read_edge_target(m_element)
  492. print(self.candidates)
  493. source_candidates = self.candidates[self.model_names[m_source]]
  494. target_candidates = self.candidates[self.model_names[m_target]]
  495. remove = set()
  496. for candidate_name in self.candidates[m_name]:
  497. candidate_element, = self.bottom.read_outgoing_elements(self.type_model, candidate_name)
  498. candidate_source = self.type_model_names[self.bottom.read_edge_source(candidate_element)]
  499. if candidate_source not in source_candidates:
  500. if len(source_candidates.intersection(set(self.sub_types[candidate_source]))) == 0:
  501. remove.add(candidate_name)
  502. candidate_target = self.type_model_names[self.bottom.read_edge_target(candidate_element)]
  503. if candidate_target not in target_candidates:
  504. if len(target_candidates.intersection(set(self.sub_types[candidate_target]))) == 0:
  505. remove.add(candidate_name)
  506. self.candidates[m_name] = self.candidates[m_name].difference(remove)
  507. def build_morphisms(self):
  508. """
  509. Build the morphisms between an instance and a type model that structurally match
  510. """
  511. if not all([len(c) == 1 for c in self.candidates.values()]):
  512. raise RuntimeError("Cannot build incomplete or ambiguous morphism.")
  513. mapping = {k: v.pop() for k, v in self.candidates.items()}
  514. for m_name, tm_name in mapping.items():
  515. # morphism to class/assoc
  516. m_element, = self.bottom.read_outgoing_elements(self.model, m_name)
  517. tm_element, = self.bottom.read_outgoing_elements(self.type_model, tm_name)
  518. self.bottom.create_edge(m_element, tm_element, "Morphism")
  519. # morphism for attributes and attribute links
  520. structure = self.structures[tm_name]
  521. for attr_name, _, attr_type in structure:
  522. try:
  523. # attribute node
  524. attr_element, = self.bottom.read_outgoing_elements(self.model, f"{m_name}.{attr_name}")
  525. attr_type_element, = self.bottom.read_outgoing_elements(self.type_model, attr_type)
  526. self.bottom.create_edge(attr_element, attr_type_element, "Morphism")
  527. # attribute link
  528. attr_link_element, = self.bottom.read_outgoing_elements(self.model, f"{m_name}.{attr_name}_link")
  529. attr_link_type_element, = self.bottom.read_outgoing_elements(self.type_model, f"{tm_name}_{attr_name}")
  530. self.bottom.create_edge(attr_link_element, attr_link_type_element, "Morphism")
  531. except ValueError:
  532. pass
  533. if __name__ == '__main__':
  534. from state.devstate import DevState as State
  535. s = State()
  536. from bootstrap.scd import bootstrap_scd
  537. scd = bootstrap_scd(s)
  538. from bootstrap.pn import bootstrap_pn
  539. ltm_pn = bootstrap_pn(s, "PN")
  540. ltm_pn_lola = bootstrap_pn(s, "PNlola")
  541. from services.pn import PN
  542. my_pn = s.create_node()
  543. PNserv = PN(my_pn, s)
  544. PNserv.create_place("p1", 5)
  545. PNserv.create_place("p2", 0)
  546. PNserv.create_transition("t1")
  547. PNserv.create_p2t("p1", "t1", 1)
  548. PNserv.create_t2p("t1", "p2", 1)
  549. cf = Conformance(s, my_pn, ltm_pn_lola)
  550. # cf = Conformance(s, scd, ltm_pn, scd)
  551. cf.precompute_structures()
  552. cf.match_structures()
  553. cf.build_morphisms()
  554. print(cf.check_nominal())