All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.clarkparsia.owlapi.modularity.locality.SyntacticLocalityEvaluator Maven / Gradle / Ivy

There is a newer version: 5.5.1
Show newest version
/* This file is part of the OWL API.
 * The contents of this file are subject to the LGPL License, Version 3.0.
 * Copyright 2011, Clark & Parsia, LLC
 * 
 * This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.
 * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more details.
 * You should have received a copy of the GNU General Public License along with this program.  If not, see http://www.gnu.org/licenses/.
 *
 * Alternatively, the contents of this file may be used under the terms of the Apache License, Version 2.0 in which case, the provisions of the Apache License Version 2.0 are applicable instead of those above.
 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
 * http://www.apache.org/licenses/LICENSE-2.0
 * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */
package com.clarkparsia.owlapi.modularity.locality;

import static org.semanticweb.owlapi.util.OWLAPIPreconditions.*;

import java.util.Collection;
import java.util.EnumSet;
import java.util.Iterator;
import java.util.Set;

import javax.annotation.Nonnull;

import org.semanticweb.owlapi.model.*;

/** Syntactic locality evaluator. */
public class SyntacticLocalityEvaluator implements LocalityEvaluator {

    @Nonnull
    protected final LocalityClass localityCls;
    @Nonnull
    private final AxiomLocalityVisitor axiomVisitor = new AxiomLocalityVisitor();
    @Nonnull
    private static final EnumSet SUPPORTED_LOCALITY_CLASSES = EnumSet.of(LocalityClass.TOP_BOTTOM,
        LocalityClass.BOTTOM_BOTTOM, LocalityClass.TOP_TOP);

    /**
     * Constructs a new locality evaluator for the given locality class.
     * 
     * @param localityClass
     *        the locality class for this evaluator
     */
    public SyntacticLocalityEvaluator(@Nonnull LocalityClass localityClass) {
        localityCls = checkNotNull(localityClass, "localityClass cannot be null");
        if (!SUPPORTED_LOCALITY_CLASSES.contains(localityClass)) {
            throw new OWLRuntimeException("Unsupported locality class: " + localityClass);
        }
    }

    /**
     * Returns all supported locality classes.
     * 
     * @return a set containing all supported locality classes
     */
    public static Set supportedLocalityClasses() {
        return SUPPORTED_LOCALITY_CLASSES;
    }

    /**
     * This is a convenience method for determining whether a given data range
     * expression is the top datatype or a built-in datatype. This is used in
     * the bottom- and top-equivalence evaluators for treating cardinality
     * restrictions.
     * 
     * @param dataRange
     *        a data range expression
     * @return {@code true} if the specified data range expression is the top
     *         datatype or a built-in datatype; {@code false} otherwise
     */
    protected static boolean isTopOrBuiltInDatatype(@Nonnull OWLDataRange dataRange) {
        if (dataRange.isDatatype()) {
            OWLDatatype dataType = dataRange.asOWLDatatype();
            return dataType.isTopDatatype() || dataType.isBuiltIn();
        } else {
            return false;
        }
    }

    /**
     * This is a convenience method for determining whether a given data range
     * expression is the top datatype or a built-in infinite datatype. This is
     * used in the bottom- and top-equivalence evaluators for treating
     * cardinality restrictions.
     * 
     * @param dataRange
     *        a data range expression
     * @return {@code true} if the specified data range expression is the top
     *         datatype or a built-in infinite datatype; {@code false} otherwise
     */
    protected static boolean isTopOrBuiltInInfiniteDatatype(@Nonnull OWLDataRange dataRange) {
        if (dataRange.isDatatype()) {
            OWLDatatype dataType = dataRange.asOWLDatatype();
            return dataType.isTopDatatype() || dataType.isBuiltIn() && !dataType.getBuiltInDatatype().isFinite();
        } else {
            return false;
        }
    }

    // TODO (TS): only visit logical axioms if possible
    private class AxiomLocalityVisitor implements OWLAxiomVisitor {

        @Nonnull
        private final BottomEquivalenceEvaluator bottomEvaluator = new BottomEquivalenceEvaluator();
        private boolean isLocal;
        private Collection signature;
        @Nonnull
        private final TopEquivalenceEvaluator topEvaluator = new TopEquivalenceEvaluator();

        /** Instantiates a new axiom locality visitor. */
        AxiomLocalityVisitor() {
            topEvaluator.setBottomEvaluator(bottomEvaluator);
            bottomEvaluator.setTopEvaluator(topEvaluator);
        }

        @Nonnull
        protected Collection getSignature() {
            return verifyNotNull(signature);
        }

