Permalink
Cannot retrieve contributors at this time
executable file
203 lines (167 sloc)
7.18 KB
Name already in use
A tag already exists with the provided branch name. Many Git commands accept both tag and branch names, so creating this branch may cause unexpected behavior. Are you sure you want to create this branch?
peps/generate_rss.py /
Go to fileThis commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#!/usr/bin/env python3 | |
# This file is placed in the public domain or under the | |
# CC0-1.0-Universal license, whichever is more permissive. | |
import datetime | |
import email.utils | |
from pathlib import Path | |
import re | |
import docutils.frontend | |
from docutils import nodes | |
from docutils import utils | |
from docutils.parsers import rst | |
from docutils.parsers.rst import roles | |
from feedgen import entry | |
from feedgen import feed | |
# get the directory with the PEP sources | |
PEP_ROOT = Path(__file__).parent | |
# Monkeypatch feedgen.util.formatRFC2822 | |
def _format_rfc_2822(dt: datetime.datetime) -> str: | |
return email.utils.format_datetime(dt, usegmt=True) | |
entry.formatRFC2822 = feed.formatRFC2822 = _format_rfc_2822 | |
line_cache: dict[Path, dict[str, str]] = {} | |
# Monkeypatch PEP and RFC reference roles to match Sphinx behaviour | |
EXPLICIT_TITLE_RE = re.compile(r'^(.+?)\s*(?<!\x00)<(.*?)>$', re.DOTALL) | |
def _pep_reference_role(role, rawtext, text, lineno, inliner, | |
options={}, content=[]): | |
matched = EXPLICIT_TITLE_RE.match(text) | |
if matched: | |
title = utils.unescape(matched.group(1)) | |
target = utils.unescape(matched.group(2)) | |
else: | |
target = utils.unescape(text) | |
title = "PEP " + utils.unescape(text) | |
pep_str, _, fragment = target.partition("#") | |
try: | |
pepnum = int(pep_str) | |
if pepnum < 0 or pepnum > 9999: | |
raise ValueError | |
except ValueError: | |
msg = inliner.reporter.error( | |
f'PEP number must be a number from 0 to 9999; "{pep_str}" is invalid.', | |
line=lineno) | |
prb = inliner.problematic(rawtext, rawtext, msg) | |
return [prb], [msg] | |
# Base URL mainly used by inliner.pep_reference; so this is correct: | |
ref = (inliner.document.settings.pep_base_url | |
+ inliner.document.settings.pep_file_url_template % pepnum) | |
if fragment: | |
ref += "#" + fragment | |
roles.set_classes(options) | |
return [nodes.reference(rawtext, title, refuri=ref, **options)], [] | |
def _rfc_reference_role(role, rawtext, text, lineno, inliner, | |
options={}, content=[]): | |
matched = EXPLICIT_TITLE_RE.match(text) | |
if matched: | |
title = utils.unescape(matched.group(1)) | |
target = utils.unescape(matched.group(2)) | |
else: | |
target = utils.unescape(text) | |
title = "RFC " + utils.unescape(text) | |
pep_str, _, fragment = target.partition("#") | |
try: | |
rfcnum = int(pep_str) | |
if rfcnum < 0 or rfcnum > 9999: | |
raise ValueError | |
except ValueError: | |
msg = inliner.reporter.error( | |
f'RFC number must be a number from 0 to 9999; "{pep_str}" is invalid.', | |
line=lineno) | |
prb = inliner.problematic(rawtext, rawtext, msg) | |
return [prb], [msg] | |
ref = (inliner.document.settings.rfc_base_url + inliner.rfc_url % rfcnum) | |
if fragment: | |
ref += "#" + fragment | |
roles.set_classes(options) | |
return [nodes.reference(rawtext, title, refuri=ref, **options)], [] | |
roles.register_canonical_role("pep-reference", _pep_reference_role) | |
roles.register_canonical_role("rfc-reference", _rfc_reference_role) | |
def first_line_starting_with(full_path: Path, text: str) -> str: | |
# Try and retrieve from cache | |
if full_path in line_cache: | |
return line_cache[full_path].get(text, "") | |
# Else read source | |
line_cache[full_path] = path_cache = {} | |
for line in full_path.open(encoding="utf-8"): | |
if line.startswith("Created:"): | |
path_cache["Created:"] = line.removeprefix("Created:").strip() | |
elif line.startswith("Title:"): | |
path_cache["Title:"] = line.removeprefix("Title:").strip() | |
elif line.startswith("Author:"): | |
path_cache["Author:"] = line.removeprefix("Author:").strip() | |
# Once all have been found, exit loop | |
if path_cache.keys == {"Created:", "Title:", "Author:"}: | |
break | |
return path_cache.get(text, "") | |
def pep_creation(full_path: Path) -> datetime.datetime: | |
created_str = first_line_starting_with(full_path, "Created:") | |
if full_path.stem == "pep-0102": | |
# remove additional content on the Created line | |
created_str = created_str.split(" ", 1)[0] | |
return datetime.datetime.strptime(created_str, "%d-%b-%Y") | |
def parse_rst(full_path: Path) -> nodes.document: | |
text = full_path.read_text(encoding="utf-8") | |
settings = docutils.frontend.get_default_settings(rst.Parser) | |
document = utils.new_document(f'<{full_path}>', settings=settings) | |
rst.Parser(rfc2822=True).parse(text, document) | |
return document | |
def pep_abstract(full_path: Path) -> str: | |
"""Return the first paragraph of the PEP abstract""" | |
for node in parse_rst(full_path).findall(nodes.section): | |
if node.next_node(nodes.title).astext() == "Abstract": | |
return node.next_node(nodes.paragraph).astext().strip().replace("\n", " ") | |
return "" | |
def main(): | |
# get list of peps with creation time (from "Created:" string in pep source) | |
peps_with_dt = sorted((pep_creation(path), path) for path in PEP_ROOT.glob("pep-????.???")) | |
# generate rss items for 10 most recent peps | |
items = [] | |
for dt, full_path in peps_with_dt[-10:]: | |
try: | |
pep_num = int(full_path.stem.split("-")[-1]) | |
except ValueError: | |
continue | |
title = first_line_starting_with(full_path, "Title:") | |
author = first_line_starting_with(full_path, "Author:") | |
if "@" in author or " at " in author: | |
parsed_authors = email.utils.getaddresses([author]) | |
# ideal would be to pass as a list of dicts with names and emails to | |
# item.author, but FeedGen's RSS <author/> output doesn't pass W3C | |
# validation (as of 12/06/2021) | |
joined_authors = ", ".join(f"{name} ({email_address})" for name, email_address in parsed_authors) | |
else: | |
joined_authors = author | |
url = f"https://peps.python.org/pep-{pep_num:0>4}/" | |
item = entry.FeedEntry() | |
item.title(f"PEP {pep_num}: {title}") | |
item.link(href=url) | |
item.description(pep_abstract(full_path)) | |
item.guid(url, permalink=True) | |
item.published(dt.replace(tzinfo=datetime.timezone.utc)) # ensure datetime has a timezone | |
item.author(email=joined_authors) | |
items.append(item) | |
# The rss envelope | |
desc = """ | |
Newest Python Enhancement Proposals (PEPs) - Information on new | |
language features, and some meta-information like release | |
procedure and schedules. | |
""" | |
# Setup feed generator | |
fg = feed.FeedGenerator() | |
fg.language("en") | |
fg.generator("") | |
fg.docs("https://cyber.harvard.edu/rss/rss.html") | |
# Add metadata | |
fg.title("Newest Python PEPs") | |
fg.link(href="https://peps.python.org") | |
fg.link(href="https://peps.python.org/peps.rss", rel="self") | |
fg.description(" ".join(desc.split())) | |
fg.lastBuildDate(datetime.datetime.utcnow().replace(tzinfo=datetime.timezone.utc)) | |
# Add PEP information (ordered by newest first) | |
for item in items: | |
fg.add_entry(item) | |
# output directory for target HTML files | |
out_dir = PEP_ROOT / "build" | |
out_dir.mkdir(exist_ok=True) | |
out_dir.joinpath("peps.rss").write_bytes(fg.rss_str(pretty=True)) | |
if __name__ == "__main__": | |
main() |