refactor: extract buildFeed helper, add generateAtomFeed

This commit is contained in:
Julien Herr
2026-05-21 07:34:29 +02:00
parent ad5bd9a79a
commit 4d8d1fdc82
2 changed files with 130 additions and 4 deletions
+107
View File
@@ -0,0 +1,107 @@
import { describe, it, expect } from "vitest";
import { generateRssFeed, generateAtomFeed } from "./feed-generator";
import { FeedConfig, EmailData } from "../types";
const mockFeedConfig: FeedConfig = {
title: "Test Newsletter",
description: "A test feed",
site_url: "https://test.getmynews.app/rss/abc123",
feed_url: "https://test.getmynews.app/rss/abc123",
language: "en",
created_at: 1700000000000,
};
const mockEmails: EmailData[] = [
{
subject: "Hello World",
from: "Alice <alice@example.com>",
content: "<p>Hello from Alice</p>",
receivedAt: 1700000001000,
headers: {},
},
];
const BASE_URL = "https://test.getmynews.app";
const FEED_ID = "abc123";
describe("generateRssFeed", () => {
it("returns RSS 2.0 with channel element", () => {
const result = generateRssFeed(mockFeedConfig, mockEmails, BASE_URL, FEED_ID);
expect(result).toContain("<channel>");
expect(result).toContain("<title>Test Newsletter</title>");
});
it("includes rss self-link in RSS output", () => {
const result = generateRssFeed(mockFeedConfig, mockEmails, BASE_URL, FEED_ID);
expect(result).toContain(`${BASE_URL}/rss/${FEED_ID}`);
});
it("includes email entries as <item> elements", () => {
const result = generateRssFeed(mockFeedConfig, mockEmails, BASE_URL, FEED_ID);
expect(result).toContain("<item>");
expect(result).toContain("Hello World");
});
it("works with empty emails array", () => {
const result = generateRssFeed(mockFeedConfig, [], BASE_URL, FEED_ID);
expect(result).toContain("<channel>");
expect(result).not.toContain("<item>");
});
});
describe("generateAtomFeed", () => {
it("returns Atom 1.0 namespace", () => {
const result = generateAtomFeed(mockFeedConfig, mockEmails, BASE_URL, FEED_ID);
expect(result).toContain('xmlns="http://www.w3.org/2005/Atom"');
});
it("contains <feed> root element", () => {
const result = generateAtomFeed(mockFeedConfig, mockEmails, BASE_URL, FEED_ID);
expect(result).toContain("<feed");
expect(result).toContain("</feed>");
});
it("includes feed title", () => {
const result = generateAtomFeed(mockFeedConfig, mockEmails, BASE_URL, FEED_ID);
expect(result).toContain("Test Newsletter");
});
it("includes <entry> elements for each email", () => {
const result = generateAtomFeed(mockFeedConfig, mockEmails, BASE_URL, FEED_ID);
expect(result).toContain("<entry>");
expect(result).toContain("Hello World");
});
it("includes author information", () => {
const result = generateAtomFeed(mockFeedConfig, mockEmails, BASE_URL, FEED_ID);
expect(result).toContain("Alice");
});
it("self-link points to atom URL", () => {
const result = generateAtomFeed(mockFeedConfig, mockEmails, BASE_URL, FEED_ID);
expect(result).toContain(`${BASE_URL}/atom/${FEED_ID}`);
});
it("includes rss alternate link", () => {
const result = generateAtomFeed(mockFeedConfig, mockEmails, BASE_URL, FEED_ID);
expect(result).toContain(`${BASE_URL}/rss/${FEED_ID}`);
});
it("works with empty emails array", () => {
const result = generateAtomFeed(mockFeedConfig, [], BASE_URL, FEED_ID);
expect(result).toContain("<feed");
expect(result).not.toContain("<entry>");
});
it("handles config without description", () => {
const configNoDesc: FeedConfig = { ...mockFeedConfig, description: undefined };
const result = generateAtomFeed(configNoDesc, mockEmails, BASE_URL, FEED_ID);
expect(result).toContain('xmlns="http://www.w3.org/2005/Atom"');
});
it("handles config with author field", () => {
const configWithAuthor: FeedConfig = { ...mockFeedConfig, author: "Bob" };
const result = generateAtomFeed(configWithAuthor, mockEmails, BASE_URL, FEED_ID);
expect(result).toContain("Bob");
});
});
+23 -4
View File
@@ -13,12 +13,12 @@ function parseFromAddress(from: string): { name: string; email?: string } {
return { name: from.trim() }; return { name: from.trim() };
} }
export function generateRssFeed( function buildFeed(
feedConfig: FeedConfig, feedConfig: FeedConfig,
emails: EmailData[], emails: EmailData[],
baseUrl: string, baseUrl: string,
feedId: string, feedId: string,
): string { ): Feed {
const feed = new Feed({ const feed = new Feed({
title: feedConfig.title, title: feedConfig.title,
description: feedConfig.description || "", description: feedConfig.description || "",
@@ -29,7 +29,8 @@ export function generateRssFeed(
generator: "Email-to-RSS", generator: "Email-to-RSS",
copyright: `Copyright © ${new Date().getFullYear()} ${feedConfig.title}`, copyright: `Copyright © ${new Date().getFullYear()} ${feedConfig.title}`,
feedLinks: { feedLinks: {
rss: feedConfig.feed_url, rss: `${baseUrl}/rss/${feedId}`,
atom: `${baseUrl}/atom/${feedId}`,
}, },
author: feedConfig.author author: feedConfig.author
? { ? {
@@ -52,5 +53,23 @@ export function generateRssFeed(
}); });
} }
return feed.rss2(); return feed;
}
export function generateRssFeed(
feedConfig: FeedConfig,
emails: EmailData[],
baseUrl: string,
feedId: string,
): string {
return buildFeed(feedConfig, emails, baseUrl, feedId).rss2();
}
export function generateAtomFeed(
feedConfig: FeedConfig,
emails: EmailData[],
baseUrl: string,
feedId: string,
): string {
return buildFeed(feedConfig, emails, baseUrl, feedId).atom1();
} }