Growth hack: Publish newsletters to your Next.js site with ConvertKit API
I publish a newsletter called Tiny Improvements
My little newsletter is all about making small, incremental improvements to your work and home life, especially if you're building a product or company. I've been publishing it for a while now, and I've been using ConvertKit to manage my subscribers and send out the newsletter.
In this post, we'll go over how I use the ConvertKit API to publish a newsletter page for a site built with Next.js.
Publish past issues to grow your newsletter
I recently started publishing back-issues of my newsletter to my site, so that I can share them as evidence of the type of writing I do. In addition to being SEO-indexable, it allows me to share the newsletter with people who aren't subscribed to it, with a link to subscribe at the bottom of each issue.
I wanted to explore a way to make my process incrementally better, by using the ConvertKit API to automate the publishing process with the same workflow that I use to send newsletters to subscribers.
Using the ConvertKit API to publish newsletters to a Next.js site
Setting up an automated newsletter page is fairly simple, at least in theory:
- Write newsletters and send them, using ConvertKit's broadcast features (for the sake of this post, we'll use the term "newsletter" and "broadcast" somewhat interchangeably).
- Use the ConvertKit API's list-broadcasts endpoint to query for every published broadcast, to populate a newsletter index page
- For each published broadcast, use the ConvertKit API to get the details of that broadcast, and then render the content of that broadcast on a page.
- For SEO purposes, add each published newsletter page to
sitemap.xml
, with all the appropriate metadata.
To get newsletters from the ConvertKit API, I set up 2 helper functions in /src/utils/convertKit.js
. First is getNewsletter()
, which takes a broadcast ID and returns the details of that broadcast.
1export const getNewsletter = async (broadcastId) => {2const res = await fetch(3`https://api.convertkit.com/v3/broadcasts/${broadcastId}?api_secret=${CONVERTKIT_API_SECRET}`4);56// rename "broadcast" to "newsletter" for consistency7const { broadcast: newsletter } = await res.json();89return newsletter;10};
The second is getAllNewsletters()
, which returns a list of all published broadcasts. In its simplest form, this function looks like this:
1export const getAllNewsletters = async () => {2const response = await fetch(3`https://api.convertkit.com/v3/broadcasts?api_secret=${CONVERTKIT_API_SECRET}`4);5const data = await response.json();67// rename "broadcasts" to "newsletters" for consistency8const { broadcasts: newsletters } = data;910return newsletters;11};
Keeping your ConvertKit API Secret key safe on Next.js sites
For this example, we're using CONVERTKIT_API_SECRET
environment variable only on the server-rendered parts of our site, because we do not want it to be exposed to the client. To make this value accessible only server-side, add it to a file called .env
or .env.local
, according to Next.js documentation. Make sure you add .env
and .env.local
to your .gitignore
file, so you don't accidentally commit it to your repository.
Importantly, this environment variable should only be exposed server-side, because we didn't name it with a NEXT_PUBLIC_
prefix - to read more about using environment variables to web clients, check out Next.js documentation.
Rendering the newsletter index page
Now that we have a way to get a list of all published newsletters, we can use that to render a page with a preview of each past issue. I created a new file called /src/pages/newsletter/index.js
to render the newsletter index page.
On this page, we use getStaticProps
to fetch our list of newsletters on the server, and then pass them to render function, to display a list of links to each newsletter page.
note: I stripped out the styling/presentation code from this example to keep it focused on data loading logic.
1import { Link } from 'next/link';2import { getAllNewsletters } from '../../utils/convertKit';34import slugify from '../../utils/slugify';56export const getStaticProps = async () => {7const newsletters = await getAllNewsletters();89return {10props: {11newsletters,12},13};14};1516const NewsletterIndexPage = ({ newsletters }) => {17return (18<>19<h1>All Newsletters</h1>20<ul>21{newsletters?.map((newsletter) => (22<li key={newsletter.id}>23<Link href={`/newsletter/${slugify(newsletter.subject)}`}>24{newsletter.subject} - {formatDate(newsletter.published_at)}25</Link>26</li>27))}28</ul>29</>30);31};3233export default NewsletterIndexPage;
You may have noticed that I'm also using a helper function called slugify
, which uses the npm package slugify
. This function takes a string and returns a skewer-case-version-of-it, which is useful for creating browser-friendly URLs from strings.
1import makeSlug from 'slugify';23export const slugify = (string) => {4return makeSlug(string, {5replacement: '-',6remove: /[\[\],?*+~.()'"!:@]/g,7strict: true,8lower: true,9});10};1112export default slugify;
This is an optional step that caused me a fair bit of heartache - the easiest way to link to individual newsletter pages is to use the broadcast ID, but I wanted to use a more human-readable URL, so I used slugify
to create a URL-friendly version of the newsletter subject.
The downsides of this approach are:
- each newsletter must have a unique URL
- there's not a way to efficiently map a URL to a broadcast ID, so we have to iterate over all broadcasts to find the one that matches the URL (which you'll see in the next section)
For now, I can live with this approach, since this computation is done once during build time, and until I have hundreds of newsletters, it shouldn't be too big of a performance hit.
Rendering a single newsletter page
Now that we have a page listing all published newsletters, we need to create a page template to render a single newsletter. I created a new file called /src/pages/newsletter/[subject].js
to render the newsletter index page (note, once again, that presentation and styling code is removed here for simplicify):
1import { useEffect } from 'react';2import {3broadcastTemplateParse,4getAllNewsletters,5} from '../../utils/convertKit';6import slugify from '../../utils/slugify';78export const getStaticPaths = async () => {9const newsletters = await getAllNewsletters();1011return {12paths: newsletters.map((newsletter) => ({13params: {14subject: slugify(newsletter.subject),15},16})),17fallback: false,18};19};2021export const getStaticProps = async (context) => {22const newsletters = await getAllNewsletters();2324const { subject } = context.params;25const slug = slugify(subject);2627const newsletter = newsletters.find(28(newsletter) => slugify(newsletter.subject) == slug29);3031return {32props: {33newsletter,34},35};36};3738const SingleNewsletterPage = ({ newsletter }) => {39const { subject } = newsletter;4041return (42<>43<h1>{subject}</h1>44<div45dangerouslySetInnerHTML={{46__html: broadcastTemplateParse({ template: newsletter.content }),47}}48/>49</>50);51};5253export default SingleNewsletterPage;
The getStaticPaths
function is used to generate a list of all possible URLs that should be pre-rendered. In this case, we're using the getAllNewsletters
function to fetch a list of all published newsletters, and then using the slugify
function to create a URL based on the newsletter's subject.
The getStaticProps
function is used to fetch the data for a single newsletter, based on the URL. We use the getAllNewsletters
function to fetch a list of all published newsletters, and then use newsletters.find()
to look for the newsletter whose subject line matches the URL for the current page when run through slugify
. This uses a JavaScript API called Array.find(), which is used to find single item in an array that matches a condition.
Render the body using dangersoulySetInnerHTML
To render the contents of the newsletter, we are using a scary-looking React API pattern - dangerouslySetInnerHTML
. This is a React API that allows you to render HTML that you have received from an API, and is generally considered a security risk. However, in this case, we are using it to render HTML that we have generated ourselves, and we are at least somewhat confident that it is safe to render. Generally speaking, this is a technique that shouldn't be used unless you are confident that the HTML you are rendering is safe.
Helper functions for displaying ConvertKit newsletter content
We're also using 2 helper functions, which are defined in /src/utils/convertKit.js
:
1import { Liquid } from 'liquidjs';2const engine = new Liquid();34export const broadcastTemplateParse = ({ template, data }) => {5const t = template.replaceAll('"', '"');6const res = engine.parseAndRenderSync(t, data);7return res;8};910export const newsletterHasValidThumbnail = (newsletter) => {11const { thumbnail_url: thumbnailUrl } = newsletter;12if (!thumbnailUrl) return false;13if (thumbnailUrl.startsWith('https://functions-js.convertkit.com/icons'))14return false;15return true;16};
The newsletterHasValidThumbnail
function is a helper function that checks if the newsletter has a valid thumbnail image. ConvertKit provides a default thumbnail image for all newsletters, but you can also upload your own, by adding an image to your newsletter's body. This function checks if the thumbnail is the default image, and if so, returns false
so we can skip rendering it.
The broadcastTemplateParse
function is a helper function that takes a newsletter template and parses it with the Liquid templating language. This is how ConvertKit allows you to customize the HTML of your newsletter, and use dynamic data like the subscriber's name, or the date the newsletter was sent.
An important caveat: There are lots of little cases where this might not work perfectly, due to all of the amazing things you can do with Liquid in ConvertKit. At the moment, my newsletter uses fairly basic liquid customization - this may not do the trick for you if you're using more advanced features. Please test your implementation thoroughly!
Adding more detail to the index page
At the time of writing this post, ConvertKit's v3 API is in "active development" - and as such, it's likely not feature complete. For example, the current API for list-broadcasts provides a minimal amount of data about each broadcast:
1curl https://api.convertkit.com/v3/broadcasts?api_secret=<your_secret_api_key>
1{2"broadcasts": [3{4"id": 1,5"created_at": "2014-02-13T21:45:16.000Z",6"subject": "Welcome to my Newsletter!"7},8{9"id": 2,10"created_at": "2014-02-20T11:40:11.000Z",11"subject": "Check out my latest blog posts!"12},13{14"id": 3,15"created_at": "2014-02-29T08:21:18.000Z",16"subject": "How to get my free masterclass"17}18]
Since we want to create an index page that shows a thumbnail for each newsletter, we need to query the API for more information about each newsletter. At the moment, the best way to do this is to query the API for each newsletter individually. It's not ideal, since we need to send an API call for each newsletter shown on the index page, but I'm hoping that with a bit of feedback, ConvertKit will add more data to the list-broadcasts API endpoint.
We already have a function to query the API for a single newsletter (remember getNewsletter
from earlier?), so we can use that to query the API for each newsletter on the index page.
Filter out un-published newsletters
The list-broadcasts endpoint also doesn't currently return whether or not a given broadcast was published yet - which means that if you draft a future broadcast on your ConvertKit dashboard, it will still be returned by the list API. I don't want to share these yet-to-be-sent newsletters on any of my sites, so we'll add some logic to getAllNewsletters
to filter out any newsletters that haven't been published yet.
Don't render newsletters with duplicate subject lines
There are some cases where I send the same newsletter multiple times, with the same subject line, to different audiences - usually because I've forgotten to include some details in the original broadcast. This results in multiple broadcasts with essentially the same content. I don't want to show these on my site, so I added some logic to deduple any newsletters that have the same subject line, displaying only the most recent one (which should be the most correct, in theory).
This is the updated code for getAllNewsletters
in convertKit.js
:
1export const getAllNewsletters = async () => {2const response = await fetch(3`https://api.convertkit.com/v3/broadcasts?api_secret=${CONVERTKIT_API_SECRET}`4);5const data = await response.json();6const { broadcasts } = data;78// get details for each individual newsletter9const newsletters = await Promise.all(10broadcasts.map((broadcast) => getNewsletter(broadcast.id))11);1213// hackish dedupe here - sometimes we publish a newsletter a _second_ time as a correction, or to a different audience.14// In those cases, they should always have the same subject. We're using subject as a way to deduplicate newsletters in that case15const dedupedBySubject = {};1617// return only newsletters that have been published, and sort by most recent to oldest18newsletters19.filter((newsletter) => !!newsletter.published_at)20.sort((a, b) => new Date(b.published_at) - new Date(a.published_at))21.forEach((newsletter) => {22// the first one we encounter in this order will be the newest, so if there's one already we don't make any changes to the object map23if (!dedupedBySubject[newsletter.subject])24dedupedBySubject[newsletter.subject] = newsletter;25});2627let nls = [];28// iterate over the map we have of subject->newsletter, and push into a fresh array, which we'll return29for (let subject in dedupedBySubject) {30nls.push(dedupedBySubject[subject]);31}3233return nls;34};
And with that, our basic implementation is complete - we've created a /newsletter
page that shows a list of all of our newsletters, with a thumbnail image for each one. We've also added a /newsletter/[subject]
page that shows the full content of a given newsletter.
Next steps
There are a few things I'd like to add to this implementation in the future:
- I'm unhappy with the bottleneck I've created for myself using
slugify
on the subject line to create the URL slug. I'd like to find a more efficient way to do this. A reasonable fallback for now might be to use theBroadcastId
as the slug instead of the subject line, but those aren't human-readable. - This implementation presents some challenges for SEO - getting newsletter entries to show up in my site's
sitemap.xml
is currently done at build time, which means that new newsletters won't show up in the sitemap until I rebuild the site. In the future, I'll either need to generate the sitemap dynamically, or implement Incremental Static Regeneration so that new newsletters are added to the sitemap as soon as they're published. - I'm pretty diligent about embedding opengraph metadata on the pages on my site, including images and excerpts from the article to use as previews on sites like twitter and LinkedIn. This implementation makes that tricky, as determining which image to use as the cover for a given newsletter is a bit of a challenge. Right now I'm using the first image within a newsletter. I'm not sure what the best solution is here, but I'm open to suggestions!
Wrap-up
I hope this post has been helpful to you, and that you've learned something new about how to use ConvertKit's API to create a custom newsletter index page for your Next.js site. I'd also love it if you checked out Tiny Improvements, my little newsletter about making better products and a better life - mikebifulco.com/newsletter.
If you have any questions, or if you've found a bug in my code, please feel free to reach out to me on Twitter @irreverentmike!