mirror of
https://github.com/waynesutton/markdown-site.git
synced 2026-01-11 20:08:57 +00:00
Add missing changelog entries to content/pages/changelog-page.md: v1.34.0 (2025-12-26): Blog page featured layout with hero post - blogFeatured frontmatter field for posts - Hero card displays first featured post with landscape image - 2-column featured row for remaining featured posts - 3-column grid for regular posts v1.35.0 (2025-12-26): Image support at top of posts and pages - showImageAtTop frontmatter field - Full-width image display above post header - Works for both posts and pages v1.36.0 (2025-12-27): Social footer component - Customizable social links (8 platform types) - Copyright with auto-updating year - showSocialFooter frontmatter field for per-page control - Configurable via siteConfig.socialFooter v1.37.0 (2025-12-27): Newsletter Admin UI - Three-column admin interface at /newsletter-admin - Subscriber management with search and filters - Send newsletter panel (post selection or custom email) - Weekly digest automation (Sunday 9am UTC) - Developer notifications (subscriber alerts, weekly stats) - Markdown-to-HTML conversion for custom emails
102 lines
2.8 KiB
TypeScript
102 lines
2.8 KiB
TypeScript
#!/usr/bin/env npx ts-node
|
|
|
|
/**
|
|
* Send newsletter for a specific post to all subscribers
|
|
*
|
|
* Usage:
|
|
* npm run newsletter:send <post-slug>
|
|
*
|
|
* Example:
|
|
* npm run newsletter:send setup-guide
|
|
*
|
|
* Environment variables (from .env.local):
|
|
* - VITE_CONVEX_URL: Convex deployment URL
|
|
* - SITE_URL: Your site URL (default: https://markdown.fast)
|
|
* - SITE_NAME: Your site name (default: "Newsletter")
|
|
*
|
|
* Note: AGENTMAIL_API_KEY and AGENTMAIL_INBOX must be set in Convex dashboard
|
|
*/
|
|
|
|
import { ConvexHttpClient } from "convex/browser";
|
|
import { api } from "../convex/_generated/api";
|
|
import dotenv from "dotenv";
|
|
|
|
// Load environment variables
|
|
dotenv.config({ path: ".env.local" });
|
|
dotenv.config({ path: ".env.production.local" });
|
|
|
|
async function main() {
|
|
const postSlug = process.argv[2];
|
|
|
|
if (!postSlug) {
|
|
console.error("Usage: npm run newsletter:send <post-slug>");
|
|
console.error("Example: npm run newsletter:send setup-guide");
|
|
process.exit(1);
|
|
}
|
|
|
|
const convexUrl = process.env.VITE_CONVEX_URL;
|
|
if (!convexUrl) {
|
|
console.error(
|
|
"Error: VITE_CONVEX_URL not found. Run 'npx convex dev' to create .env.local"
|
|
);
|
|
process.exit(1);
|
|
}
|
|
|
|
const siteUrl = process.env.SITE_URL || "https://markdown.fast";
|
|
const siteName = process.env.SITE_NAME || "Newsletter";
|
|
|
|
console.log(`Sending newsletter for post: ${postSlug}`);
|
|
console.log(`Site URL: ${siteUrl}`);
|
|
console.log(`Site name: ${siteName}`);
|
|
console.log("");
|
|
|
|
const client = new ConvexHttpClient(convexUrl);
|
|
|
|
try {
|
|
// First check if post exists
|
|
const post = await client.query(api.posts.getPostBySlug, { slug: postSlug });
|
|
if (!post) {
|
|
console.error(`Error: Post "${postSlug}" not found or not published.`);
|
|
process.exit(1);
|
|
}
|
|
|
|
console.log(`Found post: "${post.title}"`);
|
|
|
|
// Get subscriber count first
|
|
const subscriberCount = await client.query(api.newsletter.getSubscriberCount);
|
|
console.log(`Active subscribers: ${subscriberCount}`);
|
|
|
|
if (subscriberCount === 0) {
|
|
console.log("No subscribers to send to.");
|
|
process.exit(0);
|
|
}
|
|
|
|
console.log("");
|
|
console.log("Sending newsletter...");
|
|
|
|
// Call the mutation directly to schedule the newsletter send
|
|
const result = await client.mutation(api.newsletter.scheduleSendPostNewsletter, {
|
|
postSlug,
|
|
siteUrl,
|
|
siteName,
|
|
});
|
|
|
|
if (result.success) {
|
|
console.log("✓ Newsletter scheduled successfully!");
|
|
console.log(result.message);
|
|
console.log("");
|
|
console.log("The newsletter is being sent in the background.");
|
|
console.log("Check the Newsletter Admin page or recent sends to see results.");
|
|
} else {
|
|
console.error("✗ Failed to send newsletter:");
|
|
console.error(result.message);
|
|
process.exit(1);
|
|
}
|
|
} catch (error) {
|
|
console.error("Error:", error);
|
|
process.exit(1);
|
|
}
|
|
}
|
|
|
|
main();
|