import express from "express"; import { SMTPServer } from "smtp-server"; import mailparser from "mailparser"; import * as sanitizeXMLString from "sanitize-xml-string"; import * as entities from "entities"; import R from "escape-string-regexp"; import { JSDOM } from "jsdom"; import { promises as fs } from "fs"; import writeFileAtomic from "write-file-atomic"; import cryptoRandomString from "crypto-random-string"; import html from "tagged-template-noop"; export const WEB_PORT = process.env.WEB_PORT ?? 8000; export const EMAIL_PORT = process.env.EMAIL_PORT ?? 2525; export const BASE_URL = process.env.BASE_URL ?? "http://localhost:8000"; export const EMAIL_DOMAIN = process.env.EMAIL_DOMAIN ?? "localhost"; export const ISSUE_REPORT = process.env.ISSUE_REPORT ?? "mailto:kill-the-newsletter@leafac.com"; export const webServer = express() .use(express.static("static")) .use(express.urlencoded({ extended: true })) .get("/", (req, res) => res.send(layout(newInbox()))) .post("/", async (req, res, next) => { try { const { name } = req.body; const identifier = createIdentifier(); const renderedCreated = created(identifier); await writeFileAtomic( feedFilePath(identifier), feed( identifier, X(name), entry( identifier, createIdentifier(), `“${X(name)}” Inbox Created`, "Kill the Newsletter!", X(renderedCreated) ) ) ); res.send( layout(html`
“${H(name)}” Inbox Created
${renderedCreated} `) ); } catch (error) { console.error(error); next(error); } }) .get( alternatePath(":feedIdentifier", ":entryIdentifier"), async (req, res, next) => { try { const { feedIdentifier, entryIdentifier } = req.params; const path = feedFilePath(feedIdentifier); let text; try { text = await fs.readFile(path, "utf8"); } catch { return res.sendStatus(404); } const feed = new JSDOM(text, { contentType: "text/xml" }); const document = feed.window.document; const link = document.querySelector( `link[href="${alternateURL(feedIdentifier, entryIdentifier)}"]` ); if (link === null) return res.sendStatus(404); res.send( entities.decodeXML( link.parentElement!.querySelector("content")!.textContent! ) ); } catch (error) { console.error(error); next(error); } } ) .listen(WEB_PORT); export const emailServer = new SMTPServer({ disabledCommands: ["AUTH", "STARTTLS"], async onData(stream, session, callback) { try { const email = await mailparser.simpleParser(stream); const content = typeof email.html === "string" ? email.html : email.textAsHtml ?? ""; for (const address of new Set( session.envelope.rcptTo.map(({ address }) => address) )) { const match = address.match( new RegExp(`^(?Convert email newsletters into Atom feeds
Sign up for the newsletter with${feedEmail(identifier)}
Subscribe to the Atom feed at${feedURL(identifier)}
Don’t share these addresses.
They contain an identifier that other
people could use
to send you spam and to control your newsletter
subscriptions.
Enjoy your readings!
`.trim(); } function feed(identifier: string, name: string, initialEntry: string): string { return html`