feat: LetterFeed logo for feed

This commit is contained in:
Leon
2025-07-19 19:12:15 +02:00
parent 830fcc5757
commit 5abb379af0
4 changed files with 7 additions and 0 deletions

View File

@@ -12,6 +12,7 @@ LetterFeed is a self-hosted application that transforms your email newsletters i
- **Content Extraction:** Optionally, LetterFeed can extract the main article content from the email body. - **Content Extraction:** Optionally, LetterFeed can extract the main article content from the email body.
- **Email Management:** Can automatically move processed emails to a specified folder in your inbox to keep things organized. - **Email Management:** Can automatically move processed emails to a specified folder in your inbox to keep things organized.
- **Easy to Use Interface:** A simple web interface to manage your newsletters and feeds. - **Easy to Use Interface:** A simple web interface to manage your newsletters and feeds.
- **Authentication:** Optional username + password auth.
## How It Works ## How It Works

View File

@@ -16,10 +16,12 @@ def generate_feed(db: Session, newsletter_id: str):
entries = get_entries_by_newsletter(db, newsletter_id) entries = get_entries_by_newsletter(db, newsletter_id)
feed_url = f"{settings.app_base_url}/feeds/{newsletter_id}" feed_url = f"{settings.app_base_url}/feeds/{newsletter_id}"
logo_url = f"{settings.app_base_url}/logo.png"
fg = FeedGenerator() fg = FeedGenerator()
fg.id(f"urn:letterfeed:newsletter:{newsletter.id}") fg.id(f"urn:letterfeed:newsletter:{newsletter.id}")
fg.title(newsletter.name) fg.title(newsletter.name)
fg.logo(logo_url)
fg.link(href=feed_url, rel="self") fg.link(href=feed_url, rel="self")
sender_emails = ", ".join([s.email for s in newsletter.senders]) sender_emails = ", ".join([s.email for s in newsletter.senders])
fg.description(f"A feed of newsletters from {sender_emails}") fg.description(f"A feed of newsletters from {sender_emails}")

View File

@@ -182,6 +182,9 @@ def test_get_newsletter_feed(client: TestClient):
root = ET.fromstring(response.text) root = ET.fromstring(response.text)
# Atom feed uses a namespace, so we need to include it in our tag searches # Atom feed uses a namespace, so we need to include it in our tag searches
ns = {"atom": "http://www.w3.org/2005/Atom"} ns = {"atom": "http://www.w3.org/2005/Atom"}
logo = root.find("atom:logo", ns)
assert logo is not None
assert logo.text == "http://localhost:8000/logo.png"
entry_titles = [ entry_titles = [
entry.find("atom:title", ns).text for entry in root.findall("atom:entry", ns) entry.find("atom:title", ns).text for entry in root.findall("atom:entry", ns)
] ]

View File

@@ -40,6 +40,7 @@ def test_generate_feed(db_session: Session):
# In a real scenario, you'd use an XML parser to validate structure and content more thoroughly # In a real scenario, you'd use an XML parser to validate structure and content more thoroughly
assert f"<title>{newsletter.name}</title>" in feed_xml.decode() assert f"<title>{newsletter.name}</title>" in feed_xml.decode()
assert f"<id>urn:letterfeed:newsletter:{newsletter.id}</id>" in feed_xml.decode() assert f"<id>urn:letterfeed:newsletter:{newsletter.id}</id>" in feed_xml.decode()
assert "<logo>http://localhost:8000/logo.png</logo>" in feed_xml.decode()
assert "<title>First Entry</title>" in feed_xml.decode() assert "<title>First Entry</title>" in feed_xml.decode()
assert "<title>Second Entry</title>" in feed_xml.decode() assert "<title>Second Entry</title>" in feed_xml.decode()
assert ( assert (