Skip to content

Commit

Permalink
Updated sync_reasoner() such that it works for FaCT++
Browse files Browse the repository at this point in the history
  • Loading branch information
jesper-friis committed Nov 24, 2024
1 parent 938dbb4 commit d6ca0be
Showing 1 changed file with 64 additions and 46 deletions.
110 changes: 64 additions & 46 deletions ontopy/ontology.py
Original file line number Diff line number Diff line change
Expand Up @@ -1343,67 +1343,85 @@ def sync_reasoner(
Keyword arguments are passed to the underlying owlready2 function.
"""
# pylint: disable=too-many-branches

removed_equivalent = defaultdict(list)
removed_subclasses = defaultdict(list)
# pylint: disable=too-many-branches,too-many-locals
# pylint: disable=unexpected-keyword-arg
removed_gspo = [] # obj: (ontology, s, p, o)
removed_gspod = [] # data: (ontology, s, p, o, d)

if reasoner == "FaCT++":
sync = sync_reasoner_factpp
remove_custom_datatypes = True
elif reasoner == "Pellet":
sync = owlready2.sync_reasoner_pellet
remove_custom_datatypes = False
elif reasoner == "HermiT":
sync = owlready2.sync_reasoner_hermit

# Remove custom data propertyes, otherwise HermiT will crash
datatype_iri = "http://www.w3.org/2000/01/rdf-schema#Datatype"

for cls in self.classes(imported=include_imported):
remove_eq = []
for i, r in enumerate(cls.equivalent_to):
if isinstance(r, owlready2.Restriction):
if (
hasattr(r.value.__class__, "iri")
and r.value.__class__.iri == datatype_iri
):
remove_eq.append(i)
removed_equivalent[cls].append(r)
for i in reversed(remove_eq):
del cls.equivalent_to[i]

remove_subcls = []
for i, r in enumerate(cls.is_a):
if isinstance(r, owlready2.Restriction):
if (
hasattr(r.value.__class__, "iri")
and r.value.__class__.iri == datatype_iri
):
remove_subcls.append(i)
removed_subclasses[cls].append(r)
for i in reversed(remove_subcls):
del cls.is_a[i]

remove_custom_datatypes = False
else:
raise ValueError(
f"Unknown reasoner '{reasoner}'. Supported reasoners "
"are 'Pellet', 'HermiT' and 'FaCT++'."
)

# For some reason we must visit all entities once before running
# the reasoner...
list(self.get_entities())
if include_imported:
ontologies = self.get_imported_ontologies(recursive=True)

Check warning on line 1367 in ontopy/ontology.py

View check run for this annotation

Codecov / codecov/patch

ontopy/ontology.py#L1367

Added line #L1367 was not covered by tests
else:
ontologies = [self]

with self:
if include_imported:
sync(self.world, **kwargs)
else:
sync(self, **kwargs)
if remove_custom_datatypes:
datatype = self._abbreviate(
"http://www.w3.org/2000/01/rdf-schema#Datatype"
)
for onto in ontologies:
# Collect all defined rdfs:Datatype instances
for s, p, o in onto._get_obj_triples_spo_spo(o=datatype):
for s2, p2, o2 in onto._get_obj_triples_spo_spo(s=s):
removed_gspo.append((onto, s2, p2, o2))

Check warning on line 1379 in ontopy/ontology.py

View check run for this annotation

Codecov / codecov/patch

ontopy/ontology.py#L1378-L1379

Added lines #L1378 - L1379 were not covered by tests

# Datatype instances that are known to crash the reasoner
datatypes = (
"http://www.w3.org/2002/07/owl#rational",
"http://www.w3.org/2001/XMLSchema#NCName",
"http://www.w3.org/2001/XMLSchema#NMTOKEN",
"http://www.w3.org/2001/XMLSchema#Name",
"http://www.w3.org/2001/XMLSchema#base64Binary",
"http://www.w3.org/2001/XMLSchema#dateTimeStamp",
"http://www.w3.org/2001/XMLSchema#hexBinary",
"http://www.w3.org/2001/XMLSchema#language",
"http://www.w3.org/2001/XMLSchema#nonPositiveInteger",
"http://www.w3.org/2001/XMLSchema#normalizedString",
"http://www.w3.org/2001/XMLSchema#token",
"http://www.w3.org/2001/XMLSchema#unsignedByte",
"http://www.w3.org/2001/XMLSchema#unsignedInt",
"http://www.w3.org/2001/XMLSchema#unsignedLong",
"http://www.w3.org/2001/XMLSchema#unsignedShort",
)
for dtype in datatypes:
d = onto._abbreviate(dtype)
for s, p, o in onto._get_obj_triples_spo_spo(o=d):
for s2, p2, o2 in onto._get_obj_triples_spo_spo(s=s):
removed_gspo.append((onto, s2, p2, o2))

Check warning on line 1403 in ontopy/ontology.py

View check run for this annotation

Codecov / codecov/patch

ontopy/ontology.py#L1402-L1403

Added lines #L1402 - L1403 were not covered by tests

# Restore removed custom data properties
for cls, eqs in removed_equivalent.items():
cls.extend(eqs)
for cls, subcls in removed_subclasses.items():
cls.extend(subcls)
# Remove triples selected for removal
try:
for g, s, p, o in removed_gspo:
g._del_obj_triple_spo(s, p, o)

Check warning on line 1408 in ontopy/ontology.py

View check run for this annotation

Codecov / codecov/patch

ontopy/ontology.py#L1408

Added line #L1408 was not covered by tests
for g, s, p, o, d in removed_gspod:
g._del_data_triple_spod(s, p, o, d)

Check warning on line 1410 in ontopy/ontology.py

View check run for this annotation

Codecov / codecov/patch

ontopy/ontology.py#L1410

Added line #L1410 was not covered by tests

# Run reasoner
with self:
if include_imported:
sync(self.world, **kwargs)

Check warning on line 1415 in ontopy/ontology.py

View check run for this annotation

Codecov / codecov/patch

ontopy/ontology.py#L1415

Added line #L1415 was not covered by tests
else:
sync(self, **kwargs)

# Restore removed triples
finally:
for g, s, p, o in removed_gspo:
g._add_obj_triple_spo(s, p, o)

Check warning on line 1422 in ontopy/ontology.py

View check run for this annotation

Codecov / codecov/patch

ontopy/ontology.py#L1422

Added line #L1422 was not covered by tests
for g, s, p, o, d in removed_gspod:
g.world._del_data_triple_spod(s, p, o, d)

Check warning on line 1424 in ontopy/ontology.py

View check run for this annotation

Codecov / codecov/patch

ontopy/ontology.py#L1424

Added line #L1424 was not covered by tests

def sync_attributes( # pylint: disable=too-many-branches
self,
Expand Down

0 comments on commit d6ca0be

Please sign in to comment.