Source code for stdpopsim.citations

"""
Citation management for stdpopsim. Provides utilities for printing
citation information associated with different entities derived from the
literature that are used within a simulation.
"""

import collections
import urllib.request

import attr


# TODO this should be an enum.Enum


class CiteReason:
    ENGINE = "simulation engine"
    DEM_MODEL = "demographic model"
    GEN_MAP = "genetic map"
    POP_SIZE = "population size"
    GEN_TIME = "generation time"
    MUT_RATE = "mutation rate"
    REC_RATE = "recombination rate"
    GENE_CONVERSION = "gene conversion parameters"
    ASSEMBLY = "genome assembly"
    ANNOTATION = "genome annotation"
    DFE = "distribution of fitness effects"
    STDPOPSIM = "stdpopsim"


[docs] @attr.s class Citation: """ A reference to the literature that should be acknowledged by users of stdpopsim. :ivar doi: The DOI for the publication providing the definitive reference. :vartype doi: str :ivar author: Short author list, .e.g, "Author 1 et. al". :vartype author: str :ivar year: Year of publication as a 4 digit integer, e.g. 2008. :vartype year: int """ doi = attr.ib(type=str, kw_only=True) author = attr.ib(type=str, kw_only=True) year = attr.ib(type=int, kw_only=True) reasons = attr.ib(factory=set, kw_only=True) def __str__(self): return f"{self.author}, {self.year}: {self.doi}" def displaystr(self): s = ["["] for i, reason in enumerate(self.reasons): if i != 0: s.append(", ") s.append(reason) s.append("]\n") s.append(str(self)) return "".join(s)
[docs] def because(self, reasons): """Returns a new Citation with the given reasons.""" if not isinstance(reasons, set): reasons = {reasons} return self.__class__( author=self.author, year=self.year, doi=self.doi, reasons=self.reasons | reasons, )
[docs] @staticmethod def merge(citations): """Returns a deduplicated list of Citation objects.""" cset = collections.OrderedDict() for citation in citations: if citation.doi in cset: citation = cset[citation.doi].because(citation.reasons) cset[citation.doi] = citation return list(cset.values())
[docs] def assert_valid(self): """ Checks that this citation is valid by checking the types and values of the instance variables. """ assert isinstance(self.author, str) assert len(self.author) > 0 assert isinstance(self.doi, str) assert len(self.doi) > 0 parsed = urllib.parse.urlparse(self.doi) assert parsed.scheme.startswith("http") assert isinstance(self.year, int) assert self.year > 0 for reason in self.reasons: assert isinstance(reason, str)
# TODO check that it's in the set of accepted reasons.
[docs] def fetch_bibtex(self): """Retrieve the bibtex of a citation from Crossref.""" req = urllib.request.Request(self.doi) req.add_header("Accept", "text/bibliography; style=bibtex") with urllib.request.urlopen(req) as con: return con.read().decode()
_stdpopsim_citation = Citation( doi="https://doi.org/10.7554/eLife.54967", year="2020", author="Adrion et al.", reasons={CiteReason.STDPOPSIM}, ) _catalog_citation = Citation( doi="https://doi.org/10.7554/eLife.84874", year="2023", author="Lauterbur et al.", reasons={CiteReason.STDPOPSIM}, )