a large tunnel in a forest

Adding RSS Feed Content and Fixing Markdown Image Paths in Astro

May 4th, 2024

9 min read

Aug 12th, 2024

Having an RSS feed is a nice way to notify your readers when your new content has been published.

I have a yellow “Subscribe” button at the end of every blog post which points to my rss.xml file where anyone could subscribe using an RSS reader.

To do this, AstroJS makes it easy for you to create an RSS feed by using their official plugin @astrojs/rss.

However, when it comes to adding feed content, there was one little problem…

The RSS content image issue

When I first integrated it, I tried adding the full contents of my post to the RSS feed.

I realized the output image src were all using a relative path that didn’t exist because of how Astro works with local images.

Incorrect image path in RSS feed

I couldn’t find a direct fix for this initially so I put it off.

With that said, I excluded using the post content and simply used the title and description for my RSS feed.

Now, when I visit an RSS reader like Feedly and look for my website, I get a list of feeds.

It’s exactly what I want, but it looks pretty bare-bones.

Bare Feedly RSS feed item

All a user can do at this point is to follow the link to visit the blog to read it.

But what if I want users to read from their preferred source?

Wouldn’t it be nice if someone wanted to read on Medium, dev.to, or their own RSS reader?

That’s why I found the want to add the post content and begin cross-posting to other platforms so that my blog has more reach.

Cross-posting on dev.to

More recently, I wanted to cross-post over to dev.to to get more visibility on my writings.

You can connect your RSS feed to dev.to under “Settings -> Extensions”.

Add RSS feed in dev.to settings

If you want, you can specify to dev.to that you want your RSS feed to be the canonical URL. Be sure to check the box if this is your desire.

From there, you can click the “Fetch feed now” button and it will pull all your feed items into your dashboard as draft posts.

Dev.to dashboard showing draft posts

Everything works as expected but the contents of the blog weren’t included because the RSS feed item didn’t have any content that dev.to can use.

Dev.to draft post with no content

Before I get to the solution, I want to talk about how AstroJS outputs the image files.

AstroJS image optimization

Let’s understand a bit what is happening with the images in your markdown.

In this photo, you can see I have an _images folder alongside my markdowns that map to a relative path in a blog post.

Markdown file showing image relative path

If you’re using an import alias or relative path that is not the public directory in your markdowns, Astro will copy and optimize the images and place them in a static folder called _astro/ at build time.

When AstroJS is transforming your markdowns to HTML, it will then replace all the image paths with the one found in the _astro folder.

Here is an example of what that folder looks like.

Output directory _astro from image optimization

By default, AstroJS will give the file names a hash and also convert it to webp for a smaller footprint.

Pretty simple, right?

Okay, it’s time to revisit adding the RSS feed content.

Building the RSS feed content

Using the AstroJS RSS tutorial as a base, we’ll add onto it to make sure our images correctly point to a URL.

This is the basic code to compile your RSS feed.

import rss from '@astrojs/rss';
import { getCollection } from 'astro:content';
import sanitizeHtml from 'sanitize-html';
import MarkdownIt from 'markdown-it';
const parser = new MarkdownIt();

export async function GET(context) {
  const blog = await getCollection('blog');
  return rss({
    title: 'Buzz’s Blog',
    description: 'A humble Astronaut’s guide to the stars',
    site: context.site,
    items: blog.map((post) => ({
      link: `/blog/${post.slug}/`,
      // Note: this will not process components or JSX expressions in MDX files.
      content: sanitizeHtml(parser.render(post.body), {
        allowedTags: sanitizeHtml.defaults.allowedTags.concat(['img'])
      }),
      ...post.data,
    })),
  });
}

In the parser.render(post.body), we are passing in the contents of our markdown into render which includes the relative image paths.

That is why in our final output for our rss.xml, we get those incorrect URLs.

At this point, I had an idea to fix this. What if before I pass the HTML string into sanitizeHtml(), I modify the markdown or HTML image paths myself, would that work?

Let’s see…

Add dependency node-html-parser

We’re going to need an HTML parser so we can easily manipulate objects instead of strings.

Install node-html-parser.

The Node HTML parser will convert the string output from the rendered markdown from markdown-it and create a DOM-like HTML structure.

We can query against this structure like how we would use the DOM API for browsers.

Install it with your preferred package manager.

I use pnpm so that command would be pnpm add node-html-parser.

The image relative path fix

Here is my solution to this problem.

import rss from "@astrojs/rss";
import sanitizeHtml from "sanitize-html";
import MarkdownIt from "markdown-it";
import { allPosts } from "@utils/getCollection";
import { parse as htmlParser } from "node-html-parser";
import { getImage } from "astro:assets";

import type { AstroGlobal } from "astro";
import type { RSSFeedItem } from "@astrojs/rss";
const markdownParser = new MarkdownIt();

// get dynamic import of images as a map collection
const imagesGlob = import.meta.glob<{ default: ImageMetadata }>(
  "/src/content/posts/_images/**/*.{jpeg,jpg,png,gif}", // add more image formats if needed
);

