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

org.semanticweb.owlapi.modularity.AtomicDecomposition Maven / Gradle / Ivy

/* This file is part of the OWL API.
 * The contents of this file are subject to the LGPL License, Version 3.0.
 * Copyright 2020, Marc Robin Nolte
 * 
 * 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 org.semanticweb.owlapi.modularity;

import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.function.Function;
import java.util.stream.Collectors;
import java.util.stream.Stream;

import javax.annotation.Nonnull;

import org.semanticweb.owlapi.model.HasAxioms;
import org.semanticweb.owlapi.model.HasContainsAxiom;
import org.semanticweb.owlapi.model.OWLAxiom;
import org.semanticweb.owlapi.model.OWLOntology;
import org.semanticweb.owlapi.model.parameters.Imports;
import org.semanticweb.owlapi.modularity.locality.LocalityClass;
import org.semanticweb.owlapi.modularity.locality.LocalityModuleExtractor;
import org.semanticweb.owlapi.modularity.locality.SyntacticLocalityModuleExtractor;

/**
 * Class to represent the atomic decomposition of a set of axioms.
 *
 * @author Marc Robin Nolte
 */
public final class AtomicDecomposition implements HasAxioms {

    /**
     * Class to represent atoms.
     *
     * @author Marc Robin Nolte
     */
    public final class Atom implements HasAxioms, HasContainsAxiom {

        /**
         * The {@link OWLAxiom}s of this {@link Atom}.
         */
        protected final @Nonnull Set axiomSet = new HashSet<>();

        /**
         * The {@link Atom}s this Atom depends on (excluding this atom), that is, all atoms (A) with
         * (this) > (A).
         */
        protected final @Nonnull Set dependencies = new HashSet<>();

        /**
         * The {@link Atom}s that depend on this {@link Atom} (excluding this atom), that is, all
         * atoms (A) with (A) > (this).
         */
        protected final @Nonnull Set dependents = new HashSet<>();

        /**
         * Instantiates a new {@link Atom}. The {@link OWLAxiom}s of this {@link Atom} are directly
         * manipulated within {@link AtomicDecomposition#decompose()}.
         */
        protected Atom() {}

        @Override
        public @Nonnull Stream axioms() {
            return axiomSet.stream();
        }

        @Override
        public boolean containsAxiom(OWLAxiom axiom) {
            return axiomSet.contains(axiom);
        }

        /**
         * Returns the {@link Atom}s this {@link Atom} (directly or indirectly) depends on,
         * excluding this atom. In the literature, {@code all atoms (A) with (this) > (A)}.
         *
         * @return The {@link Atom}s this {@link Atom} depends on, excluding this atom
         */
        public @Nonnull Stream dependencies() {
            return dependencies.stream();
        }

        /**
         * Returns the {@link Atom}s that (directly or indirectly) depend on this {@link Atom},
         * excluding this atom. In the literature, {@code all atoms (A) with (A) > (this)}.
         *
         * @return The {@link Atom}s that depend on this {@link Atom}, excluding this atom
         */
        public @Nonnull Stream dependents() {
            return dependents.stream();
        }

        /**
         * Checks if this {@link Atom} (directly or indirectly) depends on the given {@link Atom}.
         * In the literature, {@code if (this) > (given)}. Note that this returns false, if the
         * given atom is @{code ==} to this atom.
         *
         * @param atom The {@link Atom} to check if this {@link Atom} depends on it. Cannot be null.
         * @return If this {@link Atom} depends on the given {@link Atom}
         * @throws NullPointerException If the given {@link Atom} was {@code null}
         */
        public boolean dependsOn(Atom atom) {
            return dependencies.contains(Objects.requireNonNull(atom));
        }

        /**
         * Returns the {@link Atom}s this {@link Atom} DIRECTLY depends on, excluding this atom. In
         * the literature,
         * {@code all atoms (A) with (this) > (A), such there is no other atom (B) with
         * (this) > (B) > (A)}. This method is equal to calling
         * {@code thisAtom.dependencies(thisAtom::directlyDependsOn)} and on each call is in
         * O(this.getAxiomCount()^2) .
         *
         * @return The {@link Atom}s this {@link Atom} DIRECTLY depends on, excluding this atom
         */
        public @Nonnull Stream directDependencies() {
            return dependencies().filter(this::directlyDependsOn);
        }

        /**
         * Returns the {@link Atom}s that DIRECTLY depend on this {@link Atom}, excluding this atom.
         * In the literature,
         * {@code all atoms (A) with (A) > (this), such there is no other atom (B)
         * with (A) > (B) > (this)}. This method is equal to calling
         * {@code thisAtom.dependencies(thisAtom::isDirectDependencyOf)} and on each call is in
         * O(this.getAxiomCount()^2) .
         *
         * @return The {@link Atom}s this {@link Atom} DIRECTLY depends on, excluding this atom
         */
        public @Nonnull Stream directDependents() {
            return dependents().filter(this::isDirectDependencyOf);
        }

        /**
         * Checks if this {@link Atom} DIRECTLY depends on the given one. In the literature,
         * {@code if
         * (this) > (given) and there is no other atom (B) with (this) > (B) > (given)}. On
         * each call this method is in O(this.getAxiomCount()). Note that this returns false, if the
         * given atom is @{code ==} to this atom.
         *
         * @param atom The {@link Atom} to check if this one directly depends on it
         * @return If this {@link Atom} DIRECTLY depends on the given one
         * @throws NullPointerException if the given {@link Atom} was {@code null}
         */
        public boolean directlyDependsOn(Atom atom) {
            return dependsOn(atom) && dependencies().noneMatch(next -> dependsOn(atom));
        }

        /**
         * Returns the number of axioms within this {@link Atom}.
         *
         * @return The number of axioms within this {@link Atom}
         */
        public int getAxiomCount() {
            return axiomSet.size();
        }

        /**
         * Checks if the given {@link Atom} (directly or indirectly) depends on this {@link Atom}.
         * In the literature, {@code if (given) > (this)}. Note that this returns false, if the
         * given atom is @{code ==} to this atom.
         *
         * @param atom The {@link Atom} to check if it depends on this {@link Atom}. Cannot be null.
         * @return If the given {@link Atom} depends on the this {@link Atom}
         * @throws NullPointerException If the given {@link Atom} was {@code null}
         */
        public boolean isDependencyOf(Atom atom) {
            return dependents.contains(Objects.requireNonNull(atom));
        }

        /**
         * Checks if the given {@link Atom} DIRECTLY depends on this {@link Atom}. In the
         * literature,
         * {@code if (given) > (this) and there is no other atom (B) with (given) > (B)
         * > (this)}. On each call this method is in O(this.getAxiomCount()). Note that this
         * returns false, if the given atom is @{code ==} to this atom.
         *
         * @param atom The atom to check if it directly depends on this one
         * @return If the given {@link Atom} DIRECTLY depends on this one
         * @throws NullPointerException if the given {@link Atom} was {@code null}
         */
        public boolean isDirectDependencyOf(Atom atom) {
            return isDependencyOf(atom) && dependents().noneMatch(next -> isDependencyOf(atom));
        }
    }

    /**
     * A mapping from {@link OWLAxiom}s to their respective {@link Atom}. Gets computed within
     * {@link AtomicDecomposition#decompose()}.
     */
    private final @Nonnull Map atomOf;

    /**
     * The axiom base of this {@link AtomicDecomposition}.
     */
    private final @Nonnull Set axioms;

    /**
     * The {@link ModuleExtractor} used by this {@link AtomicDecomposition} to
     */
    private final @Nonnull ModuleExtractor moduleExtractor;

    /**
     * A mapping from {@link OWLAxiom}s to the modules of the axiom base of this
     * {@link AtomicDecomposition}. Gets computed within {@link AtomicDecomposition#decompose()}
     * when calling {@link AtomicDecomposition#moduleToSignatureOf(OWLAxiom)}.
     */
    private final @Nonnull Map> moduleToSignatureOf;

