Create an Eleventy podcast feed

By Marc Littlemore
9 min read

At the end of last year I converted my DJ Cruze website from Wordpress to Eleventy. Much like this website, it's easy to deploy using Netlify. Moving to a statically generated site made it much easier to update and avoided me having to maintain a server, PHP, Wordpress versions, and so on.

For many years I created the DJ Cruze podcast show so I wanted to move over the podcast RSS feed from Wordpress to Eleventy. Creating a custom RSS feed based on my podcast episodes took a bit of work so I thought I'd share how I did it.

Eleventy supports a wide range of data sources.

For my podcast I define a JSON file which contains the metadata needed for the podcast. All of this data will be used to populate the RSS feed. You may have this information in other data files in your Eleventy site but I wanted to keep most of the data for the podcast together.

It's stored in the data directory like this: /data/podcast.json. This will expose a global data object called podcast which matches the filename of the JSON file and can be used in your template files.

{
  "title": "DJ Cruze House Music Podcast",
  "description": "DJ Cruze is in the house! Spinning funky and chunky house music since 1988. Manchester is in the house!",
  "category": "Music",
  "author": "DJ Cruze",
  "channelImage": "/images/podcasts/dj-cruze-podcast-logo-1400x1400.jpg",
  "owner": {
    "name": "Marc Littlemore",
    "email": "info@djcruze.co.uk"
  },
  "feedPath": "/podcasts/feed.xml",
  "episode": {
    "defaultDescription": "The latest episode of the DJ Cruze podcast features new and old funky and chunky house music."
  }
}

Most of the metadata properties should be obvious.

Both iTunes and Google Podcasts support the standard RSS schema but with additional tags. Google Podcasts is essentially the same as an iTunes feeds and uses the iTunes specific tags in its feed too. Make sure that your category property matches the expected iTunes categories and be careful that you also match the correct case. If you don't, it may well be rejected. In my case I'm using the "Music" category. Note that you can use sub-categories too but I'm not in my example.

In Wordpress, each of my podcasts was a new post with a specific tag. I exported these as Markdown files and added some custom frontmatter to mark it up with per-podcast metadata.

---
media:
    # The episode number
    episode: 57

    # The image file for this episode
    # You can ignore this if you don't want per-episode artwork
    image: '/images/podcasts/dj-cruze-podcast-episode-57-june-2011.jpg'

    # Your MP3 file 
    content: '/podcasts/dj-cruze-podcast-episode-57-june-2011.mp3'

    # The duration of the episode in seconds
    duration: '4230'

    # The filesize of the MP3 file in bytes
    fileSize: '67973718'

    # A per-episode description if you want it
    description: 'Another DJ set of funky house music from DJ Cruze!'

# Additional layout data here...
---

I use Eleventy's directory specific data files to mark up all podcasts with the tag podcast. The enables us to create an Eleventy collection of podcasts to iterate through.

{
  "tags": ["podcast"]
}

Our podcast feed needs some custom filters adding to the Eleventy configuration.

RSS feeds expect some dates using the RFC822 date format. There's an npm package we can install to do the conversion for us. Install it as a development dependency like this:

npm install --save-dev rfc822-date

We also need to escape some of our text to ensure any special characters are encoded correctly. I use the Lodash escape method to do this. Again, lets install this a development dependency as follows:

npm install --save-dev lodash.escape

Lastly, we need to know what the date of the last episode was so we can add a special LastBuildDate tag. This tells any clients when the podcast was last modified. By adding this, any new podcasts will update to the latest date and allow new episodes to be downloaded.

Here are the additional filters we add to our .eleventy.js configuration file. I've only added the filters we've created for the podcast feed and intentionally left out any other Eleventy configuration.

const escape = require('lodash.escape');
const rfc822Date = require('rfc822-date');

module.exports = (eleventyConfig) => {
    // RSS
    eleventyConfig.addLiquidFilter('rfc822Date', (dateValue) => {
        return rfc822Date(dateValue);
    });

    // Escape characters for XML feed
    eleventyConfig.addLiquidFilter('xmlEscape', (value) => {
        return escape(value);
    });

    // Newest date in the collection
    eleventyConfig.addFilter('collectionLastUpdatedDate', (collection) => {
        if (!collection || !collection.length) {
            throw new Error(
                'Collection is empty in collectionLastUpdatedDate filter.'
            );
        }

        return rfc822Date(
            new Date(
                Math.max(...collection.map((item) => {
                    return item.date;
                }))
            )
        );
    });

    // Rest of Eleventy config goes here...
};

Now we can create a post with a custom template in it to define our podcast URL. In this template we can iterate over our podcast collection and render each podcast episode correctly into the expected RSS XML format.

I created a podcast Liquid template file with a /podcast/feed.xml permalink. I'm excluding this file from other collections and from my sitemap. You might want to do this too.

The podcast feed is split up into the intial channel metadata which is mostly created from our podcast.json metadata. Ensure that any text is escaped correctly by passing it through the escape filter.

