Source code for oc_ocdm.abstract_entity

#!/usr/bin/python
# -*- coding: utf-8 -*-
# Copyright (c) 2016, Silvio Peroni <essepuntato@gmail.com>
#
# Permission to use, copy, modify, and/or distribute this software for any purpose
# with or without fee is hereby granted, provided that the above copyright notice
# and this permission notice appear in all copies.
#
# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
# REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
# FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT,
# OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
# DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
# ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
# SOFTWARE.
from __future__ import annotations

from abc import ABC
from typing import TYPE_CHECKING

from oc_ocdm.support.support import create_type, create_literal, get_short_name, is_dataset, is_string_empty
from rdflib import URIRef, RDFS, RDF, Literal, Graph

if TYPE_CHECKING:
    from typing import Optional, List, ClassVar, Dict, Tuple, Iterable
    from rdflib import term


[docs]class AbstractEntity(ABC): """ Abstract class which represents a generic entity from the OCDM. It sits at the top of the entity class hierarchy. """ short_name_to_type_iri: ClassVar[Dict[str, URIRef]] = {} def __init__(self) -> None: """ Constructor of the ``AbstractEntity`` class. """ self.g: Graph = Graph() self.res: URIRef = URIRef("") self.short_name: str = ""
[docs] def remove_every_triple(self) -> None: """ Remover method that removes every triple from the current entity. **WARNING: the OCDM specification requires that every entity has at least one triple that defines its type (through the** ``rdf:type`` **RDF predicate). If such triple is not subsequently restored by the user, the entity will be considered as to be deleted since it wouldn't be valid anymore.** :return: None """ self.g.remove((None, None, None))
# LABEL
[docs] def get_label(self) -> Optional[str]: """ Getter method corresponding to the ``rdfs:label`` RDF predicate. :return: The requested value if found, None otherwise """ return self._get_literal(RDFS.label)
[docs] def create_label(self, string: str) -> None: """ Setter method corresponding to the ``rdfs:label`` RDF predicate. **WARNING: this is a functional property, hence any existing value will be overwritten!** :param string: The value that will be set as the object of the property related to this method :type string: str :return: None """ self.remove_label() self._create_literal(RDFS.label, string)
[docs] def remove_label(self) -> None: """ Remover method corresponding to the ``rdfs:label`` RDF predicate. :return: None """ self.g.remove((self.res, RDFS.label, None))
def _create_literal(self, p: URIRef, s: str, dt: URIRef = None, nor: bool = True) -> None: """ Adds an RDF triple with a literal object inside the graph of the entity :param p: The predicate :type p: URIRef :param s: The string to add as a literal value :type s: str :param dt: The object's datatype, if present :type dt: URIRef, optional :param nor: Whether to normalize the graph or not :type nor: bool, optional :return: None """ create_literal(self.g, self.res, p, s, dt, nor) # TYPE
[docs] def get_types(self) -> List[URIRef]: """ Getter method corresponding to the ``rdf:type`` RDF predicate. :return: A list containing the requested values if found, None otherwise """ uri_list: List[URIRef] = self._get_multiple_uri_references(RDF.type) return uri_list
def _create_type(self, res_type: URIRef) -> None: """ Setter method corresponding to the ``rdf:type`` RDF predicate. **WARNING: the OCDM specification admits at most two types for an entity. The main type cannot be edited or removed. Any existing secondary type will be overwritten!** :param res_type: The value that will be set as the object of the property related to this method :type res_type: URIRef :return: None """ self.remove_type() # <-- It doesn't remove the main type! create_type(self.g, self.res, res_type)
[docs] def remove_type(self) -> None: """ Remover method corresponding to the ``rdf:type`` RDF predicate. **WARNING: the OCDM specification requires at least one type for an entity. This method removes any existing secondary type, without removing the main type.** :return: None """ self.g.remove((self.res, RDF.type, None)) # Restore the main type IRI iri_main_type: URIRef = self.short_name_to_type_iri[self.short_name] create_type(self.g, self.res, iri_main_type)
# Overrides __str__ method def __str__(self) -> str: return str(self.res)
[docs] def add_triples(self, iterable_of_triples: Iterable[Tuple[term]]) -> None: """ A utility method that allows to add a batch of triples into the graph of the entity. **WARNING: Only triples that have this entity as their subject will be imported!** :param iterable_of_triples: A collection of triples to be added to the entity :type iterable_of_triples: Iterable[Tuple[term]] :return: None """ for s, p, o in iterable_of_triples: if s == self.res: # This guarantees that only triples belonging to the resource will be added self.g.add((s, p, o))
def _get_literal(self, predicate: URIRef) -> Optional[str]: result: Optional[str] = None for o in self.g.objects(self.res, predicate): if type(o) == Literal: result = str(o) break return result def _get_multiple_literals(self, predicate: URIRef) -> List[str]: result: List[str] = [] for o in self.g.objects(self.res, predicate): if type(o) == Literal: result.append(str(o)) return result def _get_uri_reference(self, predicate: URIRef, short_name: str = None) -> Optional[URIRef]: result: Optional[URIRef] = None for o in self.g.objects(self.res, predicate): if type(o) == URIRef: if not is_string_empty(short_name): # If a particular short_name is explicitly requested, # then the following additional check must be performed: if (short_name == '_dataset_' and is_dataset(o)) or get_short_name(o) == short_name: result = o break else: result = o break return result def _get_multiple_uri_references(self, predicate: URIRef, short_name: str = None) -> List[URIRef]: result: List[URIRef] = [] for o in self.g.objects(self.res, predicate): if type(o) == URIRef: if not is_string_empty(short_name): # If a particular short_name is explicitly requested, # then the following additional check must be performed: if (short_name == '_dataset_' and is_dataset(o)) or get_short_name(o) == short_name: result.append(o) else: result.append(o) return result