export async function GET(context: AstroGlobal) {
  if (!context.site) {
    throw Error("site not set");
  }

  const feed: RSSFeedItem[] = [];

  for (const post of allPosts) {
    // convert markdown to html string
    const body = markdownParser.render(post.body);
    // convert html string to DOM-like structure
    const html = htmlParser.parse(body);
    // hold all img tags in variable images
    const images = html.querySelectorAll("img");

    for (const img of images) {
      const src = img.getAttribute("src")!;

      // Relative paths that are optimized by Astro build
      if (src.startsWith("./")) {
        // remove prefix of `./`
        const prefixRemoved = src.replace("./", "");
        // create prefix absolute path from root dir
        const imagePathPrefix = `/src/content/posts/${prefixRemoved}`;

        // call the dynamic import and return the module
        const imagePath = await imagesGlob[imagePathPrefix]?.()?.then(
          (res) => res.default,
        );

        if (imagePath) {
          const optimizedImg = await getImage({ src: imagePath });
          // set the correct path to the optimized image
          img.setAttribute(
            "src",
            context.site + optimizedImg.src.replace("/", ""),
          );
        }
      } else if (src.startsWith("/images")) {
        // images starting with `/images/` is the public dir
        img.setAttribute("src", context.site + src.replace("/", ""));
      } else {
        throw Error("src unknown");
      }
    }

    feed.push({
      title: post.data.title,
      description: post.data.description,
      author: `${post.data.author.email} (${post.data.author.name})`,
      pubDate: post.data.pubDate,
      categories: post.data.tags,
      link: `/posts/${post.slug}`,
      // sanitize the new html string with corrected image paths
      content: sanitizeHtml(html.toString(), {
        allowedTags: sanitizeHtml.defaults.allowedTags.concat(["img"]),
      }),
    });
  }

  return rss({
    title: "Billy Le | Blog",
    description:
      "My creative outlet is a reflection of the experiences I've encountered—whether in learning, facing setbacks, or achieving success—as a software developer.",
    site: context.site,
    items: feed,
    stylesheet: "/pretty-feed-v3.xsl",
    xmlns: {
      atom: "http://www.w3.org/2005/Atom",
    },
    customData: [
      "<language>en-us</language>",
      `<atom:link href="${new URL("rss.xml", context.site)}" rel="self" type="application/rss+xml" />`,
    ].join(""),
    trailingSlash: false,
  });
}

I littered the code with comments but I’ll try to break this down to make more sense of it.

The most important part is the import.meta.glob().

Thanks to Henri Fournier, from the Astro Lounge Discord Support Channel, for this tip.

You can read more about dynamically importing your images and the import.meta.glob() in more detail.

The imagesGlob variable is an object that stores keys as the paths, and the values as dynamic import functions.

const imagesGlob = {
  '/src/content/posts/_images/content-collection-in-sync/types-to-any.png': [Function: /src/content/posts/_images/content-collection-in-sync/types-to-any.png],
  '/src/content/posts/_images/creating-toc/remark-toc-md.png': [Function: /src/content/posts/_images/creating-toc/remark-toc-md.png],
  '/src/content/posts/_images/creating-toc/remark-toc-static.png': [Function: /src/content/posts/_images/creating-toc/remark-toc-static.png],
  ...
}

Each dynamic import contains the ImageMetadata object which looks like this:

export interface ImageMetadata {
    src: string;
    width: number;
    height: number;
    format: ImageInputFormat;
    orientation?: number;
}

With that ready to go, I loop over all my blog posts and convert them to HTML using the node-html-parser.

const body = markdownParser.render(post.body);
const html = htmlParser.parse(body);
const images = html.querySelectorAll("img");

From there, I loop through the images and use the dynamic imports from imagesGlob to get the correct path from the ImageMetadata.

And that looks like this:

const src = img.getAttribute("src")!;

if (src.startsWith("./")) {
  // remove prefix of `./`
  const prefixRemoved = src.replace("./", "");
  // create prefix absolute path from root dir
  const imagePathPrefix = `/src/content/posts/${prefixRemoved}`;

  // call the dynamic import and return the module
  const imagePath = await imagesGlob[imagePathPrefix]?.()?.then(
    (res) => res.default,
  );

  if (imagePath) {
    const optimizedImg = await getImage({ src: imagePath });
    // set the correct path to the optimized image
    img.setAttribute(
      "src",
      context.site + optimizedImg.src.replace("/", ""),
    );
  }
}

The rest of the else if/else statements check if the image is from my public/images directory or if it’s an unknown source, I throw an error.

Finally, I push the feed item into an array and when we sanitize our HTML, we call html.toString() which is passed into sanitizeHtml().

Check RSS output fix

Now it’s time to see if everything works.

Run the build command, I’m using pnpm build.

Check the contents of your _dist folder and look for your RSS XML file.

Corrected image path in RSS feed

🥳 Yes! It’s looking good and seems to be pointing to the path correctly.

Okay, time to push it live and test Feedly and dev.to.

Verify fix on supported platforms

If you had an RSS feed on Feedly before adding content, they won’t update as the date is encoded on their servers.

Only new content will be updated with the new RSS changes or when you change your feed URL.

Viewing from Feedly, I only have one item that correctly shows the contents of my blog post.

And the images are working!

Corrected Feedly RSS feed

That’s awesome!

Over on dev.to, I deleted all my drafts with no content.

Once I removed all my old draft posts, I fetched my updated RSS feed using the “Fetch feed now” button.

And now, I see that all the contents are there ready to be published!

Dev.to draft posts with content

Conclusion

With the RSS feed content in place, we can now fetch our blog posts on platforms that support RSS feeds.

This is great because we can benefit by sharing our blog with different platforms and give our readers a choice where they receive updates and read new content.

In this post, I’ve gone over the issue of the incorrect image path when creating an RSS feed from markdown to XML with AstroJS.

To fix this, we had to convert the markdown to HTML and modify the image src to use the correct path. That’s all thanks to the import.meta.glob(), a useful Vite utility function.

After verifying our fix, we can view places like Feedly and dev.to, to continue sharing our posts.

If you want to see how you can add a featured image per blog post for your RSS. Check out Web Reaper’s blog post on how you can do that.

Well, that’s all I can think of. Let me know what you think and if there is anything I can add.

Thanks for reading, and as always, have a good one! 😊

RSS Subscribe