lumbunglib/lumbung-feed-aggregator/rss_aggregator.py

255 lines
6.8 KiB
Python
Raw Normal View History

2021-12-15 10:30:10 +00:00
#!/bin/python3
2021-12-15 10:41:35 +00:00
# lumbung.space rss feed aggregator
# © 2021 roel roscam abbing gplv3 etc
2021-12-15 10:30:10 +00:00
import os
import shutil
import time
2021-12-15 10:41:35 +00:00
from ast import literal_eval as make_tuple
from urllib.parse import urlparse
2021-12-15 10:30:10 +00:00
import arrow
2021-12-15 10:41:35 +00:00
import feedparser
import jinja2
import requests
from bs4 import BeautifulSoup
from slugify import slugify
2021-12-15 10:30:10 +00:00
def write_etag(feed_name, feed_data):
"""
save timestamp of when feed was last modified
"""
2021-12-15 10:41:35 +00:00
etag = ""
modified = ""
if "etag" in feed_data:
2021-12-15 10:30:10 +00:00
etag = feed_data.etag
2021-12-15 10:41:35 +00:00
if "modified" in feed_data:
2021-12-15 10:30:10 +00:00
modified = feed_data.modified
if etag or modified:
2021-12-15 10:41:35 +00:00
with open(os.path.join("etags", feed_name + ".txt"), "w") as f:
2021-12-15 10:30:10 +00:00
f.write(str((etag, modified)))
2021-12-15 10:41:35 +00:00
2021-12-15 10:30:10 +00:00
def get_etag(feed_name):
"""
return timestamp of when feed was last modified
"""
2021-12-15 10:41:35 +00:00
fn = os.path.join("etags", feed_name + ".txt")
etag = ""
modified = ""
2021-12-15 10:30:10 +00:00
if os.path.exists(fn):
2021-12-15 10:41:35 +00:00
etag, modified = make_tuple(open(fn, "r").read())
2021-12-15 10:30:10 +00:00
return etag, modified
2021-12-15 10:41:35 +00:00
2021-12-15 10:30:10 +00:00
def create_frontmatter(entry):
"""
2021-12-15 10:41:35 +00:00
parse RSS metadata and return as frontmatter
2021-12-15 10:30:10 +00:00
"""
2021-12-15 10:41:35 +00:00
if "published" in entry:
2021-12-15 10:30:10 +00:00
published = entry.published_parsed
2021-12-15 10:41:35 +00:00
if "updated" in entry:
2021-12-15 10:30:10 +00:00
published = entry.updated_parsed
published = arrow.get(published)
2021-12-15 10:41:35 +00:00
if "author" in entry:
2021-12-15 10:30:10 +00:00
author = entry.author
else:
2021-12-15 10:41:35 +00:00
author = ""
2021-12-15 10:30:10 +00:00
tags = []
2021-12-15 10:41:35 +00:00
if "tags" in entry:
# TODO finish categories
2021-12-15 10:30:10 +00:00
for t in entry.tags:
2021-12-15 10:41:35 +00:00
tags.append(t["term"])
2021-12-15 10:30:10 +00:00
frontmatter = {
2021-12-15 10:41:35 +00:00
"title": entry.title,
"date": published.format(),
"summary": "",
"author": author,
"original_link": entry.link,
"feed_name": entry["feed_name"],
"tags": str(tags),
2021-12-15 10:30:10 +00:00
}
return frontmatter
2021-12-15 10:41:35 +00:00
2021-12-15 10:30:10 +00:00
def create_post(post_dir, entry):
"""
write hugo post based on RSS entry
"""
frontmatter = create_frontmatter(entry)
if not os.path.exists(post_dir):
os.makedirs(post_dir)
2021-12-15 10:41:35 +00:00
if "content" in entry:
2021-12-15 10:30:10 +00:00
post_content = entry.content[0].value
else:
post_content = entry.summary
parsed_content = parse_posts(post_dir, post_content)
2021-12-15 10:41:35 +00:00
with open(os.path.join(post_dir, "index.html"), "w") as f: # n.b. .html
2021-12-15 10:30:10 +00:00
post = template.render(frontmatter=frontmatter, content=parsed_content)
f.write(post)
2021-12-15 10:41:35 +00:00
print("created post for", entry.title, "({})".format(entry.link))
2021-12-15 10:30:10 +00:00
def grab_media(post_directory, url):
"""
download media linked in post to have local copy
if download succeeds return new local path otherwise return url
"""
2021-12-15 10:41:35 +00:00
image = urlparse(url).path.split("/")[-1]
2021-12-15 10:30:10 +00:00
try:
if not os.path.exists(os.path.join(post_directory, image)):
2021-12-15 10:41:35 +00:00
# TODO: stream is true is a conditional so we could check the headers for things, mimetype etc
2021-12-15 10:30:10 +00:00
response = requests.get(url, stream=True)
if response.ok:
2021-12-15 10:41:35 +00:00
with open(os.path.join(post_directory, image), "wb") as img_file:
2021-12-15 10:30:10 +00:00
shutil.copyfileobj(response.raw, img_file)
2021-12-15 10:41:35 +00:00
print("Downloaded cover image", image)
2021-12-15 10:30:10 +00:00
return image
return image
elif os.path.exists(os.path.join(post_directory, image)):
return image
except Exception as e:
2021-12-15 10:41:35 +00:00
print("Failed to download image", url)
2021-12-15 10:30:10 +00:00
print(e)
return url
def parse_posts(post_dir, post_content):
"""
parse the post content to for media items
replace foreign image with local copy
filter out iframe sources not in allowlist
"""
soup = BeautifulSoup(post_content, "html.parser")
2021-12-15 10:41:35 +00:00
allowed_iframe_sources = ["youtube.com", "vimeo.com", "tv.lumbung.space"]
2021-12-15 10:30:10 +00:00
media = []
2021-12-15 10:41:35 +00:00
for img in soup(["img", "object"]):
local_image = grab_media(post_dir, img["src"])
if img["src"] != local_image:
img["src"] = local_image
2021-12-15 10:30:10 +00:00
2021-12-15 10:41:35 +00:00
for iframe in soup(["iframe"]):
if not any(source in iframe["src"] for source in allowed_iframe_sources):
print("filtered iframe: {}...".format(iframe["src"][:25]))
2021-12-15 10:30:10 +00:00
iframe.decompose()
return soup.decode()
2021-12-15 10:41:35 +00:00
2021-12-15 10:30:10 +00:00
def grab_feed(feed_url):
"""
check whether feed has been updated
2021-12-15 10:41:35 +00:00
download & return it if it has
2021-12-15 10:30:10 +00:00
"""
feed_name = urlparse(feed_url).netloc
2021-12-15 10:41:35 +00:00
2021-12-15 10:30:10 +00:00
etag, modified = get_etag(feed_name)
try:
if modified:
data = feedparser.parse(feed_url, modified=modified)
elif etag:
data = feedparser.parse(feed_url, etag=etag)
else:
data = feedparser.parse(feed_url)
except Exception as e:
2021-12-15 10:41:35 +00:00
print("Error grabbing feed")
2021-12-15 10:30:10 +00:00
print(feed_name)
print(e)
return False
print(data.status, feed_url)
if data.status == 200:
2021-12-15 10:41:35 +00:00
# 304 means the feed has not been modified since we last checked
2021-12-15 10:30:10 +00:00
write_etag(feed_name, data)
return data
return False
2021-12-15 10:41:35 +00:00
feed_urls = open("feeds_list.txt", "r").read().splitlines()
2021-12-15 10:30:10 +00:00
start = time.time()
2021-12-15 10:41:35 +00:00
if not os.path.exists("etags"):
os.mkdir("etags")
2021-12-15 10:30:10 +00:00
2021-12-15 10:41:35 +00:00
env = jinja2.Environment(loader=jinja2.FileSystemLoader(os.path.curdir))
2021-12-15 10:30:10 +00:00
2021-12-15 10:41:35 +00:00
output_dir = os.environ.get(
"OUTPUT_DIR", "/home/r/Programming/lumbung.space/lumbung.space-web/content/posts/"
)
# output_dir = os.environ.get('OUTPUT_DIR', 'network/')
2021-12-15 10:30:10 +00:00
if not os.path.exists(output_dir):
os.makedirs(output_dir)
2021-12-15 10:41:35 +00:00
template = env.get_template("post_template.md")
2021-12-15 10:30:10 +00:00
2021-12-15 10:41:35 +00:00
# add iframe to the allowlist of feedparser's sanitizer,
# this is now handled in parse_post()
feedparser.sanitizer._HTMLSanitizer.acceptable_elements |= {"iframe"}
2021-12-15 10:30:10 +00:00
for feed_url in feed_urls:
feed_name = urlparse(feed_url).netloc
feed_dir = os.path.join(output_dir, feed_name)
if not os.path.exists(feed_dir):
os.makedirs(feed_dir)
existing_posts = os.listdir(feed_dir)
data = grab_feed(feed_url)
if data:
for entry in data.entries:
2021-12-15 10:41:35 +00:00
# if 'tags' in entry:
# for tag in entry.tags:
# for x in ['lumbung.space', 'D15', 'lumbung']:
# if x in tag['term']:
# print(entry.title)
entry["feed_name"] = feed_name
2021-12-15 10:30:10 +00:00
post_name = slugify(entry.title)
post_dir = os.path.join(output_dir, feed_name, post_name)
2021-12-15 10:41:35 +00:00
if post_name not in existing_posts:
# if there is a blog entry we dont already have, make it
2021-12-15 10:30:10 +00:00
create_post(post_dir, entry)
2021-12-15 10:41:35 +00:00
elif post_name in existing_posts:
# if we already have it, update it
create_post(post_dir, entry)
existing_posts.remove(
post_name
) # create list of posts which have not been returned by the feed
2021-12-15 10:30:10 +00:00
2021-12-15 10:41:35 +00:00
for post in existing_posts:
# remove blog posts no longer returned by the RSS feed
print("deleted", post)
2021-12-15 10:30:10 +00:00
shutil.rmtree(os.path.join(feed_dir, slugify(post)))
end = time.time()
print(end - start)