Source code for swh.indexer.bibtex
# Copyright (C) 2023-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 calendar
import collections
import json
import secrets
import sys
from typing import Any, Dict, List, Optional
import uuid
import iso8601
from pybtex.database import Entry, Person
from pybtex.database.output.bibtex import Writer
from pybtex.plugin import register_plugin
import rdflib
from swh.indexer.codemeta import compact, expand
from swh.indexer.metadata_dictionary.cff import CffMapping
from swh.indexer.namespaces import RDF, SCHEMA, SPDX_LICENSES
from swh.model.swhids import ObjectType, QualifiedSWHID
TMP_ROOT_URI_PREFIX = "https://www.softwareheritage.org/schema/2022/indexer/tmp-node/"
"""IRI used for `skolemization <https://www.w3.org/TR/rdf11-concepts/#section-skolemization>`_;
it is not used outside :func:`codemeta_to_bibtex`.
"""
MACRO_PREFIX = "macro" + secrets.token_urlsafe(16).replace("_", "")
[docs]
class BibTeXWithMacroWriter(Writer):
[docs]
def quote(self, s):
r"""
>>> w = BibTeXWithMacroWriter()
>>> print(w.quote(f'{MACRO_PREFIX}:jan'))
jan
"""
if s.startswith(f"{MACRO_PREFIX}:"):
return s[len(MACRO_PREFIX) + 1 :]
return super().quote(s)
register_plugin("pybtex.database.output", "bibtex_with_macro", BibTeXWithMacroWriter)
[docs]
def codemeta_to_bibtex(
doc: Dict[str, Any], swhid: Optional[QualifiedSWHID] = None
) -> str:
doc = compact(doc, False)
identifiers = []
if "id" in doc:
identifiers.append(doc["id"])
else:
doc["id"] = f"_:{uuid.uuid4()}"
id_: rdflib.term.Node
if doc["id"].startswith("_:"):
id_ = rdflib.term.BNode(doc["id"][2:])
else:
# using a base in case the id is not an absolute URI
id_ = rdflib.term.URIRef(doc["id"], base=TMP_ROOT_URI_PREFIX)
doc["id"] = str(id_)
# workaround for https://github.com/codemeta/codemeta/pull/322
if "identifier" in doc:
if isinstance(doc["identifier"], list):
for identifier in doc["identifier"]:
if isinstance(identifier, str) and "/" not in identifier:
identifiers.append(identifier)
elif isinstance(doc["identifier"], str) and "/" not in doc["identifier"]:
identifiers.append(doc["identifier"])
doc = expand(doc)
g = rdflib.Graph().parse(
data=json.dumps(doc),
format="json-ld",
# replace invalid URIs with blank node ids, instead of discarding whole nodes:
generalized_rdf=True,
)
persons: Dict[str, List[Person]] = collections.defaultdict(list)
fields: Dict[str, Any] = {}
def add_person(
persons: List[Person],
person_id: rdflib.term.Node,
role_property: rdflib.term.URIRef,
) -> None:
# If the node referenced by 'person_id' is actually a Role node, we need to look
# deeper for the actual person node
if (person_id, RDF.type, SCHEMA.Role) in g:
for _, _, inner_person in g.triples((person_id, role_property, None)):
add_person(persons, inner_person, role_property)
person = Person()
for _, _, name in g.triples((person_id, SCHEMA.name, None)):
if (person_id, RDF.type, SCHEMA.Organization) in g:
# prevent interpreting the name as "Firstname Lastname" and reformatting
# it to "Lastname, Firstname"
person.last_names.append(name)
else:
person = Person(name)
for _, _, given_name in g.triples((person_id, SCHEMA.givenName, None)):
person.first_names.append(given_name)
for _, _, family_name in g.triples((person_id, SCHEMA.familyName, None)):
person.last_names.append(family_name)
if str(person) and person not in persons:
persons.append(person)
def add_affiliations(person: rdflib.term.Node) -> None:
for _, _, organization in g.triples((person, SCHEMA.affiliation, None)):
add_person(
persons["organization"], organization, role_property=SCHEMA.affiliation
)
# abstract
for _, _, description in g.triples((id_, SCHEMA.description, None)):
fields["abstract"] = description
break
for _, _, author_or_author_list in g.triples((id_, SCHEMA.author, None)):
# schema.org-style authors, which are single values
add_person(
persons["author"], author_or_author_list, role_property=SCHEMA.author
)
# codemeta-style authors, which are an ordered list
if author_or_author_list == RDF.nil:
# Workaround for https://github.com/RDFLib/rdflib/pull/2818
continue
for author in rdflib.collection.Collection(g, author_or_author_list):
add_person(persons["author"], author, role_property=SCHEMA.author)
add_affiliations(author)
# date
for _, _, date in g.triples((id_, SCHEMA.datePublished, None)):
fields["date"] = date
break
else:
for _, _, date in g.triples((id_, SCHEMA.dateCreated, None)):
fields["date"] = date
break
else:
for _, _, date in g.triples((id_, SCHEMA.dateModified, None)):
fields["date"] = date
break
if "date" in fields:
try:
parsed_date = iso8601.parse_date(fields["date"])
fields["year"] = str(parsed_date.year)
fields["month"] = (
f"{MACRO_PREFIX}:{calendar.month_abbr[parsed_date.month].lower()}"
)
except iso8601.ParseError:
pass
# identifier, doi, hal_id
entry_key = None
for _, _, identifier in g.triples((id_, SCHEMA.identifier, None)):
identifiers.append(identifier)
for identifier in identifiers:
if entry_key is None and "/" not in identifier:
# Avoid URLs
entry_key = identifier
if identifier.startswith("https://doi.org/"):
fields["doi"] = identifier
if identifier.startswith("hal-"):
fields["hal_id"] = identifier
# editor
for _, _, editor in g.triples((id_, SCHEMA.editor, None)):
add_person(persons["editor"], editor, role_property=SCHEMA.editor)
add_affiliations(editor)
# file
for _, _, download_url in g.triples((id_, SCHEMA.downloadUrl, None)):
fields["file"] = download_url
break
# license (represented by "Person" as it's the only way to make pybtex format
# them as a list)
for _, _, license in g.triples((id_, SCHEMA.license, None)):
if license is None:
continue
license_ = str(license)
if license_.startswith(str(SPDX_LICENSES)):
license_ = license_[len(str(SPDX_LICENSES)) :]
if license_.endswith(".html"):
license_ = license_[:-5]
persons["license"].append(Person(last=license_))
# publisher
for _, _, publisher in g.triples((id_, SCHEMA.publisher, None)):
add_person(persons["publisher"], publisher, role_property=SCHEMA.publisher)
add_affiliations(publisher)
# repository
for _, _, code_repository in g.triples((id_, SCHEMA.codeRepository, None)):
fields["repository"] = code_repository
break
# title
for _, _, name in g.triples((id_, SCHEMA.name, None)):
fields["title"] = name
break
# url
for _, _, name in g.triples((id_, SCHEMA.url, None)):
fields["url"] = name
break
# version
for _, _, version in g.triples((id_, SCHEMA.softwareVersion, None)):
fields["version"] = version
break
else:
for _, _, version in g.triples((id_, SCHEMA.version, None)):
fields["version"] = version
# entry_type
if swhid:
fields["swhid"] = str(swhid)
if swhid.object_type == ObjectType.SNAPSHOT:
entry_type = "software"
elif swhid.object_type == ObjectType.CONTENT:
entry_type = "codefragment"
else:
entry_type = "softwareversion"
elif "version" in fields:
entry_type = "softwareversion"
else:
entry_type = "software"
entry = Entry(
entry_type,
persons=persons,
fields=fields,
)
entry.key = entry_key or "REPLACEME"
return entry.to_string(bib_format="bibtex_with_macro")
[docs]
def cff_to_bibtex(content: str, swhid: Optional[QualifiedSWHID] = None) -> str:
codemeta = CffMapping().translate(raw_content=content.encode("utf-8"))
if codemeta is None:
codemeta = {}
return codemeta_to_bibtex(codemeta, swhid)
if __name__ == "__main__":
for filename in sys.argv[1:]:
if filename == "-":
print(codemeta_to_bibtex(json.load(sys.stdin)))
else:
with open(filename) as f:
print(codemeta_to_bibtex(json.load(f)))