Growth hack: Publish newsletters to your Next.js site with ConvertKit API

Use previous issues of your newsletter to attract new subscribers, using the ConvertKit API with your Next.js site.

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.

Note: This is a follow-up to my previous post on using ConvertKit with Remix. If your site is built using Remix.run, you'll probably want to head over there.

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:

  1. 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).
  2. Use the ConvertKit API's list-broadcasts endpoint to query for every published broadcast, to populate a newsletter index page
  3. 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.
  4. 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.

1
export const getNewsletter = async (broadcastId) => {
2
const res = await fetch(
3
`https://api.convertkit.com/v3/broadcasts/${broadcastId}?api_secret=${CONVERTKIT_API_SECRET}`
4
);
5
6
// rename "broadcast" to "newsletter" for consistency
7
const { broadcast: newsletter } = await res.json();
8
9
return newsletter;
10
};

The second is getAllNewsletters(), which returns a list of all published broadcasts. In its simplest form, this function looks like this:

1
export const getAllNewsletters = async () => {
2
const response = await fetch(
3
`https://api.convertkit.com/v3/broadcasts?api_secret=${CONVERTKIT_API_SECRET}`
4
);
5
const data = await response.json();
6
7
// rename "broadcasts" to "newsletters" for consistency
8
const { broadcasts: newsletters } = data;
9
10
return newsletters;
11
};

We are using an environment variable to supply the variable CONVERTKIT_API_SECRET - it is super important to keep this value secret, and not share it on the client-side of your app.

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.

1
import { Link } from 'next/link';
2
import { getAllNewsletters } from '../../utils/convertKit';
3
4
import slugify from '../../utils/slugify';
5
6
export const getStaticProps = async () => {
7
const newsletters = await getAllNewsletters();
8
9
return {
10
props: {
11
newsletters,
12
},
13
};
14
};
15
16
const NewsletterIndexPage = ({ newsletters }) => {
17
return (
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
};
32
33
export 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.

1
import makeSlug from 'slugify';
2
3
export const slugify = (string) => {
4
return makeSlug(string, {
5
replacement: '-',
6
remove: /[\[\],?*+~.()'"!:@]/g,
7
strict: true,
8
lower: true,
9
});
10
};
11
12
export 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):

1
import { useEffect } from 'react';
2
import {
3
broadcastTemplateParse,
4
getAllNewsletters,
5
} from '../../utils/convertKit';
6
import slugify from '../../utils/slugify';
7
8
export const getStaticPaths = async () => {
9
const newsletters = await getAllNewsletters();
10
11
return {
12
paths: newsletters.map((newsletter) => ({
13
params: {
14
subject: slugify(newsletter.subject),
15
},
16
})),
17
fallback: false,
18
};
19
};
20
21
export const getStaticProps = async (context) => {
22
const newsletters = await getAllNewsletters();
23
24
const { subject } = context.params;
25
const slug = slugify(subject);
26
27
const newsletter = newsletters.find(
28
(newsletter) => slugify(newsletter.subject) == slug
29
);
30
31
return {
32
props: {
33
newsletter,
34
},
35
};
36
};
37
38
const SingleNewsletterPage = ({ newsletter }) => {
39
const { subject } = newsletter;
40
41
return (
42
<>
43
<h1>{subject}</h1>
44
<div
45
dangerouslySetInnerHTML={{
46
__html: broadcastTemplateParse({ template: newsletter.content }),
47
}}
48
/>
49
</>
50
);
51
};
52
53
export 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:

1
import { Liquid } from 'liquidjs';
2
const engine = new Liquid();
3
4
export const broadcastTemplateParse = ({ template, data }) => {
5
const t = template.replaceAll('&quot;', '"');
6
const res = engine.parseAndRenderSync(t, data);
7
return res;
8
};
9
10
export const newsletterHasValidThumbnail = (newsletter) => {
11
const { thumbnail_url: thumbnailUrl } = newsletter;
12
if (!thumbnailUrl) return false;
13
if (thumbnailUrl.startsWith('https://functions-js.convertkit.com/icons'))
14
return false;
15
return 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:

1
curl 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:

1
export const getAllNewsletters = async () => {
2
const response = await fetch(
3
`https://api.convertkit.com/v3/broadcasts?api_secret=${CONVERTKIT_API_SECRET}`
4
);
5
const data = await response.json();
6
const { broadcasts } = data;
7
8
// get details for each individual newsletter
9
const newsletters = await Promise.all(
10
broadcasts.map((broadcast) => getNewsletter(broadcast.id))
11
);
12
13
// 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 case
15
const dedupedBySubject = {};
16
17
// return only newsletters that have been published, and sort by most recent to oldest
18
newsletters
19
.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 map
23
if (!dedupedBySubject[newsletter.subject])
24
dedupedBySubject[newsletter.subject] = newsletter;
25
});
26
27
let nls = [];
28
// iterate over the map we have of subject->newsletter, and push into a fresh array, which we'll return
29
for (let subject in dedupedBySubject) {
30
nls.push(dedupedBySubject[subject]);
31
}
32
33
return 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 the BroadcastId 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!

Mike Bifulco headshot

Subscribe to Tiny Improvements

A newsletter for product builders, startup founders, and indiehackers, who design with intention, and my thoughts on living a life you love in a busy world.

    Typically once a week, straight from me to you. 😘 Unsubscribe anytime.


    Get in touch to → Sponsor Tiny Improvements

    ***

    More great resources

    Articles about React.jsArticles about Remix.runArticles about Next.jsArticles for developersArticles for JavaScript developersArticles about CSSArticles about User Experience (UX)Articles about tools I useArticles about productivityBrowse all topics →
    © 2019-2023 Mike Bifulco

    Get in touch to → Sponsor Tiny Improvements

    Disclaimer: 👋🏽 Hi there. I work as a co-founder & CTO at Craftwork. These are my opinions, and not necessarily the views of my employer.
    Built with Next. Source code on GitHub.