import { Context } from "hono"; import { html, raw } from "hono/html"; import { Env } from "../types"; import { processEmailContent } from "../utils/html-processor"; import { formatBytes } from "../utils/format"; import { FeedRepository } from "../domain/feed-repository"; import { FeedId } from "../domain/value-objects/feed-id"; import { isExpired } from "../domain/feed"; export async function handle(c: Context<{ Bindings: Env }>): Promise { const feedId = c.req.param("feedId"); const receivedAt = parseInt(c.req.param("entryId") ?? "", 10); if (!feedId || isNaN(receivedAt)) { return new Response("Not Found", { status: 404 }); } const repo = FeedRepository.from(c.env); const id = FeedId.fromTrusted(feedId); const [feedMetadata, feedConfig] = await Promise.all([ repo.getMetadata(id), repo.getConfig(id), ]); if (!feedMetadata) { return new Response("Feed not found", { status: 404 }); } if (feedConfig && isExpired(feedConfig)) { return new Response("Feed has expired", { status: 410 }); } const metaEntry = feedMetadata.emails.find( (e) => e.receivedAt === receivedAt, ); if (!metaEntry) { return new Response("Entry not found", { status: 404 }); } const emailData = await repo.getEmail(metaEntry.key); if (!emailData) { return new Response("Entry not found", { status: 404 }); } c.header( "Content-Security-Policy", "default-src 'none'; style-src 'unsafe-inline'; img-src *; frame-src 'none'", ); const attachments = emailData.attachments ?? []; const attachmentsSection = attachments.length ? html`

Attachments

` : ""; return c.html( html` ${emailData.subject}

${emailData.subject}

From:
${emailData.from}
Date:
${new Date(emailData.receivedAt).toUTCString()}
${raw( processEmailContent(emailData.content, emailData.attachments), )}
${attachmentsSection} `, ); }