Source code for nefelibata.assistants.archive_links

import json
import logging
import re
import urllib.parse
from datetime import datetime
from datetime import timedelta
from datetime import timezone
from pathlib import Path
from typing import Any
from typing import Dict

import dateutil.parser
import requests
from bs4 import BeautifulSoup
from nefelibata.assistants import Assistant
from nefelibata.assistants import Scope
from nefelibata.post import Post
from nefelibata.utils import json_storage
from nefelibata.utils import modify_html

_logger = logging.getLogger(__name__)

SAVE_TIMEOUT = timedelta(seconds=10)
RETRY_TIMEOUT = timedelta(days=1)


[docs]class ArchiveLinksAssistant(Assistant): scopes = [Scope.POST] def _safe(self, resource: str): if resource.startswith(self.config["url"]): return True # if the blog uses an external endpoint for webmention that's ok if ( "webmention" in self.config and resource == self.config["webmention"]["endpoint"] ): return True # ignore links to archive itself parsed = urllib.parse.urlparse(resource) return parsed.netloc.endswith("archive.org")
[docs] def process_post(self, post: Post, force: bool = False) -> None: # store all links to the post directory with information about where and # when they were saved, so they can be used by templates post_directory = post.file_path.parent storage = post_directory / "archives.json" archives: Dict[str, Any] with json_storage(storage) as archives: self._archive_links(post, archives) # now enrich links in the generated HTML with the saved link index_html = post.file_path.with_suffix(".html") soup: BeautifulSoup with modify_html(index_html) as soup: # remove existing archive notes for el in soup.find_all("span", attrs={"class": "archive"}): el.decompose() for el in soup.find_all("a", href=re.compile("http")): url = el.attrs["href"] if url not in archives: continue el.attrs["data-archive-url"] = archives[url]["url"] el.attrs["data-archive-date"] = archives[url]["date"] span = soup.new_tag("span", attrs={"class": "archive"}) anchor = soup.new_tag("a", href=archives[url]["url"]) anchor.string = "archived" span.extend(["[", anchor, "]"]) el.insert_after(span)
def _archive_links(self, post: Post, archives: Dict[str, Any]) -> None: # find links from the post HTML soup = BeautifulSoup(post.html, "html.parser") for el in soup.find_all("a", href=re.compile("http")): url = el.attrs["href"] if not url or self._safe(url): continue if url in archives: # if the URL has been archived already, skip if archives[url]["url"]: continue # if the URL has not been archived but we tried # recently, skip it to not overload archive.org if ( datetime.now(tz=timezone.utc) - dateutil.parser.parse(archives[url]["date"]) < RETRY_TIMEOUT ): continue # save to archive.org _logger.info(f"Requesting to save {url}") try: response = requests.get( f"https://web.archive.org/save/{url}", timeout=SAVE_TIMEOUT.total_seconds(), ) except requests.exceptions.ReadTimeout: continue content_location = response.headers.get("Content-Location") archived_url = ( f"https://web.archive.org{content_location}" if content_location else None ) # add link to archived version archives[url] = { "url": archived_url, "date": datetime.now(tz=timezone.utc).isoformat(), }