"""Code for handling object representation of a PEP.""" from __future__ import annotations import dataclasses from email.parser import HeaderParser from pathlib import Path from pep_sphinx_extensions.pep_zero_generator.constants import ACTIVE_ALLOWED from pep_sphinx_extensions.pep_zero_generator.constants import HIDE_STATUS from pep_sphinx_extensions.pep_zero_generator.constants import SPECIAL_STATUSES from pep_sphinx_extensions.pep_zero_generator.constants import STATUS_ACTIVE from pep_sphinx_extensions.pep_zero_generator.constants import STATUS_PROVISIONAL from pep_sphinx_extensions.pep_zero_generator.constants import STATUS_VALUES from pep_sphinx_extensions.pep_zero_generator.constants import TYPE_STANDARDS from pep_sphinx_extensions.pep_zero_generator.constants import TYPE_VALUES from pep_sphinx_extensions.pep_zero_generator.errors import PEPError @dataclasses.dataclass(order=True, frozen=True) class _Author: """Represent PEP authors.""" full_name: str # The author's name. email: str # The author's email address. class PEP: """Representation of PEPs. Attributes: number : PEP number. title : PEP title. pep_type : The type of PEP. Can only be one of the values from TYPE_VALUES. status : The PEP's status. Value must be found in STATUS_VALUES. authors : A list of the authors. """ # The required RFC 822 headers for all PEPs. required_headers = {"PEP", "Title", "Author", "Status", "Type", "Created"} def __init__(self, filename: Path): """Init object from an open PEP file object. pep_file is full text of the PEP file, filename is path of the PEP file, author_lookup is author exceptions file """ self.filename: Path = filename # Parse the headers. pep_text = filename.read_text(encoding="utf-8") metadata = HeaderParser().parsestr(pep_text) required_header_misses = PEP.required_headers - set(metadata.keys()) if required_header_misses: _raise_pep_error(self, f"PEP is missing required headers {required_header_misses}") try: self.number = int(metadata["PEP"]) except ValueError: _raise_pep_error(self, "PEP number isn't an integer") # Check PEP number matches filename if self.number != int(filename.stem[4:]): _raise_pep_error(self, f"PEP number does not match file name ({filename})", pep_num=True) # Title self.title: str = metadata["Title"] # Type self.pep_type: str = metadata["Type"] if self.pep_type not in TYPE_VALUES: _raise_pep_error(self, f"{self.pep_type} is not a valid Type value", pep_num=True) # Status status = metadata["Status"] if status in SPECIAL_STATUSES: status = SPECIAL_STATUSES[status] if status not in STATUS_VALUES: _raise_pep_error(self, f"{status} is not a valid Status value", pep_num=True) # Special case for Active PEPs. if status == STATUS_ACTIVE and self.pep_type not in ACTIVE_ALLOWED: msg = "Only Process and Informational PEPs may have an Active status" _raise_pep_error(self, msg, pep_num=True) # Special case for Provisional PEPs. if status == STATUS_PROVISIONAL and self.pep_type != TYPE_STANDARDS: msg = "Only Standards Track PEPs may have a Provisional status" _raise_pep_error(self, msg, pep_num=True) self.status: str = status # Parse PEP authors self.authors: list[_Author] = _parse_author(metadata["Author"]) if not self.authors: raise _raise_pep_error(self, "no authors found", pep_num=True) # Topic (for sub-indices) _topic = metadata.get("Topic", "").lower().split(",") self.topic: set[str] = {topic for topic_raw in _topic if (topic := topic_raw.strip())} # Other headers self.created = metadata["Created"] self.discussions_to = metadata["Discussions-To"] self.python_version = metadata["Python-Version"] self.replaces = metadata["Replaces"] self.requires = metadata["Requires"] self.resolution = metadata["Resolution"] self.superseded_by = metadata["Superseded-By"] if metadata["Post-History"]: # Squash duplicate whitespace self.post_history = " ".join(metadata["Post-History"].split()) else: self.post_history = None def __repr__(self) -> str: return f"4} - {self.title}>" def __lt__(self, other: PEP) -> bool: return self.number < other.number def __eq__(self, other): return self.number == other.number @property def shorthand(self) -> str: """Return reStructuredText tooltip for the PEP type and status.""" type_code = self.pep_type[0].upper() if self.status in HIDE_STATUS: return f":abbr:`{type_code} ({self.pep_type}, {self.status})`" status_code = self.status[0].upper() return f":abbr:`{type_code}{status_code} ({self.pep_type}, {self.status})`" @property def details(self) -> dict[str, str | int]: """Return the line entry for the PEP.""" return { "number": self.number, "title": self.title, # a tooltip representing the type and status "shorthand": self.shorthand, # the author list as a comma-separated with only last names "authors": ", ".join(author.full_name for author in self.authors), # The targeted Python-Version (if present) or the empty string "python_version": self.python_version or "", } @property def full_details(self) -> dict[str, str | int]: """Returns all headers of the PEP as a dict.""" return { "number": self.number, "title": self.title, "authors": ", ".join(author.full_name for author in self.authors), "discussions_to": self.discussions_to, "status": self.status, "type": self.pep_type, "topic": ", ".join(sorted(self.topic)), "created": self.created, "python_version": self.python_version, "post_history": self.post_history, "resolution": self.resolution, "requires": self.requires, "replaces": self.replaces, "superseded_by": self.superseded_by, "url": f"https://peps.python.org/pep-{self.number:0>4}/", } def _raise_pep_error(pep: PEP, msg: str, pep_num: bool = False) -> None: if pep_num: raise PEPError(msg, pep.filename, pep_number=pep.number) raise PEPError(msg, pep.filename) jr_placeholder = ",Jr" def _parse_author(data: str) -> list[_Author]: """Return a list of author names and emails.""" author_list = [] data = (data.replace("\n", " ") .replace(", Jr", jr_placeholder) .rstrip().removesuffix(",")) for author_email in data.split(", "): if ' <' in author_email: author, email = author_email.removesuffix(">").split(" <") else: author, email = author_email, "" author = author.strip() if author == "": raise ValueError("Name is empty!") author = author.replace(jr_placeholder, ", Jr") email = email.lower() author_list.append(_Author(author, email)) return author_list