        /**
         * Checks if is local.
         * 
         * @param axiom
         *        the axiom
         * @param sig
         *        the sig
         * @return true, if is local
         */
        public boolean isLocal(@Nonnull OWLAxiom axiom, @Nonnull Collection sig) {
            signature = checkNotNull(sig, "sig cannot be null");
            isLocal = false;
            checkNotNull(axiom, "axiom cannot be null").accept(this);
            return isLocal;
        }

        @Override
        public void visit(OWLDatatypeDefinitionAxiom axiom) {
            isLocal = true;
        }

        // BUGFIX: (TS) Asymm OP axioms are local in the *_BOTTOM case:
        // The empty object property is asymmetric!
        // BUGFIX: (DT) OP in signature makes the axiom non-local
        @Override
        public void visit(OWLAsymmetricObjectPropertyAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getProperty().getNamedProperty());
                break;
            case TOP_TOP:
                isLocal = false;
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLClassAssertionAxiom axiom) {
            isLocal = topEvaluator.isTopEquivalent(axiom.getClassExpression(), getSignature(), localityCls);
        }

        @Override
        public void visit(OWLDataPropertyAssertionAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = false;
                break;
            case TOP_TOP:
                isLocal = !getSignature().contains(axiom.getProperty().asOWLDataProperty());
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLDataPropertyDomainAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getProperty().asOWLDataProperty()) || topEvaluator
                    .isTopEquivalent(axiom.getDomain(), getSignature(), localityCls);
                break;
            case TOP_TOP:
                isLocal = topEvaluator.isTopEquivalent(axiom.getDomain(), getSignature(), localityCls);
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS, 2) Added the cases where the filler is top-equiv
        @Override
        public void visit(OWLDataPropertyRangeAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getProperty().asOWLDataProperty()) || axiom.getRange()
                    .isTopDatatype();
                break;
            case TOP_TOP:
                isLocal = axiom.getRange().isTopDatatype();
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLSubDataPropertyOfAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getSubProperty().asOWLDataProperty());
                break;
            case TOP_TOP:
                isLocal = !getSignature().contains(axiom.getSuperProperty().asOWLDataProperty());
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS) Declaration axioms are local.
        // They need to be added to the module after the locality checks have
        // been performed.
        @Override
        public void visit(OWLDeclarationAxiom axiom) {
            isLocal = true;
        }

        // BUGFIX: (TS) Different individuals axioms are local, too.
        // They need to be added to the module after the locality checks have
        // been performed.
        @Override
        public void visit(OWLDifferentIndividualsAxiom axiom) {
            isLocal = true;
        }

        // BUGFIX: (TS) An n-ary disj classes axiom is local
        // iff at most one of the involved class expressions is not
        // bot-equivalent.
        @Override
        public void visit(OWLDisjointClassesAxiom axiom) {
            Collection disjs = axiom.getClassExpressions();
            int size = disjs.size();
            if (size == 1) {
                // XXX actually being here means the axiom is not OWL 2
                // conformant
                isLocal = true;
            } else {
                boolean nonBottomEquivDescFound = false;
                for (OWLClassExpression desc : disjs) {
                    assert desc != null;
                    if (!bottomEvaluator.isBottomEquivalent(desc, getSignature(), localityCls)) {
                        if (nonBottomEquivDescFound) {
                            isLocal = false;
                            return;
                        } else {
                            nonBottomEquivDescFound = true;
                        }
                    }
                }
            }
            isLocal = true;
        }

        // BUGFIX (TS): Added the case where it *is* local
        @Override
        public void visit(OWLDisjointDataPropertiesAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                Collection disjs = axiom.getProperties();
                int size = disjs.size();
                if (size == 1) {
                    // XXX actually being here means the axiom is not OWL 2
                    // conformant
                    isLocal = true;
                } else {
                    boolean nonBottomEquivPropFound = false;
                    for (OWLDataPropertyExpression dpe : disjs) {
                        if (getSignature().contains(dpe.asOWLDataProperty())) {
                            if (nonBottomEquivPropFound) {
                                isLocal = false;
                                return;
                            } else {
                                nonBottomEquivPropFound = true;
                            }
                        }
                    }
                }
                isLocal = true;
                break;
            case TOP_TOP:
                isLocal = false;
                break;
            default:
                break;
            }
        }

        // BUGFIX (TS): Added the case where it *is* local
        @Override
        public void visit(OWLDisjointObjectPropertiesAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                Collection disjs = axiom.getProperties();
                int size = disjs.size();
                if (size == 1) {
                    // XXX actually being here means the axiom is not OWL 2
                    // conformant
                    isLocal = true;
                } else {
                    boolean nonBottomEquivPropFound = false;
                    for (OWLObjectPropertyExpression ope : disjs) {
                        if (getSignature().contains(ope.getNamedProperty())) {
                            if (nonBottomEquivPropFound) {
                                isLocal = false;
                                return;
                            } else {
                                nonBottomEquivPropFound = true;
                            }
                        }
                    }
                }
                isLocal = true;
                break;
            case TOP_TOP:
                isLocal = false;
                break;
            default:
                break;
            }
        }

        // BUGFIX (TS): added the two cases where a disj union axiom *is* local:
        // - if LHS and all class expr on RHS are bot-equiv
        // - if LHS is top-equiv, one expr on RHS is top-equiv and the others
        // are bot-equiv
        @Override
        public void visit(OWLDisjointUnionAxiom axiom) {
            OWLClass lhs = axiom.getOWLClass();
            Collection rhs = axiom.getClassExpressions();
            if (localityCls == LocalityClass.BOTTOM_BOTTOM) {
                // TODO (TS): "!signature.contains(lhs)" is not enough
                // because lhs could be bot
                if (!getSignature().contains(lhs)) {
                    for (OWLClassExpression desc : rhs) {
                        assert desc != null;
                        if (!bottomEvaluator.isBottomEquivalent(desc, getSignature(), localityCls)) {
                            isLocal = false;
                            return;
                        }
                    }
                    isLocal = true;
                } else {
                    isLocal = false;
                }
            } else {
                // case TOP_BOTTOM:
                // case TOP_TOP:
                // TODO (TS): "!signature.contains(lhs)" is not enough
                // because lhs could be top
                if (!getSignature().contains(lhs)) {
                    boolean topEquivDescFound = false;
                    for (OWLClassExpression desc : rhs) {
                        assert desc != null;
                        if (!bottomEvaluator.isBottomEquivalent(desc, getSignature(), localityCls)) {
                            if (topEvaluator.isTopEquivalent(desc, getSignature(), localityCls)) {
                                if (topEquivDescFound) {
                                    isLocal = false;
                                    return;
                                } else {
                                    topEquivDescFound = true;
                                }
                            } else {
                                isLocal = false;
                                return;
                            }
                        }
                    }
                    isLocal = true;
                } else {
                    isLocal = false;
                }
            }
        }

        @Override
        public void visit(OWLAnnotationAssertionAxiom axiom) {
            isLocal = true;
        }

        @Override
        public void visit(OWLEquivalentClassesAxiom axiom) {
            isLocal = true;
            Iterator eqs = axiom.getClassExpressions().iterator();
            OWLClassExpression first = eqs.next();
            assert first != null;
            // axiom is local if it contains a single class expression
            if (!eqs.hasNext()) {
                return;
            }
            // axiom is local iff either all class expressions evaluate to TOP
            // or all evaluate to BOTTOM
            // check if first class expr. is BOTTOM
            boolean isBottom = bottomEvaluator.isBottomEquivalent(first, getSignature(), localityCls);
            // if not BOTTOM or not TOP then this axiom is non-local
            if (!isBottom && !topEvaluator.isTopEquivalent(first, getSignature(), localityCls)) {
                isLocal = false;
            }
            if (isBottom) {
                // unless we find a non-locality, process all the class
                // expressions
                while (isLocal && eqs.hasNext()) {
                    OWLClassExpression next = eqs.next();
                    assert next != null;
                    // first class expr. was BOTTOM, so this one should be
                    // BOTTOM too
                    if (!bottomEvaluator.isBottomEquivalent(next, getSignature(), localityCls)) {
                        isLocal = false;
                    }
                }
            } else {
                // unless we find a non-locality, process all the class
                // expressions
                while (isLocal && eqs.hasNext()) {
                    OWLClassExpression next = eqs.next();
                    assert next != null;
                    // first class expr. was TOP, so this one should be TOP too
                    if (!topEvaluator.isTopEquivalent(next, getSignature(), localityCls)) {
                        isLocal = false;
                    }
                }
            }
        }

        @Override
        public void visit(OWLEquivalentDataPropertiesAxiom axiom) {
            Collection eqs = axiom.getProperties();
            int size = eqs.size();
            if (size == 1) {
                isLocal = true;
            } else {
                for (OWLDataPropertyExpression p : eqs) {
                    if (getSignature().contains(p.asOWLDataProperty())) {
                        isLocal = false;
                        return;
                    }
                }
                isLocal = true;
            }
        }

        @Override
        public void visit(OWLEquivalentObjectPropertiesAxiom axiom) {
            Collection eqs = axiom.getProperties();
            int size = eqs.size();
            if (size == 1) {
                isLocal = true;
            } else {
                for (OWLObjectPropertyExpression p : eqs) {
                    if (getSignature().contains(p.getNamedProperty())) {
                        isLocal = false;
                        return;
                    }
                }
                isLocal = true;
            }
        }

        @Override
        public void visit(OWLFunctionalDataPropertyAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getProperty().asOWLDataProperty());
                break;
            case TOP_TOP:
                isLocal = false;
                break;
            default:
                break;
            }
        }

        // BUGFIX (TS): replaced call to asOWLObjectProperty() with
        // getNamedProperty()
        @Override
        public void visit(OWLFunctionalObjectPropertyAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getProperty().getNamedProperty());
                break;
            case TOP_TOP:
                isLocal = false;
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLInverseFunctionalObjectPropertyAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getProperty().getNamedProperty());
                break;
            case TOP_TOP:
                isLocal = false;
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLInverseObjectPropertiesAxiom axiom) {
            isLocal = !getSignature().contains(axiom.getFirstProperty().getNamedProperty()) && !getSignature().contains(
                axiom.getSecondProperty().getNamedProperty());
        }

        // BUGFIX: (TS) Irreflexive OP axioms are local in the *_BOTTOM case:
        // The empty object property is irreflexive!
        // BUGFIX: (DT) OP in signature makes the axiom non-local
        @Override
        public void visit(OWLIrreflexiveObjectPropertyAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getProperty().getNamedProperty());
                break;
            case TOP_TOP:
                isLocal = false;
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS) Added the case where this is local. (This is dual to the
        // case of a "positive" DP assertion.)
        @Override
        public void visit(OWLNegativeDataPropertyAssertionAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getProperty().asOWLDataProperty());
                break;
            case TOP_TOP:
                isLocal = false;
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS) Added the case where this is local. (This is dual to the
        // case of a "positive" OP assertion.)
        @Override
        public void visit(OWLNegativeObjectPropertyAssertionAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getProperty().getNamedProperty());
                break;
            case TOP_TOP:
                isLocal = false;
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectPropertyAssertionAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = false;
                break;
            case TOP_TOP:
                isLocal = !getSignature().contains(axiom.getProperty().getNamedProperty());
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS) Added the cases where this is local
        @Override
        public void visit(OWLSubPropertyChainOfAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                // Axiom is local iff at least one prop in the chain is
                // bot-equiv
                for (OWLObjectPropertyExpression ope : axiom.getPropertyChain()) {
                    if (!getSignature().contains(ope.getNamedProperty())) {
                        isLocal = true;
                        return;
                    }
                }
                isLocal = false;
                break;
            case TOP_TOP:
                // Axiom is local iff RHS is top-equiv
                if (!getSignature().contains(axiom.getSuperProperty().getNamedProperty())) {
                    isLocal = true;
                } else {
                    isLocal = false;
                }
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectPropertyDomainAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getProperty().getNamedProperty()) || topEvaluator
                    .isTopEquivalent(axiom.getDomain(), getSignature(), localityCls);
                break;
            case TOP_TOP:
                isLocal = topEvaluator.isTopEquivalent(axiom.getDomain(), getSignature(), localityCls);
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectPropertyRangeAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getProperty().getNamedProperty()) || topEvaluator
                    .isTopEquivalent(axiom.getRange(), getSignature(), localityCls);
                break;
            case TOP_TOP:
                isLocal = topEvaluator.isTopEquivalent(axiom.getRange(), getSignature(), localityCls);
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLSubObjectPropertyOfAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = !getSignature().contains(axiom.getSubProperty().getNamedProperty());
                break;
            case TOP_TOP:
                isLocal = !getSignature().contains(axiom.getSuperProperty().getNamedProperty());
                break;
            default:
                break;
            }
        }

        // BUGFIX: (DT) Bottom property is not reflexive
        @Override
        public void visit(OWLReflexiveObjectPropertyAxiom axiom) {
            switch (localityCls) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isLocal = false;
                break;
            case TOP_TOP:
                isLocal = !getSignature().contains(axiom.getProperty().getNamedProperty());
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS) Same individuals axioms are local, too.
        // They need to be added to the module after the locality checks have
        // been performed.
        @Override
        public void visit(OWLSameIndividualAxiom axiom) {
            isLocal = true;
        }

        @Override
        public void visit(OWLSubClassOfAxiom axiom) {
            isLocal = bottomEvaluator.isBottomEquivalent(axiom.getSubClass(), getSignature(), localityCls)
                || topEvaluator.isTopEquivalent(axiom.getSuperClass(), getSignature(), localityCls);
        }

        @Override
        public void visit(OWLSymmetricObjectPropertyAxiom axiom) {
            isLocal = !getSignature().contains(axiom.getProperty().getNamedProperty());
        }

        @Override
        public void visit(OWLTransitiveObjectPropertyAxiom axiom) {
            isLocal = !getSignature().contains(axiom.getProperty().getNamedProperty());
        }

        // TODO: (TS) Can't we treat this in a more differentiated way?
        @Override
        public void visit(SWRLRule rule) {
            isLocal = false;
        }

        @Override
        public void visit(OWLHasKeyAxiom axiom) {
            isLocal = true;
        }

        @Override
        public void visit(OWLAnnotationPropertyDomainAxiom axiom) {
            isLocal = true;
        }

        @Override
        public void visit(OWLAnnotationPropertyRangeAxiom axiom) {
            isLocal = true;
        }

        @Override
        public void visit(OWLSubAnnotationPropertyOfAxiom axiom) {
            isLocal = true;
        }
    }

    /**
     * Used to determine if class expressions are equivalent to \bottom using
     * the provided locality class.
     */
    private static class BottomEquivalenceEvaluator implements OWLClassExpressionVisitor {

        private boolean isBottomEquivalent;
        private LocalityClass localityCls;
        private Collection signature;
        private TopEquivalenceEvaluator topEvaluator;

        /** Instantiates a new bottom equivalence evaluator. */
        BottomEquivalenceEvaluator() {}

        /**
         * Checks if is bottom equivalent.
         * 
         * @param desc
         *        the desc
         * @return true, if is bottom equivalent
         */
        private boolean isBottomEquivalent(@Nonnull OWLClassExpression desc) {
            checkNotNull(desc, "desc cannot be null").accept(this);
            return isBottomEquivalent;
        }

        /**
         * Checks if is bottom equivalent.
         * 
         * @param desc
         *        the desc
         * @param sig
         *        the sig
         * @param locality
         *        the locality
         * @return true, if is bottom equivalent
         */
        public boolean isBottomEquivalent(@Nonnull OWLClassExpression desc,
            @Nonnull Collection sig, @Nonnull LocalityClass locality) {
            localityCls = checkNotNull(locality, "locality cannot be null");
            signature = checkNotNull(sig, "sig cannot be null");
            checkNotNull(desc, "desc cannot be null").accept(this);
            return isBottomEquivalent;
        }

        @Nonnull
        protected Collection getSignature() {
            return verifyNotNull(signature);
        }

        @Nonnull
        protected LocalityClass getLocality() {
            return verifyNotNull(localityCls);
        }

        /**
         * Sets the top evaluator.
         * 
         * @param evaluator
         *        the new top evaluator
         */
        public void setTopEvaluator(@Nonnull TopEquivalenceEvaluator evaluator) {
            topEvaluator = checkNotNull(evaluator, "evaluator cannot be null");
        }

        @Override
        public void visit(OWLClass ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
                isBottomEquivalent = ce.isOWLNothing() || !ce.isOWLThing() && !getSignature().contains(ce);
                break;
            case TOP_BOTTOM:
            case TOP_TOP:
                isBottomEquivalent = ce.isOWLNothing();
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS) Even in the TOP_TOP case, this is not bottom-equiv:
        // "forall top.D" is not necessarily empty
        // BUGFIX: (TS, 3): In the TOP_TOP case, there is a bottom-equiv
        // possibility.
        @Override
        public void visit(OWLDataAllValuesFrom ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = false;
                break;
            case TOP_TOP:
                isBottomEquivalent = !getSignature().contains(ce.getProperty().asOWLDataProperty()) && !ce.getFiller()
                    .isTopDatatype();
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS) Corrected both conditions; included case n==0
        // BUGFIX: (TS, 2) Added the cases where the filler is top-equiv
        // BUGFIX: (TS, 3) Repaired the cases where the filler is top-equiv
        @Override
        public void visit(OWLDataExactCardinality ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = ce.getCardinality() > 0 && !getSignature().contains(ce.getProperty()
                    .asOWLDataProperty());
                break;
            case TOP_TOP:
                isBottomEquivalent = ce.getCardinality() == 0 && !getSignature().contains(ce.getProperty()
                    .asOWLDataProperty()) && isTopOrBuiltInDatatype(ce.getFiller()) || ce.getCardinality() > 0
                        && !getSignature().contains(ce.getProperty().asOWLDataProperty())
                        && isTopOrBuiltInInfiniteDatatype(ce.getFiller());
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS) A data max card restriction is never bottom-equiv.
        // BUGFIX: (TS, 2) Added the cases where the filler is top-equiv
        // BUGFIX: (TS, 3) Repaired the cases where the filler is top-equiv
        @Override
        public void visit(OWLDataMaxCardinality ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = false;
                break;
            case TOP_TOP:
                isBottomEquivalent = ce.getCardinality() == 0 && !getSignature().contains(ce.getProperty()
                    .asOWLDataProperty()) && isTopOrBuiltInDatatype(ce.getFiller()) || ce.getCardinality() == 1
                        && !getSignature().contains(ce.getProperty().asOWLDataProperty()) && isTopOrBuiltInDatatype(ce
                            .getFiller()) || ce.getCardinality() > 1 && !getSignature().contains(ce.getProperty()
                                .asOWLDataProperty()) && isTopOrBuiltInInfiniteDatatype(ce.getFiller());
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS) The *_BOTTOM case only works if n > 0.
        @Override
        public void visit(OWLDataMinCardinality ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = ce.getCardinality() > 0 && !getSignature().contains(ce.getProperty()
                    .asOWLDataProperty());
                break;
            case TOP_TOP:
                isBottomEquivalent = false;
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLDataSomeValuesFrom ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = !getSignature().contains(ce.getProperty().asOWLDataProperty());
                break;
            case TOP_TOP:
                isBottomEquivalent = false;
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLDataHasValue ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = !getSignature().contains(ce.getProperty().asOWLDataProperty());
                break;
            case TOP_TOP:
                isBottomEquivalent = false;
                break;
            default:
                break;
            }
        }

        // BUGFIX (TS): TOP_TOP case was missing the first conjunct
        @Override
        public void visit(OWLObjectAllValuesFrom ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = false;
                break;
            case TOP_TOP:
                isBottomEquivalent = !getSignature().contains(ce.getProperty().getNamedProperty())
                    && isBottomEquivalent(ce.getFiller());
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectComplementOf ce) {
            isBottomEquivalent = topEvaluator.isTopEquivalent(ce.getOperand(), getSignature(), getLocality());
        }

        // BUGFIX: (TS) Since an exact card restriction is a conjunction of a
        // min and a max card restriction,
        // there are cases where it is bottom-local
        @Override
        public void visit(OWLObjectExactCardinality ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = ce.getCardinality() > 0 && (!getSignature().contains(ce.getProperty()
                    .getNamedProperty()) || isBottomEquivalent(ce.getFiller()));
                break;
            case TOP_TOP:
                isBottomEquivalent = ce.getCardinality() > 0 && (isBottomEquivalent(ce.getFiller()) || !getSignature()
                    .contains(ce.getProperty().getNamedProperty()) && topEvaluator.isTopEquivalent(ce.getFiller(),
                        getSignature(), getLocality()));
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectIntersectionOf ce) {
            for (OWLClassExpression conj : ce.getOperands()) {
                assert conj != null;
                if (isBottomEquivalent(conj)) {
                    isBottomEquivalent = true;
                    return;
                }
            }
            isBottomEquivalent = false;
        }

        // BUGFIX (TS): Corrected all conditions.
        // The n==0 case doesn't affect bottom-equivalence of this type of
        // restriction,
        // but n>0 does!
        @Override
        public void visit(OWLObjectMaxCardinality ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = false;
                break;
            case TOP_TOP:
                isBottomEquivalent = ce.getCardinality() > 0 && !getSignature().contains(ce.getProperty()
                    .getNamedProperty()) && topEvaluator.isTopEquivalent(ce.getFiller(), getSignature(), getLocality());
                break;
            default:
                break;
            }
        }

        // BUGFIX (TS): Corrected all conditions, considering the case n==0
        @Override
        public void visit(OWLObjectMinCardinality ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = ce.getCardinality() > 0 && (!getSignature().contains(ce.getProperty()
                    .getNamedProperty()) || isBottomEquivalent(ce.getFiller()));
                break;
            case TOP_TOP:
                isBottomEquivalent = ce.getCardinality() > 0 && isBottomEquivalent(ce.getFiller());
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectOneOf ce) {
            isBottomEquivalent = ce.getIndividuals().isEmpty();
        }

        @Override
        public void visit(OWLObjectHasSelf ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = !getSignature().contains(ce.getProperty().getNamedProperty());
                break;
            case TOP_TOP:
                isBottomEquivalent = false;
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectSomeValuesFrom ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = !getSignature().contains(ce.getProperty().getNamedProperty())
                    || isBottomEquivalent(ce.getFiller());
                break;
            case TOP_TOP:
                isBottomEquivalent = isBottomEquivalent(ce.getFiller());
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectUnionOf ce) {
            for (OWLClassExpression disj : ce.getOperands()) {
                assert disj != null;
                if (!isBottomEquivalent(disj)) {
                    isBottomEquivalent = false;
                    return;
                }
            }
            isBottomEquivalent = true;
        }

        // BUGFIX (TS): desc.getValue() is an individual and therefore is *not*
        // bot-equiv if not in the signature
        // -> disjunct removed from *_BOTTOM case
        @Override
        public void visit(OWLObjectHasValue ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isBottomEquivalent = !getSignature().contains(ce.getProperty().getNamedProperty());
                break;
            case TOP_TOP:
                isBottomEquivalent = false;
                break;
            default:
                break;
            }
        }
    }

    /**
     * Used to determine if class expressions are equivalent to \top using the
     * provided locality class.
     */
    private static class TopEquivalenceEvaluator implements OWLClassExpressionVisitor {

        private BottomEquivalenceEvaluator bottomEvaluator;
        private boolean isTopEquivalent;
        private LocalityClass localityCls;
        private Collection signature;

        /** Instantiates a new top equivalence evaluator. */
        TopEquivalenceEvaluator() {}

        private boolean isTopEquivalent(@Nonnull OWLClassExpression desc) {
            checkNotNull(desc, "desc cannot be null").accept(this);
            return isTopEquivalent;
        }

        @Nonnull
        protected Collection getSignature() {
            return verifyNotNull(signature);
        }

        @Nonnull
        protected LocalityClass getLocality() {
            return verifyNotNull(localityCls);
        }

        /**
         * Checks if is top equivalent.
         * 
         * @param desc
         *        the desc
         * @param sig
         *        the sig
         * @param locality
         *        the locality
         * @return true, if is top equivalent
         */
        public boolean isTopEquivalent(@Nonnull OWLClassExpression desc, @Nonnull Collection sig,
            @Nonnull LocalityClass locality) {
            localityCls = checkNotNull(locality, "locality cannot be null");
            signature = checkNotNull(sig, "sig cannot be null");
            checkNotNull(desc, "desc cannot be null").accept(this);
            return isTopEquivalent;
        }

        /**
         * Sets the bottom evaluator.
         * 
         * @param evaluator
         *        the new bottom evaluator
         */
        public void setBottomEvaluator(@Nonnull BottomEquivalenceEvaluator evaluator) {
            bottomEvaluator = checkNotNull(evaluator, "evaluator cannot be null");
        }

        @Override
        public void visit(OWLClass ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
                isTopEquivalent = ce.isOWLThing();
                break;
            case TOP_BOTTOM:
            case TOP_TOP:
                isTopEquivalent = ce.isOWLThing() || !ce.isOWLNothing() && !signature.contains(ce);
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS, 2) Added the cases where the filler is top-equiv
        @Override
        public void visit(OWLDataAllValuesFrom ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = !signature.contains(ce.getProperty().asOWLDataProperty()) || ce.getFiller()
                    .isTopDatatype();
                break;
            case TOP_TOP:
                isTopEquivalent = ce.getFiller().isTopDatatype();
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS) Added the case where this is top-equiv (including n==0).
        @Override
        public void visit(OWLDataExactCardinality ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = ce.getCardinality() == 0 && !signature.contains(ce.getProperty().asOWLDataProperty());
                break;
            case TOP_TOP:
                isTopEquivalent = false;
                break;
            default:
                break;
            }
        }

        // (TS) No special handling for n==0 required.
        @Override
        public void visit(OWLDataMaxCardinality ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = !signature.contains(ce.getProperty().asOWLDataProperty());
                break;
            case TOP_TOP:
                isTopEquivalent = false;
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS) A data min card restriction is top-equiv iff the
        // cardinality is 0.
        // BUGFIX: (TS, 2) Added the cases where the filler is top-equiv
        // BUGFIX: (TS, 2) Left out redundant check cardinality > 0 in TOP_TOP
        // case
        // BUGFIX: (TS, 3) Extended the cases where the filler is top-equiv in
        // TOP_TOP
        @Override
        public void visit(OWLDataMinCardinality ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = ce.getCardinality() == 0;
                break;
            case TOP_TOP:
                isTopEquivalent = ce.getCardinality() == 0 || ce.getCardinality() == 1 && !signature.contains(ce
                    .getProperty().asOWLDataProperty()) && isTopOrBuiltInDatatype(ce.getFiller()) || ce
                        .getCardinality() > 1 && !signature.contains(ce.getProperty().asOWLDataProperty())
                        && isTopOrBuiltInInfiniteDatatype(ce.getFiller());
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS, 2) Added the cases where the filler is top-equiv
        // BUGFIX: (TS, 3) Extended the cases where the filler is top-equiv
        @Override
        public void visit(OWLDataSomeValuesFrom ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = false;
                break;
            case TOP_TOP:
                isTopEquivalent = !signature.contains(ce.getProperty().asOWLDataProperty()) && isTopOrBuiltInDatatype(ce
                    .getFiller());
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS, 2) Added the cases where this is top-equiv
        @Override
        public void visit(OWLDataHasValue ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = false;
                break;
            case TOP_TOP:
                isTopEquivalent = !signature.contains(ce.getProperty().asOWLDataProperty());
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectAllValuesFrom ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = !signature.contains(ce.getProperty().getNamedProperty()) || isTopEquivalent(ce
                    .getFiller());
                break;
            case TOP_TOP:
                isTopEquivalent = isTopEquivalent(ce.getFiller());
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectComplementOf ce) {
            isTopEquivalent = bottomEvaluator.isBottomEquivalent(ce.getOperand(), getSignature(), getLocality());
        }

        // BUGFIX: (TS) added the cases where this is top-equiv, including n==0
        @Override
        public void visit(OWLObjectExactCardinality ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = ce.getCardinality() == 0 && (!signature.contains(ce.getProperty().getNamedProperty())
                    || bottomEvaluator.isBottomEquivalent(ce.getFiller(), getSignature(), getLocality()));
                break;
            case TOP_TOP:
                isTopEquivalent = ce.getCardinality() == 0 && bottomEvaluator.isBottomEquivalent(ce.getFiller(),
                    getSignature(), getLocality());
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectIntersectionOf ce) {
            for (OWLClassExpression conj : ce.getOperands()) {
                assert conj != null;
                if (!isTopEquivalent(conj)) {
                    isTopEquivalent = false;
                    return;
                }
            }
            isTopEquivalent = true;
        }

        // BUGFIX: (TS) Added the case of a bottom-equivalent filler to both
        // conditions.
        // The n==0 case doesn't affect top-equivalence of this type of
        // restriction.
        @Override
        public void visit(OWLObjectMaxCardinality ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = !signature.contains(ce.getProperty().getNamedProperty()) || bottomEvaluator
                    .isBottomEquivalent(ce.getFiller(), getSignature(), getLocality());
                break;
            case TOP_TOP:
                isTopEquivalent = bottomEvaluator.isBottomEquivalent(ce.getFiller(), getSignature(), getLocality());
                break;
            default:
                break;
            }
        }

        // BUGFIX: (TS) Added the case n==0; repaired TOP_TOP condition
        // BUGFIX: (TS, 2) Left out redundant check cardinality > 0 in TOP_TOP
        // case
        @Override
        public void visit(OWLObjectMinCardinality ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = ce.getCardinality() == 0;
                break;
            case TOP_TOP:
                // isTopEquivalent =
                // !signature.contains(desc.getProperty().getNamedProperty())
                // && (desc.getCardinality() <= 1);
                isTopEquivalent = ce.getCardinality() == 0 || !signature.contains(ce.getProperty().getNamedProperty())
                    && isTopEquivalent(ce.getFiller());
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectOneOf ce) {
            isTopEquivalent = false;
        }

        @Override
        public void visit(OWLObjectHasSelf ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = false;
                break;
            case TOP_TOP:
                isTopEquivalent = !signature.contains(ce.getProperty().getNamedProperty());
                break;
            default:
                break;
            }
        }

        // BUGFIX (TS): added ".getNamedProperty()"
        @Override
        public void visit(OWLObjectSomeValuesFrom ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = false;
                break;
            case TOP_TOP:
                isTopEquivalent = !signature.contains(ce.getProperty().getNamedProperty()) && isTopEquivalent(ce
                    .getFiller());
                break;
            default:
                break;
            }
        }

        @Override
        public void visit(OWLObjectUnionOf ce) {
            for (OWLClassExpression conj : ce.getOperands()) {
                assert conj != null;
                if (isTopEquivalent(conj)) {
                    isTopEquivalent = true;
                    return;
                }
            }
            isTopEquivalent = false;
        }

        @Override
        public void visit(OWLObjectHasValue ce) {
            switch (getLocality()) {
            case BOTTOM_BOTTOM:
            case TOP_BOTTOM:
                isTopEquivalent = false;
                break;
            case TOP_TOP:
                isTopEquivalent = !signature.contains(ce.getProperty().getNamedProperty());
                break;
            default:
                break;
            }
        }
    }

    @Override
    public boolean isLocal(OWLAxiom axiom, Set signature) {
        return axiomVisitor.isLocal(axiom, signature);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy