-
Notifications
You must be signed in to change notification settings - Fork 29
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Added documentation on semsin, and additional command for handling IC
This PR also bundles in a new OntologyGenerator interface
- Loading branch information
Showing
16 changed files
with
760 additions
and
60 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,193 @@ | ||
.. _similarity: | ||
|
||
Computing Similarity between Entities | ||
==================================== | ||
|
||
Background | ||
---------- | ||
|
||
A common use case for ontologies is to compute :term:`Similarity` between :term:`Entities <Entity>`, | ||
or between :term:`Terms <Term>`, based on properties of the ontology. For example, | ||
when comparing two terms, we may want to score terms that are closer together in the ontology | ||
:term:`Graph` as more similar. Other measures of similarity may be on frequency of usage of terms, | ||
or on vector :term:`Embeddings <Embedding>` of terms. | ||
|
||
If :term:`Entities <Entity>` are annotated/associated with terms, we can also compute similarity | ||
between these based on the aggregate set of terms they are annotated with. The canonical use | ||
case for OAK here is :term:`Gene` associations (e.g. to phenotypes, GO terms, anatomy or cell | ||
types for expression, etc), although any entities can be compared if ontology associations exist | ||
(e.g. comparing two people basic on shared music genres or favorite foods). | ||
|
||
OAK provides various methods for computing similarity of terms and entities. These are currently | ||
focused on "classic" semantic similarity measures, such as Jaccard similarity and information-content | ||
based measures, but this may be extended in the future to support newer embedding-based methods. | ||
|
||
Like all aspects of OAK, there is a separation between :term:`Interface` and :term:`Implementation`. | ||
The :ref:`semantic similarity interface <semantic_similarity_interface>` defines the API for computing similarity | ||
and related metrics. In theory there can be many implementations of this interface, in practice there | ||
is a default implementation and the semsimian implementation. See later in this document on for details. | ||
|
||
Concepts | ||
-------- | ||
|
||
Jaccard Similarity | ||
^^^^^^^^^^^^^^^^^^^ | ||
|
||
Jaccard similarity (`Wikipedia <https://en.wikipedia.org/wiki/Jaccard_index>`_) is a measure of similarity | ||
between two sets: | ||
|
||
.. math:: | ||
J(A,B) = \frac{|A \cap B|}{|A \cup B|} | ||
where :math:`A` and :math:`B` are sets. | ||
|
||
When applied to pairs of ontology terms, the sets are typically the :term:`Reflexive` :term:`Ancestors<Ancestor>` | ||
of the terms of interest. Like all OAK graph operations, the choice of :term:`Predicates<Predicate>` used is | ||
important. | ||
|
||
- Phenotype ontologies typically compute this using :term:`SubClassOf` links only | ||
- Ontologies such as GO, Uberon, and ENVO typically use :term:`SubClassOf` and :term:`PartOf` links | ||
|
||
Information Content | ||
^^^^^^^^^^^^^^^^^^^ | ||
|
||
Information Content (IC) is a measure of the specificity of a term. It is typically computed as the log of | ||
the inverse of the frequency of the term in a corpus: | ||
|
||
.. math:: | ||
IC(t) = -log(P(t)) | ||
where :math:`P(t)` is the probability of the term in the corpus. There are two ways to compute this: | ||
|
||
- Using the ontology as the corpus | ||
- Using :term:`Associations <Association>` between entities and terms as the corpus | ||
|
||
In both cases the ontology graph is used. | ||
|
||
When the ontology is used as the corpus, the probability of a term is computed as the number of | ||
descendants of a term (reflexive) divided by the total number of terms in the ontology: | ||
|
||
.. math:: | ||
P(t) = \frac{|Desc*(t)|}{|T|} | ||
When associations are used, the probability of a term is computed as the number of entities | ||
associated with a term (directly or indirectly) divided by the total number of entities: | ||
|
||
.. math:: | ||
P(t) = \frac{|E(t)|}{|E|} | ||
where :math:`E(t)` is the set of entities associated with :math:`t` and :math:`E` is the set of all entities. | ||
|
||
To build a table of IC for all terms in HPO, using the ontology as the corpus: | ||
|
||
.. code-block:: bash | ||
runoak -i sqlite:obo:hp information-content -p i i^HP: | ||
To include associations, use the ``--use-associations`` flag, in addition to | ||
specifying the associations themselves: | ||
|
||
.. code-block:: bash | ||
runoak -g phenotype.hpoa -G hpoa -i sqlite:obo:hp information-content -p i --use-associations i^HP: | ||
IC can be used to score the :term:`Most Recent Common Ancestor` (MRCAs) of two terms; also known | ||
as Resnik similarity. | ||
|
||
Aggregate measures for comparing entities | ||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ||
|
||
Computing similarity between terms (classes) is relatively easy. But in general users want to compare | ||
*entities* such as genes. Here we broadly have two approaches: | ||
|
||
- Knowledge Graph (KG): treat the entities as nodes in the graph and use the standard ontology methods above | ||
- Aggregate Statistics | ||
|
||
With the KG approach the entities typically become leaf nodes in the graph, and we compare using standard | ||
graph methods where both association predicates (e.g. has-phenotype) and ontology predicates (e.g. is-a and part-of) | ||
are used. | ||
|
||
The more common approach is to use aggregate statistics, which better accounts for incomplete annotation and | ||
annotation bias. Here we have two entities to compare e1 and e2, where each entity is the subject | ||
of association edges pointing to terms (sometimes called a *profile*). The goal is to compare the profiles | ||
using aggregate statistics. The aggregate statistics can be performed over any individual pairwise term | ||
statistics. | ||
|
||
The simplest approach is to take the maximum pairwise similarity between terms in the two profiles; | ||
for the IC of the MRCA this would be: | ||
|
||
.. math:: | ||
max_{t1 \in e1, t2 \in e2} IC(MRCA(t1,t2)) | ||
However, this doesn't take into account all the other associations for each entity. | ||
|
||
A more common aggregate statistic is Best Match Average (BMA), which is the average of the maximum | ||
pairwise similarity between each term in the first profile and the second profile: | ||
|
||
.. math:: | ||
BMA(e1,e2) = \frac{1}{|e1|} \sum_{t1 \in e1} max_{t2 \in e2} IC(MRCA(t1,t2)) | ||
Currently this is the default used in OAK. | ||
|
||
To compare two profiles in OAK you can use the ``termset-similarity`` command, passing in each | ||
profile, separated by ``@``: | ||
|
||
.. code-block:: bash | ||
runoak -i sqlite:obo:mp termset-similarity \ | ||
MP:0010771 MP:0002169 MP:0005391 MP:0005389 MP:0005367 @\ | ||
MP:0010771 MP:0002169 MP:0005391 MP:0005389 MP:0005367 | ||
(note that like all OAK commands, any expression can be substituted as arguments, e.g. ``.idfile`` to | ||
load the IDs from a file) | ||
|
||
The result conforms to `TermSetPairwiseSimilarity <https://w3id.org/linkml/similarity/TermSetPairwiseSimilarity>`_ | ||
|
||
This assumes that you have already done the lookup using an association table. | ||
|
||
Note that in the above, the IC scores are calculated using only the ontology as corpus. | ||
You can pass in a pre-generated IC table (e.g. if you computed this using a particular association database) | ||
using the ``--information-content-file`` option. | ||
|
||
Implementations | ||
--------------- | ||
|
||
Default Implementation | ||
^^^^^^^^^^^^^^^^^^^^^ | ||
|
||
Currently the default implementation in OAK is in pure python, and may be slow for | ||
large ontologies or association databases. | ||
|
||
Semsimian | ||
^^^^^^^^^ | ||
|
||
`Semsimian <https://github.com/monarch-initiative/semsimian>`_ is a Rust implementation of | ||
semantic similarity. OAK is able to use this as an implementation, *wrapping* an existing | ||
implementation. | ||
|
||
To wrap an existing adapter using semsimian, prefix the selector with ``semsimian:`` | ||
|
||
For example: | ||
|
||
.. code-block:: bash | ||
runoak -i semsimian:sqlite:obo:hp similarity .all @ .all -O csv | ||
Note that semsimian is under active development and performance characteristics may change. | ||
|
||
Data Model | ||
~~~~~~~~~~ | ||
|
||
See the `Similarity data model <https://w3id.org/oak/similarity/>`_ for details of the data model. | ||
|
||
Further reading | ||
--------------- | ||
|
||
- Lord et al, Semantic similarity measures as tools for exploring the gene ontology https://pubmed.ncbi.nlm.nih.gov/12603061/ | ||
- Koehler et al, Clinical diagnostics in human genetics with semantic similarity searches in ontologies, https://pubmed.ncbi.nlm.nih.gov/19800049/ |
Oops, something went wrong.