    /**
     * Constructs the {@link AtomicDecomposition} for the given axiom base based on the module
     * extraction an {@link SyntacticLocalityModuleExtractor} provides by using
     * {@link LocalityClass#STAR}. This operation may take some time based on the ontology size and
     * the used module extraction algorithm.
     *
     * @param ontology The {@link OWLOntology} that is to be decomposed. Changes of the ontology
     *                 will not be considered
     */
    public AtomicDecomposition(OWLOntology ontology) {
        this(ontology.axioms(Imports.INCLUDED));
    }

    /**
     * Constructs the {@link AtomicDecomposition} for the given axiom base based on the module
     * extraction the {@link ModuleExtractor} the given {@link Function} supplies. This operation
     * may take some time based on the ontology size and the used module extraction algorithm.
     *
     * @param ontology The {@link OWLOntology} that is to be decomposed. Changes of the ontology
     *                 will not be considered
     * @param function The function used to build a {@link ModuleExtractor}. We recommend performing
     *                 possible precalculations by the module extractor
     */
    public AtomicDecomposition(OWLOntology ontology,
        Function, ? extends ModuleExtractor> function) {
        this(Objects.requireNonNull(ontology, "The given ontology may not be null").axioms(),
            function);
    }

    /**
     * Constructs the {@link AtomicDecomposition} for the given axiom base based on the module
     * extraction an {@link SyntacticLocalityModuleExtractor} provides by using
     * {@link LocalityClass#STAR}. This operation may take some time based on the ontology size and
     * the used module extraction algorithm.
     *
     * @param axioms The axiom base that is to be decomposed
     */
    public AtomicDecomposition(Stream axioms) {
        this(axioms, stream -> new SyntacticLocalityModuleExtractor(LocalityClass.STAR, stream));
    }

    /**
     * Constructs the {@link AtomicDecomposition} for the given axiom base based on the module
     * extraction the {@link ModuleExtractor} the given {@link Function} supplies. It must be
     * ensured that it satisfies the properties required to be used for atomic decomposition, see
     * "Modular Structures and Atomic Decomposition in Ontologies" by Del Vescovo et al, 2019. For
     * example, one such module extraction algorithm is provided by the subclasses of
     * {@link LocalityModuleExtractor}s.
     * 
     * This operation may take some time based on the ontology size and the used module extraction
     * algorithm.
     *
     * @param axioms   The axiom base that is to be decomposed
     * @param function The function used to build a {@link ModuleExtractor}. We recommend performing
     *                 possible precalculations by the module extractor
     */
    public AtomicDecomposition(Stream axioms,
        Function, ? extends ModuleExtractor> function) {
        this.axioms = Objects.requireNonNull(axioms, "The given axioms may not be null")
            .collect(Collectors.toSet());
        moduleExtractor =
            Objects.requireNonNull(function, "The given function may not be null").apply(axioms());
        Objects.requireNonNull(axioms, "The given function may not retrieve null");
        moduleToSignatureOf = new HashMap<>();
        atomOf = new HashMap<>();
        decompose();
    }

    /**
     * Returns the {@link Atom} associated with the given {@link OWLAxiom}. Those {@link Atom}s are
     * computed when constructing the {@link AtomicDecomposition}.
     *
     * @param axiom The {@link OWLAxiom} whose {@link Atom} should be returned
     * @return The {@link Atom} as specified above
     * @throws IllegalArgumentException If the given {@link OWLAxiom} is not part of the axiom base
     *                                  of this {@link AtomicDecomposition}
     */
    public @Nonnull Atom atomOf(OWLAxiom axiom) {
        if (!containsAxiom(Objects.requireNonNull(axiom, "The given axiom may not be null"))) {
            throw new IllegalArgumentException(
                "The given axiom must be contained in the axiom base of this atomic decomposition");
        }
        return atomOf.get(axiom);
    }

    /**
     * Returns a {@link Stream} of the {@link Atom}s computed by this {@link AtomicDecomposition}.
     *
     * @return The {@link Atom}s computed by this {@link AtomicDecomposition}
     */
    public Stream atoms() {
        return atomOf.values().stream().distinct();
    }

    @Override
    public @Nonnull Stream axioms() {
        return axioms.stream();
    }