As I'm attempting to redirect my old Wordpress theme from an old URL to a new URL, I've also added in the itunes:new-feed-url tag as follows. You won't need this unless you are migration your podcast feed too.

<itunes:new-feed-url>{{site.url}}{{podcast.feedPath}}</itunes:new-feed-url>

The second part of the podcast feed iterates through our collections.podcast feed in reverse order. In that way, we always get the latest episode first in the RSS feed. The metadata used for each episode comes from the Markdown frontmatter that we defined earlier.

The enclosure tag defines the media file needed to play the podcast. As I host my podcasts on a different URL to the DJ Cruze site itself, I have an additional URL in my site.json data file which defines the external server to load them from. This explains the site.mediaFilesUrl URL which is prepended before the content path.

Here is the full RSS template.

---
permalink: "/podcasts/feed.xml"
eleventyExcludeFromCollections: true
sitemap:
  exclude: yes
---
<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:itunes="http://www.itunes.com/dtds/podcast-1.0.dtd" xmlns:content="http://purl.org/rss/1.0/modules/content/">
  <channel>
    <title>{{podcast.title | xmlEscape}}</title>
    <link>{{site.url}}</link>
    <language>en-us</language>
    <copyright>&#169; 2005 - {{"now" | date: "%Y"}} {{podcast.author}}</copyright>
    <lastBuildDate>{{collections.podcast | collectionLastUpdatedDate}}</lastBuildDate>
    <generator>Eleventy</generator>
    <description>{{podcast.description | xmlEscape}}</description>
    <itunes:author>{{podcast.author}} </itunes:author>
    <itunes:explicit>false</itunes:explicit>
    <itunes:type>episodic</itunes:type>
    <itunes:image href="{{podcast.channelImage | prepend: site.url}}" />
    <itunes:owner>
      <itunes:name>{{podcast.owner.name}}</itunes:name>
      <itunes:email>{{podcast.owner.email}}</itunes:email>
    </itunes:owner>
    <itunes:category text="{{podcast.category}}" />
    <itunes:new-feed-url>{{site.url}}{{podcast.feedPath}}</itunes:new-feed-url>
    
    {%- for podcastEpisode in collections.podcast reversed -%}
      <item>
        <pubDate>{{ podcastEpisode.date | rfc822Date }}</pubDate>
        <link>{{ podcastEpisode.url | prepend: site.url }}</link>

        {% if podcastEpisode.data.media.guid %}
          <guid>{{ podcastEpisode.data.media.guid }}</guid>
        {% else %}
          <guid>{{ podcastEpisode.url | prepend: site.url }}</guid>
        {% endif %}

        {% if podcastEpisode.data.media.title %}
          <title>{{podcastEpisode.data.media.title | xmlEscape}}</title>
        {% else %}
          <title>{{podcastEpisode.data.title}}</title>
        {% endif %}

        {% if podcastEpisode.data.media.description %}
          <description>{{ podcastEpisode.data.media.description | xmlEscape }}</description>
        {% else %}
          <description>{{ podcast.episode.defaultDescription | xmlEscape }}</description>
        {% endif %}
        
        <enclosure
          url="{{podcastEpisode.data.media.content | prepend: site.mediaFilesUrl}}"
          length="{{podcastEpisode.data.media.fileSize}}"
          type="audio/mpeg"
        />

        {% comment %} iTunes specific tags {% endcomment %}

        {% if podcastEpisode.data.media.episodeType %}
            <itunes:episodeType>{{podcastEpisode.data.media.episodeType}}</itunes:episodeType>
        {% else %}
            <itunes:episodeType>full</itunes:episodeType>
        {% endif %}

        {% if podcastEpisode.data.media.image %}
            <itunes:image href="{{podcastEpisode.data.media.image | prepend: site.url}}" />
        {% endif %}

        {% if podcastEpisode.data.media.episode %}
            <itunes:episode>{{podcastEpisode.data.media.episode}}</itunes:episode>
        {% endif %}

        {% if podcastEpisode.data.media.season %}
            <itunes:season>{{podcastEpisode.data.media.season}}</itunes:season>
        {% endif %}

        {% if podcastEpisode.data.media.duration %}
            <itunes:duration>{{podcastEpisode.data.media.duration}}</itunes:duration>
        {% endif %}        
      </item>
    {%- endfor -%}
  </channel>
</rss>

If you head to my DJ Cruze podcast feed, you can see all of the episodes rendered correctly.

There is some metadata which could be automatically generated from the media files. I add to the duration and file size properties to my Markdown frontmatter for each podcast episode. I do this by hand but it could easily be generated by reading the file when Eleventy builds the feed.

Here is the DJ Cruze website on GitHub. Feel free to take a look through the code and adapt it for your own use.

I hope you find this useful. Let me know if you have any questions or comments.

Marc Littlemore avatar

I'm Marc Littlemore.

I’m a Senior Software Engineering Manager who works with high performing development teams and loves to help to grow other software leaders and engineers.

0
0
0
1

Want to read more?