Source code for swh.indexer.metadata_dictionary.codemeta

# Copyright (C) 2018-2024  The Software Heritage developers
# See the AUTHORS file at the top-level directory of this distribution
# License: GNU General Public License version 3, or any later version
# See top-level LICENSE file for more information

import collections
import json
import logging
import re
from typing import Any, Dict, List, Optional, Tuple, Union
import xml.etree.ElementTree as ET

import iso8601
import xmltodict

from swh.indexer.codemeta import (
    CODEMETA_TERMS,
    CODEMETA_V2_CONTEXT_URL,
    compact,
    expand,
)

from .base import BaseExtrinsicMapping, SingleFileIntrinsicMapping

ATOM_URI = "http://www.w3.org/2005/Atom"

_TAG_RE = re.compile(r"\{(?P<namespace>.*?)\}(?P<localname>.*)")
_IGNORED_NAMESPACES = ("http://www.w3.org/2005/Atom",)
_DATE_RE = re.compile("^[0-9]{4}-[0-9]{1,2}-[0-9]{1,2}$")

logger = logging.getLogger(__name__)


[docs] class CodemetaMapping(SingleFileIntrinsicMapping): """ dedicated class for CodeMeta (codemeta.json) mapping and translation """ name = "codemeta" filename = b"codemeta.json" string_fields = None
[docs] @classmethod def supported_terms(cls) -> List[str]: return [term for term in CODEMETA_TERMS if not term.startswith("@")]
[docs] def translate(self, content: bytes) -> Optional[Dict[str, Any]]: try: return self.normalize_translation(expand(json.loads(content.decode()))) except Exception: return None
[docs] class SwordCodemetaMapping(BaseExtrinsicMapping): """ dedicated class for mapping and translation from JSON-LD statements embedded in SWORD documents, optionally using Codemeta contexts, as described in the :ref:`deposit-protocol`. """ name = "sword-codemeta"
[docs] @classmethod def extrinsic_metadata_formats(cls) -> Tuple[str, ...]: return ( "sword-v2-atom-codemeta", "sword-v2-atom-codemeta-v2", )
[docs] @classmethod def supported_terms(cls) -> List[str]: return [term for term in CODEMETA_TERMS if not term.startswith("@")]
[docs] def xml_to_jsonld(self, e: ET.Element) -> Union[str, Dict[str, Any]]: # Keys are JSON-LD property names (URIs or terms). # Values are either a single string (if key is "type") or list of # other dicts with the same type recursively. # To simply annotations, we omit the single string case here. doc: Dict[str, List[Union[str, Dict[str, Any]]]] = collections.defaultdict(list) for child in e: m = _TAG_RE.match(child.tag) assert m, f"Tag with no namespace: {child}" namespace = m.group("namespace") localname = m.group("localname") if namespace == ATOM_URI and localname in ("title", "name"): # Convert Atom to Codemeta name; in case codemeta:name # is not provided or different doc["name"].append(self.xml_to_jsonld(child)) elif namespace == ATOM_URI and localname in ("author", "email"): # ditto for these author properties (note that author email is also # covered by the previous test) doc[localname].append(self.xml_to_jsonld(child)) elif namespace in _IGNORED_NAMESPACES: # SWORD-specific namespace that is not interesting to translate pass elif namespace.lower() == CODEMETA_V2_CONTEXT_URL: # It is a term defined by the context; write is as-is and JSON-LD # expansion will convert it to a full URI based on # "@context": CODEMETA_V2_CONTEXT_URL jsonld_child = self.xml_to_jsonld(child) if ( localname in ( "dateCreated", "dateModified", "datePublished", ) and isinstance(jsonld_child, str) and _DATE_RE.match(jsonld_child) ): # Dates missing a leading zero for their day/month, used # to be allowed by the deposit; so we need to reformat them # to be valid ISO8601. jsonld_child = iso8601.parse_date(jsonld_child).date().isoformat() if localname == "id": # JSON-LD only allows a single id, and they have to be strings. if localname in doc: logger.error( "Duplicate <id>s in SWORD document: %r and %r", doc[localname], jsonld_child, ) continue elif not jsonld_child: logger.error("Empty <id> value in SWORD document") continue elif not isinstance(jsonld_child, str): logger.error( "Unexpected <id> value in SWORD document: %r", jsonld_child ) continue else: doc[localname] = jsonld_child # type: ignore[assignment] else: doc[localname].append(jsonld_child) else: # Otherwise, we already know the URI doc[f"{namespace}{localname}"].append(self.xml_to_jsonld(child)) # The above needed doc values to be list to work; now we allow any type # of value as key "@value" cannot have a list as value. doc_: Dict[str, Any] = doc text = e.text.strip() if e.text else None if text: # TODO: check doc is empty, and raise mixed-content error otherwise? return text return doc_
[docs] def translate(self, content: bytes) -> Optional[Dict[str, Any]]: # Parse XML try: root = ET.fromstring(content) except ET.ParseError: logger.error("Failed to parse XML document: %s", content) return None else: # Transform to JSON-LD document doc = self.xml_to_jsonld(root) assert isinstance(doc, dict), f"Root object is not a dict: {doc}" # Add @context to JSON-LD expansion replaces the "codemeta:" prefix # hash (which uses the context URL as namespace URI for historical # reasons) into properties in `http://schema.org/` and # `https://codemeta.github.io/terms/` namespaces doc["@context"] = CODEMETA_V2_CONTEXT_URL # Normalize as a Codemeta document return self.normalize_translation(expand(doc))
[docs] def normalize_translation(self, metadata: Dict[str, Any]) -> Dict[str, Any]: return compact(metadata, forgefed=False)
[docs] def iter_keys(d): """Recursively iterates on dictionary keys""" if isinstance(d, dict): yield from d for value in d: yield from iter_keys(value) elif isinstance(d, list): for value in d: yield from iter_keys(value) else: pass
[docs] class JsonSwordCodemetaMapping(SwordCodemetaMapping): """ Variant of :class:`SwordCodemetaMapping` that reads the legacy ``sword-v2-atom-codemeta-v2-in-json`` format and converts it back to ``sword-v2-atom-codemeta-v2`` XML """ name = "json-sword-codemeta"
[docs] @classmethod def extrinsic_metadata_formats(cls) -> Tuple[str, ...]: return ("sword-v2-atom-codemeta-v2-in-json",)
[docs] def translate(self, content: bytes) -> Optional[Dict[str, Any]]: # ``content`` was generated by calling ``xmltodict.parse()`` on a XML document, # so ``xmltodict.unparse()`` is guaranteed to return a document that is # semantically equivalent to the original and pass it to SwordCodemetaMapping. try: json_doc = json.loads(content) except json.JSONDecodeError: logger.error("Failed to parse JSON document: %s", content) return None else: if "@xmlns" not in json_doc: # Technically invalid, but old versions of the deposit dropped # XMLNS information json_doc["@xmlns"] = ATOM_URI if "@xmlns:codemeta" not in json_doc and any( key.startswith("codemeta:") for key in iter_keys(json_doc) ): # ditto json_doc["@xmlns:codemeta"] = CODEMETA_V2_CONTEXT_URL if json_doc["@xmlns"] not in (ATOM_URI, [ATOM_URI]): # Technically, non-default XMLNS were allowed, but no one used them, # and we don't write this format anymore, so they do not need to be # implemented here. raise NotImplementedError(f"Unexpected XMLNS set: {json_doc}") # Root tag was stripped by swh-deposit json_doc = {"entry": json_doc} return super().translate(xmltodict.unparse(json_doc))