    /**
     * Part of the implementation of the improved algorithm for atomic decomposition. See "Improved
     * Algorithms for Module Extraction and Atomic Decomposition" by Dmitry Tsarkov, 2012
     * 
     * @param alpha alpha axiom
     * @param beta  beta axiom
     * @return seed for alpha
     */
    private @Nonnull Optional buildAtomsInModule(OWLAxiom alpha,
        Optional beta) {

        // The atom for alpha is already known
        if (atomOf.containsKey(alpha)) {
            return Optional.of(alpha);
        }
        Optional delta = getAtomSeed(alpha, beta);
        Atom atomOfDelta = atomOf.computeIfAbsent(delta.orElse(null), d -> new Atom());
        atomOfDelta.axiomSet.add(alpha);
        atomOf.put(alpha, atomOfDelta);
        if (delta.equals(beta)) {
            return beta;
        }

        moduleToSignatureOf(alpha).filter(gamma -> !gamma.equals(alpha))
            .forEach(gamma -> updateDependencies(alpha, gamma));
        return Optional.of(alpha);
    }

    protected void updateDependencies(OWLAxiom alpha, OWLAxiom gamma) {
        Optional axiom = buildAtomsInModule(gamma, Optional.of(alpha));
        Atom first = atomOf.get(axiom.get());
        Atom second = atomOf.get(alpha);
        if (first != second) {
            second.dependencies.add(first); // second > first
            first.dependents.add(second);
        }
    }

    /**
     * Returns whether or not the given {@link OWLAxiom} is part of the axiom base of this
     * {@link AtomicDecomposition}. Runs in O(1).
     *
     * @param axiom The {@link OWLAxiom} to check
     * @return Whether or not the given {@link OWLAxiom} is part of the axiom base of this
     *         {@link AtomicDecomposition}
     */
    public boolean containsAxiom(OWLAxiom axiom) {
        return axioms.contains(axiom);
    }

    /**
     * Implementation of the improved algorithm for atomic decomposition. See "Improved Algorithms
     * for Module Extraction and Atomic Decomposition" by Dmitry Tsarkov, 2012.
     */
    private void decompose() {
        for (OWLAxiom alpha : axioms) {
            buildAtomsInModule(alpha, Optional.empty());
        }
    }

    /**
     * Part of the implementation of the improved algorithm for atomic decomposition. See "Improved
     * Algorithms for Module Extraction and Atomic Decomposition" by Dmitry Tsarkov, 2012
     * 
     * @param alpha alpha axiom
     * @param beta  beta axiom
     * @return seed for alpha
     */
    private @Nonnull Optional getAtomSeed(OWLAxiom alpha, Optional beta) {

        Set moduleOfBeta;
        if (beta.isPresent()) {
            moduleOfBeta = moduleToSignatureOf.get(beta.get());
            moduleExtractor.extract(alpha.signature(), moduleOfBeta::contains)
                .forEach(moduleAxiom -> moduleToSignatureOf
                    .computeIfAbsent(alpha, x -> new HashSet<>()).add(moduleAxiom));
        } else {
            moduleOfBeta = axioms;
            moduleExtractor.extract(alpha.signature()).forEach(moduleAxiom -> moduleToSignatureOf
                .computeIfAbsent(alpha, x -> new HashSet<>()).add(moduleAxiom));
        }

        // modules are equal if size are equal
        if (moduleToSignatureOf.get(alpha).size() == moduleOfBeta.size()) {
            return beta;
        }
        return Optional.of(alpha);
    }

    /**
     * Returns the module within the {@link AtomicDecomposition#axioms()} with respect to the
     * signature of the given axiom. Those modules are computed when constructing the
     * {@link AtomicDecomposition}.
     *
     * @param axiom The axiom for whose signature the precomputed module should be returned
     * @return The module as specified above
     * @throws IllegalArgumentException If the given axiom is not part of the axiom base of this
     *                                  {@link AtomicDecomposition}
     */
    public @Nonnull Stream moduleToSignatureOf(OWLAxiom axiom) {
        if (!containsAxiom(Objects.requireNonNull(axiom, "The given axiom may not be null"))) {
            throw new IllegalArgumentException(
                "The given axiom must be contained in the axiom base of this atomic decomposition");
        }
        return moduleToSignatureOf.get(axiom).stream();
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy