mirror of
https://github.com/juherr/kill-the-news.git
synced 2026-06-20 22:03:48 +00:00
fe5728de59
Wire the NativeFeeds chip group into the per-feed emails page, add a dismissable banner that nudges users to subscribe directly, the dismiss POST route mirroring the confirmation-dismiss idiom, and the client-side handler in emails-page.ts. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
892 lines
31 KiB
TypeScript
892 lines
31 KiB
TypeScript
import { Hono } from "hono";
|
|
import { Env, EmailMetadata } from "../../types";
|
|
import { logger } from "../../infrastructure/logger";
|
|
import {
|
|
Layout,
|
|
clampText,
|
|
CopyIcon,
|
|
CheckIcon,
|
|
FeedFormats,
|
|
ExpiryBadge,
|
|
NativeFeeds,
|
|
} from "./ui";
|
|
import { unionNativeFeeds } from "../../domain/native-feed";
|
|
import {
|
|
deleteAttachmentsForEmails,
|
|
deleteKeysWithConcurrency,
|
|
} from "../../application/feed-cleanup";
|
|
import { FeedRepository } from "../../infrastructure/feed-repository";
|
|
import { FeedId } from "../../domain/value-objects/feed-id";
|
|
import {
|
|
feedEmailAddress,
|
|
baseUrl,
|
|
entryPath,
|
|
} from "../../infrastructure/urls";
|
|
import { processEmailContent } from "../../infrastructure/html-processor";
|
|
import { formatBytes } from "../../domain/format";
|
|
import { EmailAddress } from "../../domain/value-objects/email-address";
|
|
import { emailsPageScript } from "../../scripts/generated/emails-page";
|
|
import emailPreviewCss from "../../styles/email-preview.css";
|
|
|
|
type AppEnv = { Bindings: Env };
|
|
|
|
export const emailsRouter = new Hono<AppEnv>();
|
|
|
|
type CopyFieldProps = {
|
|
label: string;
|
|
value: string;
|
|
display?: string;
|
|
};
|
|
|
|
const CopyField = ({ label, value, display }: CopyFieldProps) => (
|
|
<div class="copyable">
|
|
<span class="copyable-label">{label}</span>
|
|
<div class="copyable-content">
|
|
<span class="copyable-value" data-copy={value}>
|
|
{display ?? value}
|
|
</span>
|
|
<div class="copy-icon-container">
|
|
<CopyIcon />
|
|
<CheckIcon />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
|
|
type SenderFieldProps = {
|
|
from: string;
|
|
feedId: string;
|
|
};
|
|
|
|
const SenderField = ({ from, feedId }: SenderFieldProps) => {
|
|
const parsed = EmailAddress.parse(from);
|
|
const senderEmail = parsed?.normalized ?? from.trim().toLowerCase();
|
|
const senderDomain = parsed?.domain.value ?? "";
|
|
|
|
return (
|
|
<div class="copyable">
|
|
<span class="copyable-label">From:</span>
|
|
<div class="copyable-content">
|
|
<span class="copyable-value" data-copy={from}>
|
|
{from}
|
|
</span>
|
|
<div class="copy-icon-container">
|
|
<CopyIcon />
|
|
<CheckIcon />
|
|
</div>
|
|
</div>
|
|
<div
|
|
class="sender-filter-dropdown"
|
|
data-feed-id={feedId}
|
|
data-email={senderEmail}
|
|
data-domain={senderDomain}
|
|
>
|
|
<button
|
|
type="button"
|
|
class="sender-filter-btn"
|
|
aria-label="Sender filter options"
|
|
onclick="toggleSenderFilter(this)"
|
|
>
|
|
⋮
|
|
</button>
|
|
<div class="sender-filter-menu" role="menu">
|
|
<button
|
|
type="button"
|
|
class="sender-filter-item sender-filter-allow"
|
|
data-action="allow_sender"
|
|
>
|
|
Allow {senderEmail}
|
|
</button>
|
|
<button
|
|
type="button"
|
|
class="sender-filter-item sender-filter-allow"
|
|
data-action="allow_domain"
|
|
>
|
|
Allow @{senderDomain}
|
|
</button>
|
|
<button
|
|
type="button"
|
|
class="sender-filter-item sender-filter-block"
|
|
data-action="block_sender"
|
|
>
|
|
Block {senderEmail}
|
|
</button>
|
|
<button
|
|
type="button"
|
|
class="sender-filter-item sender-filter-block"
|
|
data-action="block_domain"
|
|
>
|
|
Block @{senderDomain}
|
|
</button>
|
|
</div>
|
|
<span class="sender-filter-feedback" aria-live="polite"></span>
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
// ── View all emails for a feed ────────────────────────────────────────────────
|
|
|
|
emailsRouter.get("/feeds/:feedId/emails", async (c) => {
|
|
const env = c.env;
|
|
const repo = FeedRepository.from(env);
|
|
const feedId = c.req.param("feedId");
|
|
const message = c.req.query("message");
|
|
const count = Number(c.req.query("count") || "0");
|
|
|
|
const id = FeedId.unchecked(feedId);
|
|
const feedConfig = await repo.getConfig(id);
|
|
const feedMetadata = await repo.getMetadata(id);
|
|
|
|
if (!feedConfig || !feedMetadata) {
|
|
return c.text("Feed not found", 404);
|
|
}
|
|
|
|
const nativeFeeds = unionNativeFeeds(feedMetadata.nativeFeeds);
|
|
const emailAddress = feedEmailAddress(feedConfig.mailbox_id, env);
|
|
|
|
return c.html(
|
|
<Layout title={`${feedConfig.title} - Emails`}>
|
|
<div class="container container-wide fade-in">
|
|
<div class="header-with-actions">
|
|
<div class="header-title">
|
|
<h1>{feedConfig.title} - Emails</h1>
|
|
</div>
|
|
<div class="header-actions">
|
|
<a href="/admin" class="button button-secondary button-back">
|
|
Back to Dashboard
|
|
</a>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="card">
|
|
{feedConfig.expires_at && (
|
|
<div class="feed-header">
|
|
<ExpiryBadge expiresAt={feedConfig.expires_at} />
|
|
</div>
|
|
)}
|
|
<CopyField label="Email:" value={emailAddress} />
|
|
<FeedFormats feedId={feedId} env={env} />
|
|
<NativeFeeds feeds={nativeFeeds} />
|
|
</div>
|
|
|
|
{feedMetadata.pendingConfirmation && (
|
|
<div
|
|
class="confirmation-banner"
|
|
id="confirmation-banner"
|
|
data-feed-id={feedId}
|
|
>
|
|
<span>A subscription-confirmation email was detected.</span>
|
|
<div class="confirmation-banner-actions">
|
|
<button
|
|
type="button"
|
|
class="button button-small"
|
|
id="confirmation-dismiss"
|
|
>
|
|
Mark as confirmed
|
|
</button>
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
{nativeFeeds.length > 0 && !feedMetadata.nativeFeedDismissed && (
|
|
<div
|
|
class="confirmation-banner"
|
|
id="native-feed-banner"
|
|
data-feed-id={feedId}
|
|
>
|
|
<span>
|
|
This newsletter publishes its own feed — subscribe to it directly
|
|
from "Native feeds" above.
|
|
</span>
|
|
<div class="confirmation-banner-actions">
|
|
<button
|
|
type="button"
|
|
class="button button-small"
|
|
id="native-feed-dismiss"
|
|
>
|
|
Dismiss
|
|
</button>
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
<h2>
|
|
Emails (
|
|
<span id="email-total-count">{feedMetadata.emails.length}</span>)
|
|
</h2>
|
|
|
|
{message === "bulkDeleted" && (
|
|
<div class="card">
|
|
<p>Deleted {Number.isFinite(count) ? count : 0} email(s).</p>
|
|
</div>
|
|
)}
|
|
{message === "bulkDeleteNoop" && (
|
|
<div class="card">
|
|
<p>No emails were selected.</p>
|
|
</div>
|
|
)}
|
|
{feedMetadata.emails.length > 0 ? (
|
|
<form
|
|
action={`/admin/feeds/${feedId}/emails/bulk-delete`}
|
|
method="post"
|
|
onsubmit="return onBulkEmailDeleteSubmit(event)"
|
|
>
|
|
<div class="toolbar">
|
|
<div class="toolbar-group toolbar-group-fill">
|
|
<input
|
|
type="search"
|
|
id="email-search"
|
|
class="search"
|
|
placeholder="Search email subjects"
|
|
oninput="scheduleEmailFilter()"
|
|
/>
|
|
<span class="pill" id="email-match-count">
|
|
Showing {feedMetadata.emails.length}
|
|
</span>
|
|
<span class="pill" id="selected-email-count">
|
|
0 selected
|
|
</span>
|
|
<button
|
|
type="button"
|
|
class="button button-small button-secondary"
|
|
onclick="selectMatchingEmails()"
|
|
>
|
|
Select Results
|
|
</button>
|
|
<button
|
|
type="button"
|
|
class="button button-small button-secondary"
|
|
onclick="clearEmailSelection()"
|
|
>
|
|
Clear Selection
|
|
</button>
|
|
<button
|
|
id="bulk-delete-emails-button"
|
|
type="submit"
|
|
class="button button-small button-danger"
|
|
disabled
|
|
>
|
|
Delete Selected
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="table-wrap">
|
|
<table class="table table-emails">
|
|
<colgroup>
|
|
<col data-col="select" style="width: 44px;" />
|
|
<col data-col="subject" style="width: 520px;" />
|
|
<col data-col="receivedAt" style="width: 220px;" />
|
|
<col data-col="actions" style="width: 200px;" />
|
|
</colgroup>
|
|
<thead>
|
|
<tr>
|
|
<th>
|
|
<input
|
|
type="checkbox"
|
|
id="select-all-emails"
|
|
onchange="toggleAllEmails(this.checked)"
|
|
/>
|
|
</th>
|
|
<th
|
|
class="th-resizable"
|
|
data-sort-key="subject"
|
|
aria-sort="none"
|
|
>
|
|
<button
|
|
type="button"
|
|
class="th-button"
|
|
data-sort-key="subject"
|
|
>
|
|
Subject
|
|
<span class="sort-indicator" aria-hidden="true"></span>
|
|
</button>
|
|
<div
|
|
class="col-resizer"
|
|
data-col="subject"
|
|
title="Resize"
|
|
></div>
|
|
</th>
|
|
<th
|
|
class="th-resizable"
|
|
data-sort-key="receivedAt"
|
|
aria-sort="none"
|
|
>
|
|
<button
|
|
type="button"
|
|
class="th-button"
|
|
data-sort-key="receivedAt"
|
|
>
|
|
Received
|
|
<span class="sort-indicator" aria-hidden="true"></span>
|
|
</button>
|
|
<div
|
|
class="col-resizer"
|
|
data-col="receivedAt"
|
|
title="Resize"
|
|
></div>
|
|
</th>
|
|
<th class="th-resizable">
|
|
<span>Actions</span>
|
|
<div
|
|
class="col-resizer"
|
|
data-col="actions"
|
|
title="Resize"
|
|
></div>
|
|
</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{feedMetadata.emails.map((email: EmailMetadata) => {
|
|
const subjectDisplay = clampText(email.subject, 180);
|
|
const subjectHover = clampText(email.subject, 1000);
|
|
const attachmentCount = email.attachmentIds?.length ?? 0;
|
|
const attachmentLabel = `${attachmentCount} attachment${
|
|
attachmentCount > 1 ? "s" : ""
|
|
}`;
|
|
const isConfirmation = !!email.confirmation;
|
|
const sortSubject = subjectHover.toLowerCase();
|
|
const sortReceivedAt = String(email.receivedAt);
|
|
const searchHaystack = clampText(
|
|
email.subject,
|
|
320,
|
|
).toLowerCase();
|
|
return (
|
|
<tr
|
|
class="email-row"
|
|
data-email-key={email.key}
|
|
data-search={searchHaystack}
|
|
data-sort-subject={sortSubject}
|
|
data-sort-received-at={sortReceivedAt}
|
|
>
|
|
<td>
|
|
<input
|
|
type="checkbox"
|
|
class="email-select"
|
|
name="emailKeys"
|
|
value={email.key}
|
|
onchange="updateEmailSelectionState()"
|
|
/>
|
|
</td>
|
|
<td>
|
|
<div class="subject-cell">
|
|
{attachmentCount > 0 ? (
|
|
<span
|
|
class="attachment-indicator"
|
|
title={attachmentLabel}
|
|
aria-label={attachmentLabel}
|
|
>
|
|
<svg
|
|
width="14"
|
|
height="14"
|
|
viewBox="0 0 24 24"
|
|
fill="none"
|
|
stroke="currentColor"
|
|
stroke-width="2"
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
aria-hidden="true"
|
|
>
|
|
<path d="M21.44 11.05l-9.19 9.19a6 6 0 0 1-8.49-8.49l9.19-9.19a4 4 0 0 1 5.66 5.66l-9.2 9.19a2 2 0 0 1-2.83-2.83l8.49-8.48" />
|
|
</svg>
|
|
</span>
|
|
) : null}
|
|
{isConfirmation ? (
|
|
<span
|
|
class="confirmation-badge"
|
|
title="Subscription confirmation"
|
|
>
|
|
Confirmation
|
|
</span>
|
|
) : null}
|
|
<span class="truncate" title={subjectHover}>
|
|
{subjectDisplay}
|
|
</span>
|
|
</div>
|
|
</td>
|
|
<td>{new Date(email.receivedAt).toLocaleString()}</td>
|
|
<td>
|
|
<div class="row-actions">
|
|
<a
|
|
href={`/admin/emails/${email.key}`}
|
|
class="button button-small"
|
|
>
|
|
View
|
|
</a>
|
|
<button
|
|
type="button"
|
|
class="button button-small button-danger button-delete"
|
|
data-delete-kind="email"
|
|
data-email-key={email.key}
|
|
data-feed-id={feedId}
|
|
>
|
|
Delete
|
|
</button>
|
|
</div>
|
|
</td>
|
|
</tr>
|
|
);
|
|
})}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</form>
|
|
) : (
|
|
<div class="card">
|
|
<p>
|
|
No emails received yet. Subscribe to newsletters using the email
|
|
address above.
|
|
</p>
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{/* Config bootstrap — injects dynamic server-side data before the static compiled script */}
|
|
<script
|
|
dangerouslySetInnerHTML={{
|
|
__html: `window.__APP_CONFIG__=${JSON.stringify({ feedId })}`,
|
|
}}
|
|
/>
|
|
{/* Emails page logic compiled from src/scripts/client/emails-page.ts */}
|
|
<script dangerouslySetInnerHTML={{ __html: emailsPageScript }} />
|
|
</Layout>,
|
|
);
|
|
});
|
|
|
|
// ── View single email ─────────────────────────────────────────────────────────
|
|
|
|
emailsRouter.get("/emails/:emailKey", async (c) => {
|
|
const env = c.env;
|
|
const repo = FeedRepository.from(env);
|
|
const emailKey = c.req.param("emailKey");
|
|
|
|
const emailData = await repo.getEmail(emailKey);
|
|
|
|
if (!emailData) return c.text("Email not found", 404);
|
|
|
|
const feedId = repo.feedIdFromEmailKey(emailKey);
|
|
const feedConfig = await repo.getConfig(FeedId.unchecked(feedId));
|
|
if (!feedConfig) return c.text("Feed not found", 404);
|
|
|
|
const feedMetadata = await repo.getMetadata(FeedId.unchecked(feedId));
|
|
const confirmationLinks =
|
|
feedMetadata?.emails.find((e) => e.key === emailKey)?.confirmation?.links ??
|
|
[];
|
|
// Inline images render in place; only downloadable attachments go in the list.
|
|
const attachments = (emailData.attachments ?? []).filter((a) => !a.inline);
|
|
|
|
// The rendered preview lives in a `data:` iframe, which has no origin to
|
|
// resolve relative URLs against — so cid: refs must be rewritten to absolute
|
|
// /files URLs (and the content sanitized) before embedding.
|
|
const renderedBody = processEmailContent(
|
|
emailData.content,
|
|
emailData.attachments,
|
|
baseUrl(env),
|
|
);
|
|
const htmlContent = `<!DOCTYPE html><html><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><style>${emailPreviewCss}</style></head><body>${renderedBody}</body></html>`;
|
|
|
|
const encodedHtmlContent = (() => {
|
|
const encoder = new TextEncoder();
|
|
const bytes = encoder.encode(htmlContent);
|
|
return btoa(String.fromCharCode(...new Uint8Array(bytes)));
|
|
})();
|
|
|
|
const rawHtml = emailData.content.replace(/</g, "<").replace(/>/g, ">");
|
|
|
|
const viewScript = `
|
|
function showRendered() {
|
|
document.getElementById('rendered-view').style.display = 'block';
|
|
document.getElementById('raw-view').style.display = 'none';
|
|
document.getElementById('rendered-button').classList.add('active');
|
|
document.getElementById('raw-button').classList.remove('active');
|
|
}
|
|
function showRaw() {
|
|
document.getElementById('rendered-view').style.display = 'none';
|
|
document.getElementById('raw-view').style.display = 'block';
|
|
document.getElementById('rendered-button').classList.remove('active');
|
|
document.getElementById('raw-button').classList.add('active');
|
|
}
|
|
window.addEventListener('load', function() {
|
|
const iframe = document.querySelector('.email-iframe');
|
|
if (!iframe) return;
|
|
iframe.style.height = '500px';
|
|
try {
|
|
iframe.onload = function() {
|
|
const doc = iframe.contentDocument || iframe.contentWindow.document;
|
|
if (doc) iframe.style.height = Math.min(800, Math.max(500, doc.body.scrollHeight)) + 'px';
|
|
};
|
|
} catch (e) { /* cross-origin */ }
|
|
});
|
|
function toggleSenderFilter(btn) {
|
|
var dropdown = btn.closest('.sender-filter-dropdown');
|
|
var isOpen = dropdown.hasAttribute('data-open');
|
|
document.querySelectorAll('.sender-filter-dropdown[data-open]').forEach(function(d) {
|
|
d.removeAttribute('data-open');
|
|
});
|
|
if (!isOpen) dropdown.setAttribute('data-open', '');
|
|
}
|
|
function showSenderFeedback(feedback, ok, msg) {
|
|
feedback.textContent = msg;
|
|
feedback.className = 'sender-filter-feedback ' + (ok ? 'sender-filter-feedback-ok' : 'sender-filter-feedback-error');
|
|
setTimeout(function() {
|
|
feedback.textContent = '';
|
|
feedback.className = 'sender-filter-feedback';
|
|
}, 3000);
|
|
}
|
|
document.addEventListener('click', function(e) {
|
|
var item = e.target.closest('.sender-filter-item');
|
|
if (item) {
|
|
var dropdown = item.closest('.sender-filter-dropdown');
|
|
var action = item.dataset.action;
|
|
var value = (action === 'allow_sender' || action === 'block_sender')
|
|
? dropdown.dataset.email
|
|
: dropdown.dataset.domain;
|
|
dropdown.removeAttribute('data-open');
|
|
var feedback = dropdown.querySelector('.sender-filter-feedback');
|
|
fetch('/admin/feeds/' + dropdown.dataset.feedId + '/sender-filter', {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify({ action: action, value: value })
|
|
}).then(function(res) {
|
|
return res.json();
|
|
}).then(function(data) {
|
|
showSenderFeedback(feedback, data.ok, data.ok ? 'Saved' : (data.error || 'Error'));
|
|
}).catch(function() {
|
|
showSenderFeedback(feedback, false, 'Network error');
|
|
});
|
|
return;
|
|
}
|
|
if (!e.target.closest('.sender-filter-dropdown')) {
|
|
document.querySelectorAll('.sender-filter-dropdown[data-open]').forEach(function(d) {
|
|
d.removeAttribute('data-open');
|
|
});
|
|
}
|
|
});
|
|
`;
|
|
|
|
return c.html(
|
|
<Layout title={`Email: ${emailData.subject}`}>
|
|
<div class="container fade-in">
|
|
<div class="header-with-actions">
|
|
<div class="header-title">
|
|
<h1>Email Content</h1>
|
|
</div>
|
|
<div class="header-actions">
|
|
<a
|
|
href={`/admin/feeds/${feedId}/emails`}
|
|
class="button button-secondary button-back"
|
|
>
|
|
Back to Emails
|
|
</a>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="card">
|
|
<div class="email-meta">
|
|
<div class="email-metadata-grid">
|
|
<CopyField label="Subject:" value={emailData.subject} />
|
|
<CopyField
|
|
label="Received:"
|
|
value={new Date(emailData.receivedAt).toLocaleString()}
|
|
/>
|
|
<SenderField from={emailData.from} feedId={feedId} />
|
|
<CopyField
|
|
label="To:"
|
|
value={feedEmailAddress(feedConfig.mailbox_id, env)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
{confirmationLinks.length > 0 && (
|
|
<div class="confirmation-section">
|
|
<h2>Confirm your subscription</h2>
|
|
<p class="muted">
|
|
This looks like a subscription-confirmation email. Open the link
|
|
to confirm.
|
|
</p>
|
|
<a
|
|
class="button confirmation-primary"
|
|
href={confirmationLinks[0]}
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
>
|
|
Confirm subscription
|
|
</a>
|
|
<div class="confirmation-links">
|
|
{confirmationLinks.map((link) => (
|
|
<a href={link} target="_blank" rel="noopener noreferrer">
|
|
{link}
|
|
</a>
|
|
))}
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
<div class="toggle-view">
|
|
<button
|
|
id="rendered-button"
|
|
class="toggle-button active"
|
|
onclick="showRendered()"
|
|
>
|
|
Rendered View
|
|
</button>
|
|
<button id="raw-button" class="toggle-button" onclick="showRaw()">
|
|
Raw HTML
|
|
</button>
|
|
<a
|
|
class="toggle-view-link"
|
|
href={entryPath(feedId, emailData.receivedAt)}
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
>
|
|
Public page ↗
|
|
</a>
|
|
</div>
|
|
|
|
<div class="email-content">
|
|
<div id="rendered-view" class="email-iframe-container">
|
|
<iframe
|
|
class="email-iframe"
|
|
src={`data:text/html;base64,${encodedHtmlContent}`}
|
|
></iframe>
|
|
</div>
|
|
<div id="raw-view" class="email-raw" style="display: none;">
|
|
<pre dangerouslySetInnerHTML={{ __html: rawHtml }}></pre>
|
|
</div>
|
|
</div>
|
|
|
|
{attachments.length > 0 && (
|
|
<div class="attachments">
|
|
<h2>Attachments</h2>
|
|
<ul class="attachment-list">
|
|
{attachments.map((a) => (
|
|
<li>
|
|
<svg
|
|
width="14"
|
|
height="14"
|
|
viewBox="0 0 24 24"
|
|
fill="none"
|
|
stroke="currentColor"
|
|
stroke-width="2"
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
aria-hidden="true"
|
|
>
|
|
<path d="M21.44 11.05l-9.19 9.19a6 6 0 0 1-8.49-8.49l9.19-9.19a4 4 0 0 1 5.66 5.66l-9.2 9.19a2 2 0 0 1-2.83-2.83l8.49-8.48" />
|
|
</svg>
|
|
<a
|
|
href={`/files/${a.id}/${encodeURIComponent(a.filename)}`}
|
|
download
|
|
>
|
|
{a.filename}
|
|
</a>
|
|
<span class="attachment-size">{formatBytes(a.size)}</span>
|
|
</li>
|
|
))}
|
|
</ul>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
|
|
<script dangerouslySetInnerHTML={{ __html: viewScript }} />
|
|
</Layout>,
|
|
);
|
|
});
|
|
|
|
// ── Delete single email ───────────────────────────────────────────────────────
|
|
|
|
emailsRouter.post("/emails/:emailKey/delete", async (c) => {
|
|
const env = c.env;
|
|
const repo = FeedRepository.from(env);
|
|
const emailKey = c.req.param("emailKey");
|
|
const wantsJson = (c.req.header("Accept") || "").includes("application/json");
|
|
|
|
try {
|
|
const feedId = c.req.query("feedId");
|
|
if (!feedId) {
|
|
if (wantsJson)
|
|
return c.json({ ok: false, error: "Feed ID is required" }, 400);
|
|
return c.text("Feed ID is required", 400);
|
|
}
|
|
|
|
const feed = await repo.load(FeedId.unchecked(feedId));
|
|
|
|
await repo.deleteEmail(emailKey);
|
|
if (feed) {
|
|
const { removed } = feed.removeEmails([emailKey]);
|
|
await deleteAttachmentsForEmails(env, removed, [emailKey]);
|
|
await repo.saveMetadata(feed);
|
|
}
|
|
|
|
if (wantsJson) return c.json({ ok: true, emailKey, feedId });
|
|
return c.redirect(`/admin/feeds/${feedId}/emails`);
|
|
} catch (error) {
|
|
logger.error("Error deleting email", { emailKey, error: String(error) });
|
|
if (wantsJson)
|
|
return c.json(
|
|
{ ok: false, error: "Error deleting email. Please try again." },
|
|
400,
|
|
);
|
|
return c.text("Error deleting email. Please try again.", 400);
|
|
}
|
|
});
|
|
|
|
// ── Dismiss confirmation ──────────────────────────────────────────────────────
|
|
|
|
emailsRouter.post("/feeds/:feedId/confirmation/dismiss", async (c) => {
|
|
const env = c.env;
|
|
const repo = FeedRepository.from(env);
|
|
const feedId = c.req.param("feedId");
|
|
const wantsJson = (
|
|
c.req.header("Accept") ||
|
|
c.req.header("Content-Type") ||
|
|
""
|
|
).includes("application/json");
|
|
|
|
const feed = await repo.load(FeedId.unchecked(feedId));
|
|
if (!feed) {
|
|
return wantsJson
|
|
? c.json({ ok: false, error: "Feed not found" }, 404)
|
|
: c.text("Feed not found", 404);
|
|
}
|
|
feed.dismissConfirmation();
|
|
await repo.saveMetadata(feed);
|
|
|
|
return wantsJson
|
|
? c.json({ ok: true })
|
|
: c.redirect(`/admin/feeds/${feedId}/emails`);
|
|
});
|
|
|
|
// ── Dismiss native-feed notice ───────────────────────────────────────────────
|
|
|
|
emailsRouter.post("/feeds/:feedId/native-feed/dismiss", async (c) => {
|
|
const env = c.env;
|
|
const repo = FeedRepository.from(env);
|
|
const feedId = c.req.param("feedId");
|
|
const wantsJson = (
|
|
c.req.header("Accept") ||
|
|
c.req.header("Content-Type") ||
|
|
""
|
|
).includes("application/json");
|
|
|
|
const feed = await repo.load(FeedId.unchecked(feedId));
|
|
if (!feed) {
|
|
return wantsJson
|
|
? c.json({ ok: false, error: "Feed not found" }, 404)
|
|
: c.text("Feed not found", 404);
|
|
}
|
|
feed.dismissNativeFeed();
|
|
await repo.saveMetadata(feed);
|
|
|
|
return wantsJson
|
|
? c.json({ ok: true })
|
|
: c.redirect(`/admin/feeds/${feedId}/emails`);
|
|
});
|
|
|
|
// ── Bulk delete emails ────────────────────────────────────────────────────────
|
|
|
|
emailsRouter.post("/feeds/:feedId/emails/bulk-delete", async (c) => {
|
|
const env = c.env;
|
|
const emailStorage = env.EMAIL_STORAGE;
|
|
const repo = new FeedRepository(emailStorage);
|
|
const feedId = c.req.param("feedId");
|
|
const contentType = c.req.header("Content-Type") || "";
|
|
const wantsJson =
|
|
contentType.includes("application/json") ||
|
|
(c.req.header("Accept") || "").includes("application/json");
|
|
|
|
try {
|
|
const feed = await repo.load(FeedId.unchecked(feedId));
|
|
|
|
if (!feed) {
|
|
return wantsJson
|
|
? c.json({ ok: false, error: "Feed not found" }, 404)
|
|
: c.text("Feed not found", 404);
|
|
}
|
|
|
|
const allowedKeys = new Set(feed.emails.map((email) => email.key));
|
|
|
|
if (wantsJson) {
|
|
const body = (await c.req.json().catch(() => null)) as {
|
|
emailKeys?: unknown;
|
|
} | null;
|
|
const rawEmailKeys = Array.isArray(body?.emailKeys)
|
|
? body?.emailKeys
|
|
: [];
|
|
const emailKeys = Array.from(
|
|
new Set(rawEmailKeys.map((value) => String(value)).filter(Boolean)),
|
|
);
|
|
|
|
if (emailKeys.length === 0)
|
|
return c.json({ ok: false, error: "No emails were selected." }, 400);
|
|
if (emailKeys.length > 250) {
|
|
return c.json(
|
|
{
|
|
ok: false,
|
|
error:
|
|
"Too many emailKeys for a single request. Please delete in smaller batches.",
|
|
},
|
|
413,
|
|
);
|
|
}
|
|
|
|
const candidates = emailKeys.filter((key) => allowedKeys.has(key));
|
|
|
|
const { ok: deletedOk, failed: failedEmailKeys } =
|
|
await deleteKeysWithConcurrency(emailStorage, candidates, 35);
|
|
await deleteAttachmentsForEmails(env, feed.emails, candidates);
|
|
|
|
feed.removeEmails(deletedOk);
|
|
await repo.saveMetadata(feed);
|
|
|
|
return c.json({
|
|
ok: failedEmailKeys.length === 0,
|
|
deletedEmailKeys: deletedOk,
|
|
failedEmailKeys,
|
|
});
|
|
}
|
|
|
|
const formData = await c.req.formData();
|
|
const rawEmailKeys = formData
|
|
.getAll("emailKeys")
|
|
.map((value) => value.toString());
|
|
const emailKeys = Array.from(new Set(rawEmailKeys.filter(Boolean)));
|
|
|
|
if (emailKeys.length === 0)
|
|
return c.redirect(`/admin/feeds/${feedId}/emails?message=bulkDeleteNoop`);
|
|
|
|
const candidates = emailKeys.filter((key) => allowedKeys.has(key));
|
|
|
|
const { ok: deletedOk } = await deleteKeysWithConcurrency(
|
|
emailStorage,
|
|
candidates,
|
|
35,
|
|
);
|
|
await deleteAttachmentsForEmails(env, feed.emails, candidates);
|
|
|
|
feed.removeEmails(deletedOk);
|
|
await repo.saveMetadata(feed);
|
|
|
|
return c.redirect(
|
|
`/admin/feeds/${feedId}/emails?message=bulkDeleted&count=${deletedOk.length}`,
|
|
);
|
|
} catch (error) {
|
|
logger.error("Error bulk deleting emails", {
|
|
feedId,
|
|
error: String(error),
|
|
});
|
|
return wantsJson
|
|
? c.json(
|
|
{
|
|
ok: false,
|
|
error:
|
|
"Server error while deleting emails. This can happen if Cloudflare is rate-limiting requests or if the Worker hit a plan quota. Please try again.",
|
|
},
|
|
500,
|
|
)
|
|
: c.text("Error bulk deleting emails. Please try again.", 500);
|
|
}
|
|
});
|